Compare commits

..

34 commits

Author SHA1 Message Date
alexjg
cb409b6ffe
docs: timestamp -> time in automerge.change examples (#548) 2023-03-09 18:10:23 +00:00
Conrad Irwin
b34b46fa16
smaller automerge c (#545)
* Fix automerge-c tests on mac

* Generate significantly smaller automerge-c builds

This cuts the size of libautomerge_core.a from 25Mb to 1.6Mb on macOS
and 53Mb to 2.7Mb on Linux.

As a side-effect of setting codegen-units = 1 for all release builds the
optimized wasm files are also 100kb smaller.
2023-03-09 15:09:43 +00:00
Conrad Irwin
7b747b8341
Error instead of corrupt large op counters (#543)
Since b78211ca6, OpIds have been silently truncated to 2**32. This
causes corruption in the case the op id overflows.

This change converts the silent error to a panic, and guards against the
panic on the codepath found by the fuzzer.
2023-03-07 16:49:04 +00:00
Conrad Irwin
2c1970f664
Fix panic on invalid action (#541)
We make the validation on parsing operations in the encoded changes stricter to avoid a possible panic when applying changes.
2023-03-04 12:09:08 +00:00
christine betts
63b761c0d1
Suppress clippy warning in parse.rs + bump toolchain (#542)
* Fix rust error in parse.rs
* Bump toolchain to 1.67.0
2023-03-03 22:42:40 +00:00
Conrad Irwin
44fa7ac416
Don't panic on missing deps of change chunks (#538)
* Fix doubly-reported ops in load of change chunks

Since c3c04128f5, observers have been
called twice when calling Automerge::load() with change chunks.

* Better handle change chunks with missing deps

Before this change Automerge::load would panic if you passed a change
chunk that was missing a dependency, or multiple change chunks not in
strict dependency order. After this change these cases will error
instead.
2023-02-27 20:12:09 +00:00
Jason Kankiewicz
8de2fa9bd4
C API 2 (#530)
The AMvalue union, AMlistItem struct, AMmapItem struct, and AMobjItem struct are gone, replaced by the AMitem struct.

The AMchangeHashes, AMchanges, AMlistItems, AMmapItems, AMobjItems, AMstrs, and AMsyncHaves iterators are gone, replaced by the AMitems iterator.

The AMitem struct is opaque, getting and setting values is now achieved exclusively through function calls.

The AMitemsNext(), AMitemsPrev(), and AMresultItem() functions return a pointer to an AMitem struct so you ultimately get the same thing whether you're iterating over a sequence or calling AMmapGet() or AMlistGet().

Calling AMitemResult() on an AMitem struct will produce a new AMresult struct referencing its storage so now the AMresult struct for an iterator can be subsequently freed without affecting the AMitem structs that were filtered out of it.

The storage for a set of AMitem structs can be recombined into a single AMresult struct by passing pointers to their corresponding AMresult structs to AMresultCat().

For C/C++ programmers, I've added AMstrCmp(), AMstrdup(), AM{idxType,objType,status,valType}ToString() and AM{idxType,objType,status,valType}FromString(). It's also now possible to pass arbitrary parameters through AMstack{Item,Items,Result}() to a callback function.
2023-02-25 18:47:00 +00:00
Philip Schatz
407faefa6e
A few setup fixes (#529)
* include deno in dependencies

* install javascript dependencies

* remove redundant operation
2023-02-15 09:23:02 +00:00
Alex Good
1425af43cd @automerge/automerge@2.0.2 2023-02-15 00:06:23 +00:00
Alex Good
c92d042c87 @automerge/automerge-wasm@0.1.24 and @automerge/automerge@2.0.2-alpha.2 2023-02-14 17:59:23 +00:00
Alex Good
9271b20cf5 Correct logic when skip = B and fix formatting
A few tests were failing which exposed the fact that if skip is `B` (the
out factor of the OpTree) then we set `skip = None` and this causes us
to attempt to return `Skip` in a non root node. I ported the failing
test from JS to Rust and fixed the problem.

I also fixed the formatting issues.
2023-02-14 17:21:59 +00:00
Orion Henry
5e82dbc3c8 rework how skip works to push the logic into node 2023-02-14 17:21:59 +00:00
Conrad Irwin
2cd7427f35 Use our leb128 parser for values
This ensures that values in automerge documents are encoded correctly,
and that no extra data is smuggled in any LEB fields.
2023-02-09 15:46:22 +00:00
Alex Good
11f063cbfe
Remove nightly from CI 2023-02-09 11:06:24 +00:00
Alex Good
a24d536d16 Move automerge::SequenceTree to automerge_wasm::SequenceTree
The `SequenceTree` is only ever used in `automerge_wasm` so move it
there.
2023-02-05 11:08:33 +00:00
Alex Good
c5fde2802f @automerge/automerge-wasm@0.1.24 and @automerge/automerge@2.0.2-alpha.1 2023-02-03 16:31:46 +00:00
Alex Good
13a775ed9a Speed up loading by generating clocks on demand
Context: currently we store a mapping from ChangeHash -> Clock, where
`Clock` is the set of (ActorId, (Sequence number, max Op)) pairs derived
from the given change and it's dependencies. This clock is used to
determine what operations are visible at a given set of heads.

Problem: populating this mapping for documents with large histories
containing many actors can be very slow as for each change we have to
allocate and merge a bunch of hashmaps.

Solution: instead of creating the clocks on load, create an adjacency
list based representation of the change graph and then derive the clock
from this graph when it is needed. Traversing even large graphs is still
almost as fast as looking up the clock in a hashmap.
2023-02-03 16:15:15 +00:00
Alex Good
1e33c9d9e0 Use Automerge::load instead of load_incremental if empty
Problem: when running the sync protocol for a new document the API
requires that the user create an empty document and then call
`receive_sync_message` on that document. This results in the OpObserver
for the new document being called with every single op in the document
history. For documents with a large history this can be extremely time
consuming, but the OpObserver doesn't need to know about all the hidden
states.

Solution: Modify `Automerge::load_with` and
`Automerge::apply_changes_with` to check if the document is empty before
applying changes. If the document _is_ empty then we don't call the
observer for every change, but instead use
`automerge::observe_current_state` to notify the observer of the new
state once all the changes have been applied.
2023-02-03 10:01:12 +00:00
Alex Good
c3c04128f5 Only observe the current state on load
Problem: When loading a document whilst passing an `OpObserver` we call
the OpObserver for every change in the loaded document. This slows down
the loading process for two reasons: 1) we have to make a call to the
observer for every op 2) we cannot just stream the ops into the OpSet in
topological order but must instead buffer them to pass to the observer.

Solution: Construct the OpSet first, then only traverse the visible ops
in the OpSet, calling the observer. For documents with a deep history
this results in vastly fewer calls to the observer and also allows us to
construct the OpSet much more quickly. It is slightly different
semantically because the observer never gets notified of changes which
are not visible, but that shouldn't matter to most observers.
2023-02-03 10:01:12 +00:00
Alex Good
da55dfac7a refactor: make fields of Automerge private
The fields of `automerge::Automerge` were crate public, which made it
hard to change the structure of `Automerge` with confidence. Make all
fields private and put them behind accessors where necessary to allow
for easy internal changes.
2023-02-03 10:01:12 +00:00
alexjg
9195e9cb76
Fix deny errors (#518)
* Ignore deny errors on duplicate windows-sys

* Delete spurious lockfile in automerge-cli
2023-02-02 15:02:53 +00:00
dependabot[bot]
f8d5a8ea98
Bump json5 from 1.0.1 to 1.0.2 in /javascript/examples/create-react-app (#487)
Bumps [json5](https://github.com/json5/json5) from 1.0.1 to 1.0.2. in javascript/examples/create-react-app
2023-02-01 09:15:54 +00:00
alexjg
2a9652e642
typescript: Hide API type and make SyncState opaque (#514) 2023-02-01 09:15:00 +00:00
Conrad Irwin
a6959e70e8
More robust leb128 parsing (#515)
Before this change i64 decoding did not work for negative numbers (not a
real problem because it is only used for the timestamp of a change),
and both u64 and i64 would allow overlong LEB encodings.
2023-01-31 17:54:54 +00:00
alexjg
de5af2fffa
automerge-rs 0.3.0 and automerge-test 0.2.0 (#512) 2023-01-30 19:58:35 +00:00
alexjg
08801ab580
automerge-rs: Introduce ReadDoc and SyncDoc traits and add documentation (#511)
The Rust API has so far grown somewhat organically driven by the needs of the
javascript implementation. This has led to an API which is quite awkward and
unfamiliar to Rust programmers. Additionally there is no documentation to speak
of. This commit is the first movement towards cleaning things up a bit. We touch
a lot of files but the changes are all very mechanical. We introduce a few
traits to abstract over the common operations between `Automerge` and
`AutoCommit`, and add a whole bunch of documentation.

* Add a `ReadDoc` trait to describe methods which read value from a document.
  make `Transactable` extend `ReadDoc`
* Add a `SyncDoc` trait to describe methods necessary for synchronizing
  documents.
* Put the `SyncDoc` implementation for `AutoCommit` behind `AutoCommit::sync` to
  ensure that any open transactions are closed before taking part in the sync
  protocol
* Split `OpObserver` into two traits: `OpObserver` + `BranchableObserver`.
  `BranchableObserver` captures the methods which are only needed for observing
  transactions.
* Add a whole bunch of documentation.

The main changes Rust users will need to make is:

* Import the `ReadDoc` trait wherever you are using the methods which have been
  moved to it. Optionally change concrete paramters on functions to `ReadDoc`
  constraints.
* Likewise import the `SyncDoc` trait wherever you are doing synchronisation
  work
* If you are using the `AutoCommit::*_sync_message` methods you will need to add
  a call to `AutoCommit::sync()` first. E.g. `doc.generate_sync_message` becomes
  `doc.sync().generate_sync_message`
* If you have an implementation of `OpObserver` which you are using in an
  `AutoCommit` then split it into an implementation of `OpObserver` and
  `BranchableObserver`
2023-01-30 19:37:03 +00:00
alexjg
89a0866272
@automerge/automerge@2.0.1 (#510) 2023-01-28 21:22:45 +00:00
Alex Good
9b6a3c8691
Update README 2023-01-28 09:32:21 +00:00
alexjg
58a7a06b75
@automerge/automerge-wasm@0.1.23 and @automerge/automerge@2.0.1-alpha.6 (#509) 2023-01-27 20:27:11 +00:00
alexjg
f428fe0169
Improve typescript types (#508) 2023-01-27 17:23:13 +00:00
Conrad Irwin
931ee7e77b
Add Fuzz Testing (#498)
* Add fuzz testing for document load

* Fix fuzz crashers and add to test suite
2023-01-25 16:03:05 +00:00
alexjg
819767cc33
fix: use saturating_sub when updating cached text width (#505)
Problem: In `automerge::query::Index::change_vis` we use `-=` to
subtract the width of an operation which is being hidden from the text
widths which we store on the index of each node in the optree. This
index represents the width of all the visible text operations in this
node and below. This was causing an integer underflow error when
encountering some list operations. More specifically, when a
`ScalarValue::Str` in a list was made invisible by a later operation
which contained a _shorter_ string, the width subtracted from the indexed
text widths could be longer than the current index.

Solution: use `saturating_sub` instead. This is technically papering
over the problem because really the width should never go below zero,
but the text widths are only relevant for text objects where the
existing logic works as advertised because we don't have a `set`
operation for text indices. A more robust solution would be to track the
type of the Index (and consequently of the `OpTree`) at the type level,
but time is limited and problems are infinite.

Also, add a lengthy description of the reason we are using
`saturating_sub` so that when I read it in about a month I don't have
to redo the painful debugging process that got me to this commit.
2023-01-23 19:19:55 +00:00
Alex Currie-Clark
78adbc4ff9
Update patch types (#499)
* Update `Patch` types

* Clarify that the splice patch applies to text

* Add Splice patch type to exports

* Add new patches to javascript
2023-01-23 17:02:02 +00:00
Andrew Jeffery
1f7b109dcd
Add From<SmolStr> for ScalarValue::Str (#506) 2023-01-23 17:01:41 +00:00
196 changed files with 13772 additions and 11328 deletions

View file

@ -2,10 +2,10 @@ name: CI
on: on:
push: push:
branches: branches:
- main - main
pull_request: pull_request:
branches: branches:
- main - main
jobs: jobs:
fmt: fmt:
runs-on: ubuntu-latest runs-on: ubuntu-latest
@ -14,7 +14,7 @@ jobs:
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
with: with:
profile: minimal profile: minimal
toolchain: 1.66.0 toolchain: 1.67.0
default: true default: true
components: rustfmt components: rustfmt
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
@ -28,7 +28,7 @@ jobs:
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
with: with:
profile: minimal profile: minimal
toolchain: 1.66.0 toolchain: 1.67.0
default: true default: true
components: clippy components: clippy
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
@ -42,7 +42,7 @@ jobs:
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
with: with:
profile: minimal profile: minimal
toolchain: 1.66.0 toolchain: 1.67.0
default: true default: true
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
- name: Build rust docs - name: Build rust docs
@ -118,7 +118,7 @@ jobs:
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
with: with:
profile: minimal profile: minimal
toolchain: 1.66.0 toolchain: nightly-2023-01-26
default: true default: true
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
- name: Install CMocka - name: Install CMocka
@ -127,6 +127,8 @@ jobs:
uses: jwlawson/actions-setup-cmake@v1.12 uses: jwlawson/actions-setup-cmake@v1.12
with: with:
cmake-version: latest cmake-version: latest
- name: Install rust-src
run: rustup component add rust-src
- name: Build and test C bindings - name: Build and test C bindings
run: ./scripts/ci/cmake-build Release Static run: ./scripts/ci/cmake-build Release Static
shell: bash shell: bash
@ -136,9 +138,7 @@ jobs:
strategy: strategy:
matrix: matrix:
toolchain: toolchain:
- 1.66.0 - 1.67.0
- nightly
continue-on-error: ${{ matrix.toolchain == 'nightly' }}
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
@ -157,7 +157,7 @@ jobs:
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
with: with:
profile: minimal profile: minimal
toolchain: 1.66.0 toolchain: 1.67.0
default: true default: true
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
- run: ./scripts/ci/build-test - run: ./scripts/ci/build-test
@ -170,7 +170,7 @@ jobs:
- uses: actions-rs/toolchain@v1 - uses: actions-rs/toolchain@v1
with: with:
profile: minimal profile: minimal
toolchain: 1.66.0 toolchain: 1.67.0
default: true default: true
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
- run: ./scripts/ci/build-test - run: ./scripts/ci/build-test

View file

@ -42,9 +42,10 @@ In general we try and respect semver.
### JavaScript ### JavaScript
An alpha release of the javascript package is currently available as A stable release of the javascript package is currently available as
`@automerge/automerge@2.0.0-alpha.n` where `n` is an integer. We are gathering `@automerge/automerge@2.0.0` where. pre-release verisions of the `2.0.1` are
feedback on the API and looking to release a `2.0.0` in the next few weeks. available as `2.0.1-alpha.n`. `2.0.1*` packages are also available for Deno at
https://deno.land/x/automerge
### Rust ### Rust
@ -52,7 +53,9 @@ The rust codebase is currently oriented around producing a performant backend
for the Javascript wrapper and as such the API for Rust code is low level and for the Javascript wrapper and as such the API for Rust code is low level and
not well documented. We will be returning to this over the next few months but not well documented. We will be returning to this over the next few months but
for now you will need to be comfortable reading the tests and asking questions for now you will need to be comfortable reading the tests and asking questions
to figure out how to use it. to figure out how to use it. If you are looking to build rust applications which
use automerge you may want to look into
[autosurgeon](https://github.com/alexjg/autosurgeon)
## Repository Organisation ## Repository Organisation
@ -109,9 +112,16 @@ brew install cmake node cmocka
# install yarn # install yarn
npm install --global yarn npm install --global yarn
# install javascript dependencies
yarn --cwd ./javascript
# install rust dependencies # install rust dependencies
cargo install wasm-bindgen-cli wasm-opt cargo-deny cargo install wasm-bindgen-cli wasm-opt cargo-deny
# get nightly rust to produce optimized automerge-c builds
rustup toolchain install nightly
rustup component add rust-src --toolchain nightly
# add wasm target in addition to current architecture # add wasm target in addition to current architecture
rustup target add wasm32-unknown-unknown rustup target add wasm32-unknown-unknown

View file

@ -54,6 +54,7 @@
nodejs nodejs
yarn yarn
deno
# c deps # c deps
cmake cmake

View file

@ -3,4 +3,13 @@ module.exports = {
parser: "@typescript-eslint/parser", parser: "@typescript-eslint/parser",
plugins: ["@typescript-eslint"], plugins: ["@typescript-eslint"],
extends: ["eslint:recommended", "plugin:@typescript-eslint/recommended"], extends: ["eslint:recommended", "plugin:@typescript-eslint/recommended"],
rules: {
"@typescript-eslint/no-unused-vars": [
"error",
{
argsIgnorePattern: "^_",
varsIgnorePattern: "^_",
},
],
},
} }

View file

@ -5845,9 +5845,9 @@ json-stable-stringify-without-jsonify@^1.0.1:
integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==
json5@^1.0.1: json5@^1.0.1:
version "1.0.1" version "1.0.2"
resolved "http://localhost:4873/json5/-/json5-1.0.1.tgz#779fb0018604fa854eacbf6252180d83543e3dbe" resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593"
integrity sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow== integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==
dependencies: dependencies:
minimist "^1.2.0" minimist "^1.2.0"
@ -6165,9 +6165,9 @@ minimatch@^5.0.1:
brace-expansion "^2.0.1" brace-expansion "^2.0.1"
minimist@^1.2.0, minimist@^1.2.6: minimist@^1.2.0, minimist@^1.2.6:
version "1.2.6" version "1.2.7"
resolved "http://localhost:4873/minimist/-/minimist-1.2.6.tgz#8637a5b759ea0d6e98702cfb3a9283323c93af44" resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.7.tgz#daa1c4d91f507390437c6a8bc01078e7000c4d18"
integrity sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q== integrity sha512-bzfL1YUZsP41gmu/qjrEk0Q6i2ix/cVeAhbCbqH9u3zYutS1cLg00qhrD0M2MVdCcx4Sc0UpP2eBWo9rotpq6g==
mkdirp@~0.5.1: mkdirp@~0.5.1:
version "0.5.6" version "0.5.6"

View file

@ -4,7 +4,7 @@
"Orion Henry <orion@inkandswitch.com>", "Orion Henry <orion@inkandswitch.com>",
"Martin Kleppmann" "Martin Kleppmann"
], ],
"version": "2.0.1-alpha.5", "version": "2.0.2",
"description": "Javascript implementation of automerge, backed by @automerge/automerge-wasm", "description": "Javascript implementation of automerge, backed by @automerge/automerge-wasm",
"homepage": "https://github.com/automerge/automerge-rs/tree/main/wrappers/javascript", "homepage": "https://github.com/automerge/automerge-rs/tree/main/wrappers/javascript",
"repository": "github:automerge/automerge-rs", "repository": "github:automerge/automerge-rs",
@ -47,7 +47,7 @@
"typescript": "^4.9.4" "typescript": "^4.9.4"
}, },
"dependencies": { "dependencies": {
"@automerge/automerge-wasm": "0.1.22", "@automerge/automerge-wasm": "0.1.25",
"uuid": "^9.0.0" "uuid": "^9.0.0"
} }
} }

100
javascript/src/conflicts.ts Normal file
View file

@ -0,0 +1,100 @@
import { Counter, type AutomergeValue } from "./types"
import { Text } from "./text"
import { type AutomergeValue as UnstableAutomergeValue } from "./unstable_types"
import { type Target, Text1Target, Text2Target } from "./proxies"
import { mapProxy, listProxy, ValueType } from "./proxies"
import type { Prop, ObjID } from "@automerge/automerge-wasm"
import { Automerge } from "@automerge/automerge-wasm"
export type ConflictsF<T extends Target> = { [key: string]: ValueType<T> }
export type Conflicts = ConflictsF<Text1Target>
export type UnstableConflicts = ConflictsF<Text2Target>
export function stableConflictAt(
context: Automerge,
objectId: ObjID,
prop: Prop
): Conflicts | undefined {
return conflictAt<Text1Target>(
context,
objectId,
prop,
true,
(context: Automerge, conflictId: ObjID): AutomergeValue => {
return new Text(context.text(conflictId))
}
)
}
export function unstableConflictAt(
context: Automerge,
objectId: ObjID,
prop: Prop
): UnstableConflicts | undefined {
return conflictAt<Text2Target>(
context,
objectId,
prop,
true,
(context: Automerge, conflictId: ObjID): UnstableAutomergeValue => {
return context.text(conflictId)
}
)
}
function conflictAt<T extends Target>(
context: Automerge,
objectId: ObjID,
prop: Prop,
textV2: boolean,
handleText: (a: Automerge, conflictId: ObjID) => ValueType<T>
): ConflictsF<T> | undefined {
const values = context.getAll(objectId, prop)
if (values.length <= 1) {
return
}
const result: ConflictsF<T> = {}
for (const fullVal of values) {
switch (fullVal[0]) {
case "map":
result[fullVal[1]] = mapProxy<T>(
context,
fullVal[1],
textV2,
[prop],
true
)
break
case "list":
result[fullVal[1]] = listProxy<T>(
context,
fullVal[1],
textV2,
[prop],
true
)
break
case "text":
result[fullVal[1]] = handleText(context, fullVal[1] as ObjID)
break
case "str":
case "uint":
case "int":
case "f64":
case "boolean":
case "bytes":
case "null":
result[fullVal[2]] = fullVal[1] as ValueType<T>
break
case "counter":
result[fullVal[2]] = new Counter(fullVal[1]) as ValueType<T>
break
case "timestamp":
result[fullVal[2]] = new Date(fullVal[1]) as ValueType<T>
break
default:
throw RangeError(`datatype ${fullVal[0]} unimplemented`)
}
}
return result
}

View file

@ -100,7 +100,7 @@ export function getWriteableCounter(
path: Prop[], path: Prop[],
objectId: ObjID, objectId: ObjID,
key: Prop key: Prop
) { ): WriteableCounter {
return new WriteableCounter(value, context, path, objectId, key) return new WriteableCounter(value, context, path, objectId, key)
} }

View file

@ -14,6 +14,7 @@ export type { ChangeToEncode } from "@automerge/automerge-wasm"
export function UseApi(api: API) { export function UseApi(api: API) {
for (const k in api) { for (const k in api) {
// eslint-disable-next-line @typescript-eslint/no-extra-semi,@typescript-eslint/no-explicit-any
;(ApiHandler as any)[k] = (api as any)[k] ;(ApiHandler as any)[k] = (api as any)[k]
} }
} }

View file

@ -1,3 +1,4 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Text } from "./text" import { Text } from "./text"
import { import {
Automerge, Automerge,
@ -6,13 +7,12 @@ import {
type Prop, type Prop,
} from "@automerge/automerge-wasm" } from "@automerge/automerge-wasm"
import type { import type { AutomergeValue, ScalarValue, MapValue, ListValue } from "./types"
AutomergeValue, import {
ScalarValue, type AutomergeValue as UnstableAutomergeValue,
MapValue, MapValue as UnstableMapValue,
ListValue, ListValue as UnstableListValue,
TextValue, } from "./unstable_types"
} from "./types"
import { Counter, getWriteableCounter } from "./counter" import { Counter, getWriteableCounter } from "./counter"
import { import {
STATE, STATE,
@ -26,19 +26,38 @@ import {
} from "./constants" } from "./constants"
import { RawString } from "./raw_string" import { RawString } from "./raw_string"
type Target = { type TargetCommon = {
context: Automerge context: Automerge
objectId: ObjID objectId: ObjID
path: Array<Prop> path: Array<Prop>
readonly: boolean readonly: boolean
heads?: Array<string> heads?: Array<string>
cache: {} cache: object
trace?: any trace?: any
frozen: boolean frozen: boolean
textV2: boolean
} }
function parseListIndex(key) { export type Text2Target = TargetCommon & { textV2: true }
export type Text1Target = TargetCommon & { textV2: false }
export type Target = Text1Target | Text2Target
export type ValueType<T extends Target> = T extends Text2Target
? UnstableAutomergeValue
: T extends Text1Target
? AutomergeValue
: never
type MapValueType<T extends Target> = T extends Text2Target
? UnstableMapValue
: T extends Text1Target
? MapValue
: never
type ListValueType<T extends Target> = T extends Text2Target
? UnstableListValue
: T extends Text1Target
? ListValue
: never
function parseListIndex(key: any) {
if (typeof key === "string" && /^[0-9]+$/.test(key)) key = parseInt(key, 10) if (typeof key === "string" && /^[0-9]+$/.test(key)) key = parseInt(key, 10)
if (typeof key !== "number") { if (typeof key !== "number") {
return key return key
@ -49,7 +68,10 @@ function parseListIndex(key) {
return key return key
} }
function valueAt(target: Target, prop: Prop): AutomergeValue | undefined { function valueAt<T extends Target>(
target: T,
prop: Prop
): ValueType<T> | undefined {
const { context, objectId, path, readonly, heads, textV2 } = target const { context, objectId, path, readonly, heads, textV2 } = target
const value = context.getWithType(objectId, prop, heads) const value = context.getWithType(objectId, prop, heads)
if (value === null) { if (value === null) {
@ -61,7 +83,7 @@ function valueAt(target: Target, prop: Prop): AutomergeValue | undefined {
case undefined: case undefined:
return return
case "map": case "map":
return mapProxy( return mapProxy<T>(
context, context,
val as ObjID, val as ObjID,
textV2, textV2,
@ -70,7 +92,7 @@ function valueAt(target: Target, prop: Prop): AutomergeValue | undefined {
heads heads
) )
case "list": case "list":
return listProxy( return listProxy<T>(
context, context,
val as ObjID, val as ObjID,
textV2, textV2,
@ -80,7 +102,7 @@ function valueAt(target: Target, prop: Prop): AutomergeValue | undefined {
) )
case "text": case "text":
if (textV2) { if (textV2) {
return context.text(val as ObjID, heads) return context.text(val as ObjID, heads) as ValueType<T>
} else { } else {
return textProxy( return textProxy(
context, context,
@ -88,29 +110,36 @@ function valueAt(target: Target, prop: Prop): AutomergeValue | undefined {
[...path, prop], [...path, prop],
readonly, readonly,
heads heads
) ) as unknown as ValueType<T>
} }
case "str": case "str":
return val return val as ValueType<T>
case "uint": case "uint":
return val return val as ValueType<T>
case "int": case "int":
return val return val as ValueType<T>
case "f64": case "f64":
return val return val as ValueType<T>
case "boolean": case "boolean":
return val return val as ValueType<T>
case "null": case "null":
return null return null as ValueType<T>
case "bytes": case "bytes":
return val return val as ValueType<T>
case "timestamp": case "timestamp":
return val return val as ValueType<T>
case "counter": { case "counter": {
if (readonly) { if (readonly) {
return new Counter(val as number) return new Counter(val as number) as ValueType<T>
} else { } else {
return getWriteableCounter(val as number, context, path, objectId, prop) const counter: Counter = getWriteableCounter(
val as number,
context,
path,
objectId,
prop
)
return counter as ValueType<T>
} }
} }
default: default:
@ -118,7 +147,21 @@ function valueAt(target: Target, prop: Prop): AutomergeValue | undefined {
} }
} }
function import_value(value: any, textV2: boolean) { type ImportedValue =
| [null, "null"]
| [number, "uint"]
| [number, "int"]
| [number, "f64"]
| [number, "counter"]
| [number, "timestamp"]
| [string, "str"]
| [Text | string, "text"]
| [Uint8Array, "bytes"]
| [Array<any>, "list"]
| [Record<string, any>, "map"]
| [boolean, "boolean"]
function import_value(value: any, textV2: boolean): ImportedValue {
switch (typeof value) { switch (typeof value) {
case "object": case "object":
if (value == null) { if (value == null) {
@ -170,7 +213,10 @@ function import_value(value: any, textV2: boolean) {
} }
const MapHandler = { const MapHandler = {
get(target: Target, key): AutomergeValue | { handle: Automerge } { get<T extends Target>(
target: T,
key: any
): ValueType<T> | ObjID | boolean | { handle: Automerge } {
const { context, objectId, cache } = target const { context, objectId, cache } = target
if (key === Symbol.toStringTag) { if (key === Symbol.toStringTag) {
return target[Symbol.toStringTag] return target[Symbol.toStringTag]
@ -185,7 +231,7 @@ const MapHandler = {
return cache[key] return cache[key]
}, },
set(target: Target, key, val) { set(target: Target, key: any, val: any) {
const { context, objectId, path, readonly, frozen, textV2 } = target const { context, objectId, path, readonly, frozen, textV2 } = target
target.cache = {} // reset cache on set target.cache = {} // reset cache on set
if (val && val[OBJECT_ID]) { if (val && val[OBJECT_ID]) {
@ -221,8 +267,10 @@ const MapHandler = {
} }
case "text": { case "text": {
if (textV2) { if (textV2) {
assertString(value)
context.putObject(objectId, key, value) context.putObject(objectId, key, value)
} else { } else {
assertText(value)
const text = context.putObject(objectId, key, "") const text = context.putObject(objectId, key, "")
const proxyText = textProxy(context, text, [...path, key], readonly) const proxyText = textProxy(context, text, [...path, key], readonly)
for (let i = 0; i < value.length; i++) { for (let i = 0; i < value.length; i++) {
@ -251,7 +299,7 @@ const MapHandler = {
return true return true
}, },
deleteProperty(target: Target, key) { deleteProperty(target: Target, key: any) {
const { context, objectId, readonly } = target const { context, objectId, readonly } = target
target.cache = {} // reset cache on delete target.cache = {} // reset cache on delete
if (readonly) { if (readonly) {
@ -261,12 +309,12 @@ const MapHandler = {
return true return true
}, },
has(target: Target, key) { has(target: Target, key: any) {
const value = this.get(target, key) const value = this.get(target, key)
return value !== undefined return value !== undefined
}, },
getOwnPropertyDescriptor(target: Target, key) { getOwnPropertyDescriptor(target: Target, key: any) {
// const { context, objectId } = target // const { context, objectId } = target
const value = this.get(target, key) const value = this.get(target, key)
if (typeof value !== "undefined") { if (typeof value !== "undefined") {
@ -287,11 +335,20 @@ const MapHandler = {
} }
const ListHandler = { const ListHandler = {
get(target: Target, index) { get<T extends Target>(
target: T,
index: any
):
| ValueType<T>
| boolean
| ObjID
| { handle: Automerge }
| number
| ((_: any) => boolean) {
const { context, objectId, heads } = target const { context, objectId, heads } = target
index = parseListIndex(index) index = parseListIndex(index)
if (index === Symbol.hasInstance) { if (index === Symbol.hasInstance) {
return instance => { return (instance: any) => {
return Array.isArray(instance) return Array.isArray(instance)
} }
} }
@ -304,13 +361,13 @@ const ListHandler = {
if (index === STATE) return { handle: context } if (index === STATE) return { handle: context }
if (index === "length") return context.length(objectId, heads) if (index === "length") return context.length(objectId, heads)
if (typeof index === "number") { if (typeof index === "number") {
return valueAt(target, index) return valueAt(target, index) as ValueType<T>
} else { } else {
return listMethods(target)[index] return listMethods(target)[index]
} }
}, },
set(target: Target, index, val) { set(target: Target, index: any, val: any) {
const { context, objectId, path, readonly, frozen, textV2 } = target const { context, objectId, path, readonly, frozen, textV2 } = target
index = parseListIndex(index) index = parseListIndex(index)
if (val && val[OBJECT_ID]) { if (val && val[OBJECT_ID]) {
@ -334,7 +391,7 @@ const ListHandler = {
} }
switch (datatype) { switch (datatype) {
case "list": { case "list": {
let list let list: ObjID
if (index >= context.length(objectId)) { if (index >= context.length(objectId)) {
list = context.insertObject(objectId, index, []) list = context.insertObject(objectId, index, [])
} else { } else {
@ -352,13 +409,15 @@ const ListHandler = {
} }
case "text": { case "text": {
if (textV2) { if (textV2) {
assertString(value)
if (index >= context.length(objectId)) { if (index >= context.length(objectId)) {
context.insertObject(objectId, index, value) context.insertObject(objectId, index, value)
} else { } else {
context.putObject(objectId, index, value) context.putObject(objectId, index, value)
} }
} else { } else {
let text let text: ObjID
assertText(value)
if (index >= context.length(objectId)) { if (index >= context.length(objectId)) {
text = context.insertObject(objectId, index, "") text = context.insertObject(objectId, index, "")
} else { } else {
@ -370,7 +429,7 @@ const ListHandler = {
break break
} }
case "map": { case "map": {
let map let map: ObjID
if (index >= context.length(objectId)) { if (index >= context.length(objectId)) {
map = context.insertObject(objectId, index, {}) map = context.insertObject(objectId, index, {})
} else { } else {
@ -398,7 +457,7 @@ const ListHandler = {
return true return true
}, },
deleteProperty(target: Target, index) { deleteProperty(target: Target, index: any) {
const { context, objectId } = target const { context, objectId } = target
index = parseListIndex(index) index = parseListIndex(index)
const elem = context.get(objectId, index) const elem = context.get(objectId, index)
@ -411,7 +470,7 @@ const ListHandler = {
return true return true
}, },
has(target: Target, index) { has(target: Target, index: any) {
const { context, objectId, heads } = target const { context, objectId, heads } = target
index = parseListIndex(index) index = parseListIndex(index)
if (typeof index === "number") { if (typeof index === "number") {
@ -420,7 +479,7 @@ const ListHandler = {
return index === "length" return index === "length"
}, },
getOwnPropertyDescriptor(target: Target, index) { getOwnPropertyDescriptor(target: Target, index: any) {
const { context, objectId, heads } = target const { context, objectId, heads } = target
if (index === "length") if (index === "length")
@ -434,7 +493,7 @@ const ListHandler = {
return { configurable: true, enumerable: true, value } return { configurable: true, enumerable: true, value }
}, },
getPrototypeOf(target) { getPrototypeOf(target: Target) {
return Object.getPrototypeOf(target) return Object.getPrototypeOf(target)
}, },
ownKeys(/*target*/): string[] { ownKeys(/*target*/): string[] {
@ -476,14 +535,14 @@ const TextHandler = Object.assign({}, ListHandler, {
}, },
}) })
export function mapProxy( export function mapProxy<T extends Target>(
context: Automerge, context: Automerge,
objectId: ObjID, objectId: ObjID,
textV2: boolean, textV2: boolean,
path?: Prop[], path?: Prop[],
readonly?: boolean, readonly?: boolean,
heads?: Heads heads?: Heads
): MapValue { ): MapValueType<T> {
const target: Target = { const target: Target = {
context, context,
objectId, objectId,
@ -496,19 +555,19 @@ export function mapProxy(
} }
const proxied = {} const proxied = {}
Object.assign(proxied, target) Object.assign(proxied, target)
let result = new Proxy(proxied, MapHandler) const result = new Proxy(proxied, MapHandler)
// conversion through unknown is necessary because the types are so different // conversion through unknown is necessary because the types are so different
return result as unknown as MapValue return result as unknown as MapValueType<T>
} }
export function listProxy( export function listProxy<T extends Target>(
context: Automerge, context: Automerge,
objectId: ObjID, objectId: ObjID,
textV2: boolean, textV2: boolean,
path?: Prop[], path?: Prop[],
readonly?: boolean, readonly?: boolean,
heads?: Heads heads?: Heads
): ListValue { ): ListValueType<T> {
const target: Target = { const target: Target = {
context, context,
objectId, objectId,
@ -521,17 +580,22 @@ export function listProxy(
} }
const proxied = [] const proxied = []
Object.assign(proxied, target) Object.assign(proxied, target)
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore // @ts-ignore
return new Proxy(proxied, ListHandler) as unknown as ListValue return new Proxy(proxied, ListHandler) as unknown as ListValue
} }
interface TextProxy extends Text {
splice: (index: any, del: any, ...vals: any[]) => void
}
export function textProxy( export function textProxy(
context: Automerge, context: Automerge,
objectId: ObjID, objectId: ObjID,
path?: Prop[], path?: Prop[],
readonly?: boolean, readonly?: boolean,
heads?: Heads heads?: Heads
): TextValue { ): TextProxy {
const target: Target = { const target: Target = {
context, context,
objectId, objectId,
@ -542,7 +606,9 @@ export function textProxy(
cache: {}, cache: {},
textV2: false, textV2: false,
} }
return new Proxy(target, TextHandler) as unknown as TextValue const proxied = {}
Object.assign(proxied, target)
return new Proxy(proxied, TextHandler) as unknown as TextProxy
} }
export function rootProxy<T>( export function rootProxy<T>(
@ -554,10 +620,10 @@ export function rootProxy<T>(
return <any>mapProxy(context, "_root", textV2, [], !!readonly) return <any>mapProxy(context, "_root", textV2, [], !!readonly)
} }
function listMethods(target: Target) { function listMethods<T extends Target>(target: T) {
const { context, objectId, path, readonly, frozen, heads, textV2 } = target const { context, objectId, path, readonly, frozen, heads, textV2 } = target
const methods = { const methods = {
deleteAt(index, numDelete) { deleteAt(index: number, numDelete: number) {
if (typeof numDelete === "number") { if (typeof numDelete === "number") {
context.splice(objectId, index, numDelete) context.splice(objectId, index, numDelete)
} else { } else {
@ -572,8 +638,20 @@ function listMethods(target: Target) {
start = parseListIndex(start || 0) start = parseListIndex(start || 0)
end = parseListIndex(end || length) end = parseListIndex(end || length)
for (let i = start; i < Math.min(end, length); i++) { for (let i = start; i < Math.min(end, length); i++) {
if (datatype === "text" || datatype === "list" || datatype === "map") { if (datatype === "list" || datatype === "map") {
context.putObject(objectId, i, value) context.putObject(objectId, i, value)
} else if (datatype === "text") {
if (textV2) {
assertString(value)
context.putObject(objectId, i, value)
} else {
assertText(value)
const text = context.putObject(objectId, i, "")
const proxyText = textProxy(context, text, [...path, i], readonly)
for (let i = 0; i < value.length; i++) {
proxyText[i] = value.get(i)
}
}
} else { } else {
context.put(objectId, i, value, datatype) context.put(objectId, i, value, datatype)
} }
@ -581,7 +659,7 @@ function listMethods(target: Target) {
return this return this
}, },
indexOf(o, start = 0) { indexOf(o: any, start = 0) {
const length = context.length(objectId) const length = context.length(objectId)
for (let i = start; i < length; i++) { for (let i = start; i < length; i++) {
const value = context.getWithType(objectId, i, heads) const value = context.getWithType(objectId, i, heads)
@ -592,7 +670,7 @@ function listMethods(target: Target) {
return -1 return -1
}, },
insertAt(index, ...values) { insertAt(index: number, ...values: any[]) {
this.splice(index, 0, ...values) this.splice(index, 0, ...values)
return this return this
}, },
@ -607,7 +685,7 @@ function listMethods(target: Target) {
return last return last
}, },
push(...values) { push(...values: any[]) {
const len = context.length(objectId) const len = context.length(objectId)
this.splice(len, 0, ...values) this.splice(len, 0, ...values)
return context.length(objectId) return context.length(objectId)
@ -620,7 +698,7 @@ function listMethods(target: Target) {
return first return first
}, },
splice(index, del, ...vals) { splice(index: any, del: any, ...vals: any[]) {
index = parseListIndex(index) index = parseListIndex(index)
del = parseListIndex(del) del = parseListIndex(del)
for (const val of vals) { for (const val of vals) {
@ -638,9 +716,9 @@ function listMethods(target: Target) {
"Sequence object cannot be modified outside of a change block" "Sequence object cannot be modified outside of a change block"
) )
} }
const result: AutomergeValue[] = [] const result: ValueType<T>[] = []
for (let i = 0; i < del; i++) { for (let i = 0; i < del; i++) {
const value = valueAt(target, index) const value = valueAt<T>(target, index)
if (value !== undefined) { if (value !== undefined) {
result.push(value) result.push(value)
} }
@ -663,6 +741,7 @@ function listMethods(target: Target) {
} }
case "text": { case "text": {
if (textV2) { if (textV2) {
assertString(value)
context.insertObject(objectId, index, value) context.insertObject(objectId, index, value)
} else { } else {
const text = context.insertObject(objectId, index, "") const text = context.insertObject(objectId, index, "")
@ -698,7 +777,7 @@ function listMethods(target: Target) {
return result return result
}, },
unshift(...values) { unshift(...values: any) {
this.splice(0, 0, ...values) this.splice(0, 0, ...values)
return context.length(objectId) return context.length(objectId)
}, },
@ -749,11 +828,11 @@ function listMethods(target: Target) {
return iterator return iterator
}, },
toArray(): AutomergeValue[] { toArray(): ValueType<T>[] {
const list: AutomergeValue = [] const list: Array<ValueType<T>> = []
let value let value: ValueType<T> | undefined
do { do {
value = valueAt(target, list.length) value = valueAt<T>(target, list.length)
if (value !== undefined) { if (value !== undefined) {
list.push(value) list.push(value)
} }
@ -762,7 +841,7 @@ function listMethods(target: Target) {
return list return list
}, },
map<T>(f: (AutomergeValue, number) => T): T[] { map<U>(f: (_a: ValueType<T>, _n: number) => U): U[] {
return this.toArray().map(f) return this.toArray().map(f)
}, },
@ -774,24 +853,26 @@ function listMethods(target: Target) {
return this.toArray().toLocaleString() return this.toArray().toLocaleString()
}, },
forEach(f: (AutomergeValue, number) => undefined) { forEach(f: (_a: ValueType<T>, _n: number) => undefined) {
return this.toArray().forEach(f) return this.toArray().forEach(f)
}, },
// todo: real concat function is different // todo: real concat function is different
concat(other: AutomergeValue[]): AutomergeValue[] { concat(other: ValueType<T>[]): ValueType<T>[] {
return this.toArray().concat(other) return this.toArray().concat(other)
}, },
every(f: (AutomergeValue, number) => boolean): boolean { every(f: (_a: ValueType<T>, _n: number) => boolean): boolean {
return this.toArray().every(f) return this.toArray().every(f)
}, },
filter(f: (AutomergeValue, number) => boolean): AutomergeValue[] { filter(f: (_a: ValueType<T>, _n: number) => boolean): ValueType<T>[] {
return this.toArray().filter(f) return this.toArray().filter(f)
}, },
find(f: (AutomergeValue, number) => boolean): AutomergeValue | undefined { find(
f: (_a: ValueType<T>, _n: number) => boolean
): ValueType<T> | undefined {
let index = 0 let index = 0
for (const v of this) { for (const v of this) {
if (f(v, index)) { if (f(v, index)) {
@ -801,7 +882,7 @@ function listMethods(target: Target) {
} }
}, },
findIndex(f: (AutomergeValue, number) => boolean): number { findIndex(f: (_a: ValueType<T>, _n: number) => boolean): number {
let index = 0 let index = 0
for (const v of this) { for (const v of this) {
if (f(v, index)) { if (f(v, index)) {
@ -812,7 +893,7 @@ function listMethods(target: Target) {
return -1 return -1
}, },
includes(elem: AutomergeValue): boolean { includes(elem: ValueType<T>): boolean {
return this.find(e => e === elem) !== undefined return this.find(e => e === elem) !== undefined
}, },
@ -820,29 +901,30 @@ function listMethods(target: Target) {
return this.toArray().join(sep) return this.toArray().join(sep)
}, },
// todo: remove the any reduce<U>(
reduce<T>(f: (any, AutomergeValue) => T, initalValue?: T): T | undefined { f: (acc: U, currentValue: ValueType<T>) => U,
return this.toArray().reduce(f, initalValue) initialValue: U
): U | undefined {
return this.toArray().reduce<U>(f, initialValue)
}, },
// todo: remove the any reduceRight<U>(
reduceRight<T>( f: (acc: U, item: ValueType<T>) => U,
f: (any, AutomergeValue) => T, initialValue: U
initalValue?: T ): U | undefined {
): T | undefined { return this.toArray().reduceRight(f, initialValue)
return this.toArray().reduceRight(f, initalValue)
}, },
lastIndexOf(search: AutomergeValue, fromIndex = +Infinity): number { lastIndexOf(search: ValueType<T>, fromIndex = +Infinity): number {
// this can be faster // this can be faster
return this.toArray().lastIndexOf(search, fromIndex) return this.toArray().lastIndexOf(search, fromIndex)
}, },
slice(index?: number, num?: number): AutomergeValue[] { slice(index?: number, num?: number): ValueType<T>[] {
return this.toArray().slice(index, num) return this.toArray().slice(index, num)
}, },
some(f: (AutomergeValue, number) => boolean): boolean { some(f: (v: ValueType<T>, i: number) => boolean): boolean {
let index = 0 let index = 0
for (const v of this) { for (const v of this) {
if (f(v, index)) { if (f(v, index)) {
@ -869,7 +951,7 @@ function listMethods(target: Target) {
function textMethods(target: Target) { function textMethods(target: Target) {
const { context, objectId, heads } = target const { context, objectId, heads } = target
const methods = { const methods = {
set(index: number, value) { set(index: number, value: any) {
return (this[index] = value) return (this[index] = value)
}, },
get(index: number): AutomergeValue { get(index: number): AutomergeValue {
@ -902,10 +984,22 @@ function textMethods(target: Target) {
toJSON(): string { toJSON(): string {
return this.toString() return this.toString()
}, },
indexOf(o, start = 0) { indexOf(o: any, start = 0) {
const text = context.text(objectId) const text = context.text(objectId)
return text.indexOf(o, start) return text.indexOf(o, start)
}, },
} }
return methods return methods
} }
function assertText(value: Text | string): asserts value is Text {
if (!(value instanceof Text)) {
throw new Error("value was not a Text instance")
}
}
function assertString(value: Text | string): asserts value is string {
if (typeof value !== "string") {
throw new Error("value was not a string")
}
}

View file

@ -1,7 +1,7 @@
/** @hidden **/ /** @hidden **/
export { /** @hidden */ uuid } from "./uuid" export { /** @hidden */ uuid } from "./uuid"
import { rootProxy, listProxy, mapProxy, textProxy } from "./proxies" import { rootProxy } from "./proxies"
import { STATE } from "./constants" import { STATE } from "./constants"
import { import {
@ -20,13 +20,13 @@ export {
type Patch, type Patch,
type PatchCallback, type PatchCallback,
type ScalarValue, type ScalarValue,
Text,
} from "./types" } from "./types"
import { Text } from "./text" import { Text } from "./text"
export { Text } from "./text"
import type { import type {
API, API as WasmAPI,
Actor as ActorId, Actor as ActorId,
Prop, Prop,
ObjID, ObjID,
@ -34,17 +34,29 @@ import type {
DecodedChange, DecodedChange,
Heads, Heads,
MaterializeValue, MaterializeValue,
JsSyncState as SyncState, JsSyncState,
SyncMessage, SyncMessage,
DecodedSyncMessage, DecodedSyncMessage,
} from "@automerge/automerge-wasm" } from "@automerge/automerge-wasm"
export type { export type {
PutPatch, PutPatch,
DelPatch, DelPatch,
SplicePatch, SpliceTextPatch,
InsertPatch,
IncPatch, IncPatch,
SyncMessage, SyncMessage,
} from "@automerge/automerge-wasm" } from "@automerge/automerge-wasm"
/** @hidden **/
type API = WasmAPI
const SyncStateSymbol = Symbol("_syncstate")
/**
* An opaque type tracking the state of sync with a remote peer
*/
type SyncState = JsSyncState & { _opaque: typeof SyncStateSymbol }
import { ApiHandler, type ChangeToEncode, UseApi } from "./low_level" import { ApiHandler, type ChangeToEncode, UseApi } from "./low_level"
import { Automerge } from "@automerge/automerge-wasm" import { Automerge } from "@automerge/automerge-wasm"
@ -53,6 +65,8 @@ import { RawString } from "./raw_string"
import { _state, _is_proxy, _trace, _obj } from "./internal_state" import { _state, _is_proxy, _trace, _obj } from "./internal_state"
import { stableConflictAt } from "./conflicts"
/** Options passed to {@link change}, and {@link emptyChange} /** Options passed to {@link change}, and {@link emptyChange}
* @typeParam T - The type of value contained in the document * @typeParam T - The type of value contained in the document
*/ */
@ -70,13 +84,36 @@ export type ChangeOptions<T> = {
*/ */
export type ApplyOptions<T> = { patchCallback?: PatchCallback<T> } export type ApplyOptions<T> = { patchCallback?: PatchCallback<T> }
/**
* A List is an extended Array that adds the two helper methods `deleteAt` and `insertAt`.
*/
export interface List<T> extends Array<T> {
insertAt(index: number, ...args: T[]): List<T>
deleteAt(index: number, numDelete?: number): List<T>
}
/**
* To extend an arbitrary type, we have to turn any arrays that are part of the type's definition into Lists.
* So we recurse through the properties of T, turning any Arrays we find into Lists.
*/
export type Extend<T> =
// is it an array? make it a list (we recursively extend the type of the array's elements as well)
T extends Array<infer T>
? List<Extend<T>>
: // is it an object? recursively extend all of its properties
// eslint-disable-next-line @typescript-eslint/ban-types
T extends Object
? { [P in keyof T]: Extend<T[P]> }
: // otherwise leave the type alone
T
/** /**
* Function which is called by {@link change} when making changes to a `Doc<T>` * Function which is called by {@link change} when making changes to a `Doc<T>`
* @typeParam T - The type of value contained in the document * @typeParam T - The type of value contained in the document
* *
* This function may mutate `doc` * This function may mutate `doc`
*/ */
export type ChangeFn<T> = (doc: T) => void export type ChangeFn<T> = (doc: Extend<T>) => void
/** @hidden **/ /** @hidden **/
export interface State<T> { export interface State<T> {
@ -135,11 +172,12 @@ export function init<T>(_opts?: ActorId | InitOptions<T>): Doc<T> {
const handle = ApiHandler.create(opts.enableTextV2 || false, opts.actor) const handle = ApiHandler.create(opts.enableTextV2 || false, opts.actor)
handle.enablePatches(true) handle.enablePatches(true)
handle.enableFreeze(!!opts.freeze) handle.enableFreeze(!!opts.freeze)
handle.registerDatatype("counter", (n: any) => new Counter(n)) handle.registerDatatype("counter", (n: number) => new Counter(n))
let textV2 = opts.enableTextV2 || false const textV2 = opts.enableTextV2 || false
if (textV2) { if (textV2) {
handle.registerDatatype("str", (n: string) => new RawString(n)) handle.registerDatatype("str", (n: string) => new RawString(n))
} else { } else {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
handle.registerDatatype("text", (n: any) => new Text(n)) handle.registerDatatype("text", (n: any) => new Text(n))
} }
const doc = handle.materialize("/", undefined, { const doc = handle.materialize("/", undefined, {
@ -203,7 +241,7 @@ export function clone<T>(
// `change` uses the presence of state.heads to determine if we are in a view // `change` uses the presence of state.heads to determine if we are in a view
// set it to undefined to indicate that this is a full fat document // set it to undefined to indicate that this is a full fat document
const { heads: oldHeads, ...stateSansHeads } = state const { heads: _oldHeads, ...stateSansHeads } = state
return handle.applyPatches(doc, { ...stateSansHeads, handle }) return handle.applyPatches(doc, { ...stateSansHeads, handle })
} }
@ -267,7 +305,7 @@ export function from<T extends Record<string, unknown>>(
* @example A change with a message and a timestamp * @example A change with a message and a timestamp
* *
* ``` * ```
* doc1 = automerge.change(doc1, {message: "add another value", timestamp: 1640995200}, d => { * doc1 = automerge.change(doc1, {message: "add another value", time: 1640995200}, d => {
* d.key2 = "value2" * d.key2 = "value2"
* }) * })
* ``` * ```
@ -278,7 +316,7 @@ export function from<T extends Record<string, unknown>>(
* let patchCallback = patch => { * let patchCallback = patch => {
* patchedPath = patch.path * patchedPath = patch.path
* } * }
* doc1 = automerge.change(doc1, {message, "add another value", timestamp: 1640995200, patchCallback}, d => { * doc1 = automerge.change(doc1, {message, "add another value", time: 1640995200, patchCallback}, d => {
* d.key2 = "value2" * d.key2 = "value2"
* }) * })
* assert.equal(patchedPath, ["key2"]) * assert.equal(patchedPath, ["key2"])
@ -342,7 +380,7 @@ function _change<T>(
try { try {
state.heads = heads state.heads = heads
const root: T = rootProxy(state.handle, state.textV2) const root: T = rootProxy(state.handle, state.textV2)
callback(root) callback(root as Extend<T>)
if (state.handle.pendingOps() === 0) { if (state.handle.pendingOps() === 0) {
state.heads = undefined state.heads = undefined
return doc return doc
@ -540,62 +578,6 @@ export function getActorId<T>(doc: Doc<T>): ActorId {
*/ */
type Conflicts = { [key: string]: AutomergeValue } type Conflicts = { [key: string]: AutomergeValue }
function conflictAt(
context: Automerge,
objectId: ObjID,
prop: Prop,
textV2: boolean
): Conflicts | undefined {
const values = context.getAll(objectId, prop)
if (values.length <= 1) {
return
}
const result: Conflicts = {}
for (const fullVal of values) {
switch (fullVal[0]) {
case "map":
result[fullVal[1]] = mapProxy(context, fullVal[1], textV2, [prop], true)
break
case "list":
result[fullVal[1]] = listProxy(
context,
fullVal[1],
textV2,
[prop],
true
)
break
case "text":
if (textV2) {
result[fullVal[1]] = context.text(fullVal[1])
} else {
result[fullVal[1]] = textProxy(context, objectId, [prop], true)
}
break
//case "table":
//case "cursor":
case "str":
case "uint":
case "int":
case "f64":
case "boolean":
case "bytes":
case "null":
result[fullVal[2]] = fullVal[1]
break
case "counter":
result[fullVal[2]] = new Counter(fullVal[1])
break
case "timestamp":
result[fullVal[2]] = new Date(fullVal[1])
break
default:
throw RangeError(`datatype ${fullVal[0]} unimplemented`)
}
}
return result
}
/** /**
* Get the conflicts associated with a property * Get the conflicts associated with a property
* *
@ -645,9 +627,12 @@ export function getConflicts<T>(
prop: Prop prop: Prop
): Conflicts | undefined { ): Conflicts | undefined {
const state = _state(doc, false) const state = _state(doc, false)
if (state.textV2) {
throw new Error("use unstable.getConflicts for an unstable document")
}
const objectId = _obj(doc) const objectId = _obj(doc)
if (objectId != null) { if (objectId != null) {
return conflictAt(state.handle, objectId, prop, state.textV2) return stableConflictAt(state.handle, objectId, prop)
} else { } else {
return undefined return undefined
} }
@ -671,6 +656,7 @@ export function getLastLocalChange<T>(doc: Doc<T>): Change | undefined {
* This is useful to determine if something is actually an automerge document, * This is useful to determine if something is actually an automerge document,
* if `doc` is not an automerge document this will return null. * if `doc` is not an automerge document this will return null.
*/ */
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function getObjectId(doc: any, prop?: Prop): ObjID | null { export function getObjectId(doc: any, prop?: Prop): ObjID | null {
if (prop) { if (prop) {
const state = _state(doc, false) const state = _state(doc, false)
@ -797,7 +783,7 @@ export function decodeSyncState(state: Uint8Array): SyncState {
const sync = ApiHandler.decodeSyncState(state) const sync = ApiHandler.decodeSyncState(state)
const result = ApiHandler.exportSyncState(sync) const result = ApiHandler.exportSyncState(sync)
sync.free() sync.free()
return result return result as SyncState
} }
/** /**
@ -818,7 +804,7 @@ export function generateSyncMessage<T>(
const state = _state(doc) const state = _state(doc)
const syncState = ApiHandler.importSyncState(inState) const syncState = ApiHandler.importSyncState(inState)
const message = state.handle.generateSyncMessage(syncState) const message = state.handle.generateSyncMessage(syncState)
const outState = ApiHandler.exportSyncState(syncState) const outState = ApiHandler.exportSyncState(syncState) as SyncState
return [outState, message] return [outState, message]
} }
@ -860,7 +846,7 @@ export function receiveSyncMessage<T>(
} }
const heads = state.handle.getHeads() const heads = state.handle.getHeads()
state.handle.receiveSyncMessage(syncState, message) state.handle.receiveSyncMessage(syncState, message)
const outSyncState = ApiHandler.exportSyncState(syncState) const outSyncState = ApiHandler.exportSyncState(syncState) as SyncState
return [ return [
progressDocument(doc, heads, opts.patchCallback || state.patchCallback), progressDocument(doc, heads, opts.patchCallback || state.patchCallback),
outSyncState, outSyncState,
@ -877,7 +863,7 @@ export function receiveSyncMessage<T>(
* @group sync * @group sync
*/ */
export function initSyncState(): SyncState { export function initSyncState(): SyncState {
return ApiHandler.exportSyncState(ApiHandler.initSyncState()) return ApiHandler.exportSyncState(ApiHandler.initSyncState()) as SyncState
} }
/** @hidden */ /** @hidden */

View file

@ -3,9 +3,12 @@ import { TEXT, STATE } from "./constants"
import type { InternalState } from "./internal_state" import type { InternalState } from "./internal_state"
export class Text { export class Text {
//eslint-disable-next-line @typescript-eslint/no-explicit-any
elems: Array<any> elems: Array<any>
str: string | undefined str: string | undefined
//eslint-disable-next-line @typescript-eslint/no-explicit-any
spans: Array<any> | undefined; spans: Array<any> | undefined;
//eslint-disable-next-line @typescript-eslint/no-explicit-any
[STATE]?: InternalState<any> [STATE]?: InternalState<any>
constructor(text?: string | string[] | Value[]) { constructor(text?: string | string[] | Value[]) {
@ -25,6 +28,7 @@ export class Text {
return this.elems.length return this.elems.length
} }
//eslint-disable-next-line @typescript-eslint/no-explicit-any
get(index: number): any { get(index: number): any {
return this.elems[index] return this.elems[index]
} }
@ -73,7 +77,7 @@ export class Text {
* For example, the value `['a', 'b', {x: 3}, 'c', 'd']` has spans: * For example, the value `['a', 'b', {x: 3}, 'c', 'd']` has spans:
* `=> ['ab', {x: 3}, 'cd']` * `=> ['ab', {x: 3}, 'cd']`
*/ */
toSpans(): Array<Value | Object> { toSpans(): Array<Value | object> {
if (!this.spans) { if (!this.spans) {
this.spans = [] this.spans = []
let chars = "" let chars = ""
@ -118,7 +122,7 @@ export class Text {
/** /**
* Inserts new list items `values` starting at position `index`. * Inserts new list items `values` starting at position `index`.
*/ */
insertAt(index: number, ...values: Array<Value | Object>) { insertAt(index: number, ...values: Array<Value | object>) {
if (this[STATE]) { if (this[STATE]) {
throw new RangeError( throw new RangeError(
"object cannot be modified outside of a change block" "object cannot be modified outside of a change block"
@ -140,7 +144,7 @@ export class Text {
this.elems.splice(index, numDelete) this.elems.splice(index, numDelete)
} }
map<T>(callback: (e: Value | Object) => T) { map<T>(callback: (e: Value | object) => T) {
this.elems.map(callback) this.elems.map(callback)
} }

View file

@ -1,4 +1,5 @@
export { Text } from "./text" export { Text } from "./text"
import { Text } from "./text"
export { Counter } from "./counter" export { Counter } from "./counter"
export { Int, Uint, Float64 } from "./numbers" export { Int, Uint, Float64 } from "./numbers"
@ -10,9 +11,9 @@ export type AutomergeValue =
| ScalarValue | ScalarValue
| { [key: string]: AutomergeValue } | { [key: string]: AutomergeValue }
| Array<AutomergeValue> | Array<AutomergeValue>
| Text
export type MapValue = { [key: string]: AutomergeValue } export type MapValue = { [key: string]: AutomergeValue }
export type ListValue = Array<AutomergeValue> export type ListValue = Array<AutomergeValue>
export type TextValue = Array<AutomergeValue>
export type ScalarValue = export type ScalarValue =
| string | string
| number | number

View file

@ -22,9 +22,9 @@
* This leads to the following differences from `stable`: * This leads to the following differences from `stable`:
* *
* * There is no `unstable.Text` class, all strings are text objects * * There is no `unstable.Text` class, all strings are text objects
* * Reading strings in a `future` document is the same as reading any other * * Reading strings in an `unstable` document is the same as reading any other
* javascript string * javascript string
* * To modify strings in a `future` document use {@link splice} * * To modify strings in an `unstable` document use {@link splice}
* * The {@link AutomergeValue} type does not include the {@link Text} * * The {@link AutomergeValue} type does not include the {@link Text}
* class but the {@link RawString} class is included in the {@link ScalarValue} * class but the {@link RawString} class is included in the {@link ScalarValue}
* type * type
@ -35,7 +35,6 @@
* *
* @module * @module
*/ */
import { Counter } from "./types"
export { export {
Counter, Counter,
@ -45,32 +44,20 @@ export {
Float64, Float64,
type Patch, type Patch,
type PatchCallback, type PatchCallback,
} from "./types" type AutomergeValue,
type ScalarValue,
} from "./unstable_types"
import type { PatchCallback } from "./stable" import type { PatchCallback } from "./stable"
export type AutomergeValue = import { type UnstableConflicts as Conflicts } from "./conflicts"
| ScalarValue import { unstableConflictAt } from "./conflicts"
| { [key: string]: AutomergeValue }
| Array<AutomergeValue>
export type MapValue = { [key: string]: AutomergeValue }
export type ListValue = Array<AutomergeValue>
export type ScalarValue =
| string
| number
| null
| boolean
| Date
| Counter
| Uint8Array
| RawString
export type Conflicts = { [key: string]: AutomergeValue }
export type { export type {
PutPatch, PutPatch,
DelPatch, DelPatch,
SplicePatch, SpliceTextPatch,
InsertPatch,
IncPatch, IncPatch,
SyncMessage, SyncMessage,
} from "@automerge/automerge-wasm" } from "@automerge/automerge-wasm"
@ -124,7 +111,6 @@ export { RawString } from "./raw_string"
export const getBackend = stable.getBackend export const getBackend = stable.getBackend
import { _is_proxy, _state, _obj } from "./internal_state" import { _is_proxy, _state, _obj } from "./internal_state"
import { RawString } from "./raw_string"
/** /**
* Create a new automerge document * Create a new automerge document
@ -136,7 +122,7 @@ import { RawString } from "./raw_string"
* random actor ID * random actor ID
*/ */
export function init<T>(_opts?: ActorId | InitOptions<T>): Doc<T> { export function init<T>(_opts?: ActorId | InitOptions<T>): Doc<T> {
let opts = importOpts(_opts) const opts = importOpts(_opts)
opts.enableTextV2 = true opts.enableTextV2 = true
return stable.init(opts) return stable.init(opts)
} }
@ -160,7 +146,7 @@ export function clone<T>(
doc: Doc<T>, doc: Doc<T>,
_opts?: ActorId | InitOptions<T> _opts?: ActorId | InitOptions<T>
): Doc<T> { ): Doc<T> {
let opts = importOpts(_opts) const opts = importOpts(_opts)
opts.enableTextV2 = true opts.enableTextV2 = true
return stable.clone(doc, opts) return stable.clone(doc, opts)
} }
@ -295,6 +281,14 @@ export function getConflicts<T>(
doc: Doc<T>, doc: Doc<T>,
prop: stable.Prop prop: stable.Prop
): Conflicts | undefined { ): Conflicts | undefined {
// this function only exists to get the types to line up with future.AutomergeValue const state = _state(doc, false)
return stable.getConflicts(doc, prop) if (!state.textV2) {
throw new Error("use getConflicts for a stable document")
}
const objectId = _obj(doc)
if (objectId != null) {
return unstableConflictAt(state.handle, objectId, prop)
} else {
return undefined
}
} }

View file

@ -0,0 +1,30 @@
import { Counter } from "./types"
export {
Counter,
type Doc,
Int,
Uint,
Float64,
type Patch,
type PatchCallback,
} from "./types"
import { RawString } from "./raw_string"
export { RawString } from "./raw_string"
export type AutomergeValue =
| ScalarValue
| { [key: string]: AutomergeValue }
| Array<AutomergeValue>
export type MapValue = { [key: string]: AutomergeValue }
export type ListValue = Array<AutomergeValue>
export type ScalarValue =
| string
| number
| null
| boolean
| Date
| Counter
| Uint8Array
| RawString

View file

@ -58,6 +58,22 @@ describe("Automerge", () => {
}) })
}) })
it("should be able to insert and delete a large number of properties", () => {
let doc = Automerge.init<any>()
doc = Automerge.change(doc, doc => {
doc["k1"] = true
})
for (let idx = 1; idx <= 200; idx++) {
doc = Automerge.change(doc, doc => {
delete doc["k" + idx]
doc["k" + (idx + 1)] = true
assert(Object.keys(doc).length == 1)
})
}
})
it("can detect an automerge doc with isAutomerge()", () => { it("can detect an automerge doc with isAutomerge()", () => {
const doc1 = Automerge.from({ sub: { object: true } }) const doc1 = Automerge.from({ sub: { object: true } })
assert(Automerge.isAutomerge(doc1)) assert(Automerge.isAutomerge(doc1))
@ -267,7 +283,6 @@ describe("Automerge", () => {
}) })
assert.deepEqual(doc5, { list: [2, 1, 9, 10, 3, 11, 12] }) assert.deepEqual(doc5, { list: [2, 1, 9, 10, 3, 11, 12] })
let doc6 = Automerge.change(doc5, d => { let doc6 = Automerge.change(doc5, d => {
// @ts-ignore
d.list.insertAt(3, 100, 101) d.list.insertAt(3, 100, 101)
}) })
assert.deepEqual(doc6, { list: [2, 1, 9, 100, 101, 10, 3, 11, 12] }) assert.deepEqual(doc6, { list: [2, 1, 9, 100, 101, 10, 3, 11, 12] })

View file

@ -461,12 +461,12 @@ describe("Automerge", () => {
s1 = Automerge.change(s1, "set foo", doc => { s1 = Automerge.change(s1, "set foo", doc => {
doc.foo = "bar" doc.foo = "bar"
}) })
let deleted let deleted: any
s1 = Automerge.change(s1, "del foo", doc => { s1 = Automerge.change(s1, "del foo", doc => {
deleted = delete doc.foo deleted = delete doc.foo
}) })
assert.strictEqual(deleted, true) assert.strictEqual(deleted, true)
let deleted2 let deleted2: any
assert.doesNotThrow(() => { assert.doesNotThrow(() => {
s1 = Automerge.change(s1, "del baz", doc => { s1 = Automerge.change(s1, "del baz", doc => {
deleted2 = delete doc.baz deleted2 = delete doc.baz
@ -515,7 +515,7 @@ describe("Automerge", () => {
s1 = Automerge.change(s1, doc => { s1 = Automerge.change(s1, doc => {
doc.nested = {} doc.nested = {}
}) })
let id = Automerge.getObjectId(s1.nested) Automerge.getObjectId(s1.nested)
assert.strictEqual( assert.strictEqual(
OPID_PATTERN.test(Automerge.getObjectId(s1.nested)!), OPID_PATTERN.test(Automerge.getObjectId(s1.nested)!),
true true
@ -975,6 +975,7 @@ describe("Automerge", () => {
it("should allow adding and removing list elements in the same change callback", () => { it("should allow adding and removing list elements in the same change callback", () => {
let s1 = Automerge.change( let s1 = Automerge.change(
Automerge.init<{ noodles: Array<string> }>(), Automerge.init<{ noodles: Array<string> }>(),
// @ts-ignore
doc => (doc.noodles = []) doc => (doc.noodles = [])
) )
s1 = Automerge.change(s1, doc => { s1 = Automerge.change(s1, doc => {
@ -1848,9 +1849,8 @@ describe("Automerge", () => {
}) })
assert.deepStrictEqual(patches, [ assert.deepStrictEqual(patches, [
{ action: "put", path: ["birds"], value: [] }, { action: "put", path: ["birds"], value: [] },
{ action: "insert", path: ["birds", 0], values: [""] }, { action: "insert", path: ["birds", 0], values: ["", ""] },
{ action: "splice", path: ["birds", 0, 0], value: "Goldfinch" }, { action: "splice", path: ["birds", 0, 0], value: "Goldfinch" },
{ action: "insert", path: ["birds", 1], values: [""] },
{ action: "splice", path: ["birds", 1, 0], value: "Chaffinch" }, { action: "splice", path: ["birds", 1, 0], value: "Chaffinch" },
]) ])
}) })

View file

@ -38,4 +38,62 @@ describe("stable/unstable interop", () => {
stableDoc = unstable.merge(stableDoc, unstableDoc) stableDoc = unstable.merge(stableDoc, unstableDoc)
assert.deepStrictEqual(stableDoc.text, "abc") assert.deepStrictEqual(stableDoc.text, "abc")
}) })
it("should show conflicts on text objects", () => {
let doc1 = stable.from({ text: new stable.Text("abc") }, "bb")
let doc2 = stable.from({ text: new stable.Text("def") }, "aa")
doc1 = stable.merge(doc1, doc2)
let conflicts = stable.getConflicts(doc1, "text")!
assert.equal(conflicts["1@bb"]!.toString(), "abc")
assert.equal(conflicts["1@aa"]!.toString(), "def")
let unstableDoc = unstable.init<any>()
unstableDoc = unstable.merge(unstableDoc, doc1)
let conflicts2 = unstable.getConflicts(unstableDoc, "text")!
assert.equal(conflicts2["1@bb"]!.toString(), "abc")
assert.equal(conflicts2["1@aa"]!.toString(), "def")
})
it("should allow filling a list with text in stable", () => {
let doc = stable.from<{ list: Array<stable.Text | null> }>({
list: [null, null, null],
})
doc = stable.change(doc, doc => {
doc.list.fill(new stable.Text("abc"), 0, 3)
})
assert.deepStrictEqual(doc.list, [
new stable.Text("abc"),
new stable.Text("abc"),
new stable.Text("abc"),
])
})
it("should allow filling a list with text in unstable", () => {
let doc = unstable.from<{ list: Array<string | null> }>({
list: [null, null, null],
})
doc = stable.change(doc, doc => {
doc.list.fill("abc", 0, 3)
})
assert.deepStrictEqual(doc.list, ["abc", "abc", "abc"])
})
it("should allow splicing text into a list on stable", () => {
let doc = stable.from<{ list: Array<stable.Text> }>({ list: [] })
doc = stable.change(doc, doc => {
doc.list.splice(0, 0, new stable.Text("abc"), new stable.Text("def"))
})
assert.deepStrictEqual(doc.list, [
new stable.Text("abc"),
new stable.Text("def"),
])
})
it("should allow splicing text into a list on unstable", () => {
let doc = unstable.from<{ list: Array<string> }>({ list: [] })
doc = unstable.change(doc, doc => {
doc.list.splice(0, 0, "abc", "def")
})
assert.deepStrictEqual(doc.list, ["abc", "def"])
})
}) })

View file

@ -10,13 +10,8 @@ members = [
resolver = "2" resolver = "2"
[profile.release] [profile.release]
debug = true
lto = true lto = true
opt-level = 3 codegen-units = 1
[profile.bench] [profile.bench]
debug = true debug = true
[profile.release.package.automerge-wasm]
debug = false
opt-level = 3

View file

@ -0,0 +1,250 @@
---
Language: Cpp
# BasedOnStyle: Chromium
AccessModifierOffset: -1
AlignAfterOpenBracket: Align
AlignArrayOfStructures: None
AlignConsecutiveAssignments:
Enabled: false
AcrossEmptyLines: false
AcrossComments: false
AlignCompound: false
PadOperators: true
AlignConsecutiveBitFields:
Enabled: false
AcrossEmptyLines: false
AcrossComments: false
AlignCompound: false
PadOperators: false
AlignConsecutiveDeclarations:
Enabled: false
AcrossEmptyLines: false
AcrossComments: false
AlignCompound: false
PadOperators: false
AlignConsecutiveMacros:
Enabled: false
AcrossEmptyLines: false
AcrossComments: false
AlignCompound: false
PadOperators: false
AlignEscapedNewlines: Left
AlignOperands: Align
AlignTrailingComments: true
AllowAllArgumentsOnNextLine: true
AllowAllParametersOfDeclarationOnNextLine: false
AllowShortEnumsOnASingleLine: true
AllowShortBlocksOnASingleLine: Never
AllowShortCaseLabelsOnASingleLine: false
AllowShortFunctionsOnASingleLine: Inline
AllowShortLambdasOnASingleLine: All
AllowShortIfStatementsOnASingleLine: Never
AllowShortLoopsOnASingleLine: false
AlwaysBreakAfterDefinitionReturnType: None
AlwaysBreakAfterReturnType: None
AlwaysBreakBeforeMultilineStrings: true
AlwaysBreakTemplateDeclarations: Yes
AttributeMacros:
- __capability
BinPackArguments: true
BinPackParameters: false
BraceWrapping:
AfterCaseLabel: false
AfterClass: false
AfterControlStatement: Never
AfterEnum: false
AfterFunction: false
AfterNamespace: false
AfterObjCDeclaration: false
AfterStruct: false
AfterUnion: false
AfterExternBlock: false
BeforeCatch: false
BeforeElse: false
BeforeLambdaBody: false
BeforeWhile: false
IndentBraces: false
SplitEmptyFunction: true
SplitEmptyRecord: true
SplitEmptyNamespace: true
BreakBeforeBinaryOperators: None
BreakBeforeConceptDeclarations: Always
BreakBeforeBraces: Attach
BreakBeforeInheritanceComma: false
BreakInheritanceList: BeforeColon
BreakBeforeTernaryOperators: true
BreakConstructorInitializersBeforeComma: false
BreakConstructorInitializers: BeforeColon
BreakAfterJavaFieldAnnotations: false
BreakStringLiterals: true
ColumnLimit: 120
CommentPragmas: '^ IWYU pragma:'
QualifierAlignment: Leave
CompactNamespaces: false
ConstructorInitializerIndentWidth: 4
ContinuationIndentWidth: 4
Cpp11BracedListStyle: true
DeriveLineEnding: true
DerivePointerAlignment: false
DisableFormat: false
EmptyLineAfterAccessModifier: Never
EmptyLineBeforeAccessModifier: LogicalBlock
ExperimentalAutoDetectBinPacking: false
PackConstructorInitializers: NextLine
BasedOnStyle: ''
ConstructorInitializerAllOnOneLineOrOnePerLine: false
AllowAllConstructorInitializersOnNextLine: true
FixNamespaceComments: true
ForEachMacros:
- foreach
- Q_FOREACH
- BOOST_FOREACH
IfMacros:
- KJ_IF_MAYBE
IncludeBlocks: Preserve
IncludeCategories:
- Regex: '^<ext/.*\.h>'
Priority: 2
SortPriority: 0
CaseSensitive: false
- Regex: '^<.*\.h>'
Priority: 1
SortPriority: 0
CaseSensitive: false
- Regex: '^<.*'
Priority: 2
SortPriority: 0
CaseSensitive: false
- Regex: '.*'
Priority: 3
SortPriority: 0
CaseSensitive: false
IncludeIsMainRegex: '([-_](test|unittest))?$'
IncludeIsMainSourceRegex: ''
IndentAccessModifiers: false
IndentCaseLabels: true
IndentCaseBlocks: false
IndentGotoLabels: true
IndentPPDirectives: None
IndentExternBlock: AfterExternBlock
IndentRequiresClause: true
IndentWidth: 4
IndentWrappedFunctionNames: false
InsertBraces: false
InsertTrailingCommas: None
JavaScriptQuotes: Leave
JavaScriptWrapImports: true
KeepEmptyLinesAtTheStartOfBlocks: false
LambdaBodyIndentation: Signature
MacroBlockBegin: ''
MacroBlockEnd: ''
MaxEmptyLinesToKeep: 1
NamespaceIndentation: None
ObjCBinPackProtocolList: Never
ObjCBlockIndentWidth: 2
ObjCBreakBeforeNestedBlockParam: true
ObjCSpaceAfterProperty: false
ObjCSpaceBeforeProtocolList: true
PenaltyBreakAssignment: 2
PenaltyBreakBeforeFirstCallParameter: 1
PenaltyBreakComment: 300
PenaltyBreakFirstLessLess: 120
PenaltyBreakOpenParenthesis: 0
PenaltyBreakString: 1000
PenaltyBreakTemplateDeclaration: 10
PenaltyExcessCharacter: 1000000
PenaltyReturnTypeOnItsOwnLine: 200
PenaltyIndentedWhitespace: 0
PointerAlignment: Left
PPIndentWidth: -1
RawStringFormats:
- Language: Cpp
Delimiters:
- cc
- CC
- cpp
- Cpp
- CPP
- 'c++'
- 'C++'
CanonicalDelimiter: ''
BasedOnStyle: google
- Language: TextProto
Delimiters:
- pb
- PB
- proto
- PROTO
EnclosingFunctions:
- EqualsProto
- EquivToProto
- PARSE_PARTIAL_TEXT_PROTO
- PARSE_TEST_PROTO
- PARSE_TEXT_PROTO
- ParseTextOrDie
- ParseTextProtoOrDie
- ParseTestProto
- ParsePartialTestProto
CanonicalDelimiter: pb
BasedOnStyle: google
ReferenceAlignment: Pointer
ReflowComments: true
RemoveBracesLLVM: false
RequiresClausePosition: OwnLine
SeparateDefinitionBlocks: Leave
ShortNamespaceLines: 1
SortIncludes: CaseSensitive
SortJavaStaticImport: Before
SortUsingDeclarations: true
SpaceAfterCStyleCast: false
SpaceAfterLogicalNot: false
SpaceAfterTemplateKeyword: true
SpaceBeforeAssignmentOperators: true
SpaceBeforeCaseColon: false
SpaceBeforeCpp11BracedList: false
SpaceBeforeCtorInitializerColon: true
SpaceBeforeInheritanceColon: true
SpaceBeforeParens: ControlStatements
SpaceBeforeParensOptions:
AfterControlStatements: true
AfterForeachMacros: true
AfterFunctionDefinitionName: false
AfterFunctionDeclarationName: false
AfterIfMacros: true
AfterOverloadedOperator: false
AfterRequiresInClause: false
AfterRequiresInExpression: false
BeforeNonEmptyParentheses: false
SpaceAroundPointerQualifiers: Default
SpaceBeforeRangeBasedForLoopColon: true
SpaceInEmptyBlock: false
SpaceInEmptyParentheses: false
SpacesBeforeTrailingComments: 2
SpacesInAngles: Never
SpacesInConditionalStatement: false
SpacesInContainerLiterals: true
SpacesInCStyleCastParentheses: false
SpacesInLineCommentPrefix:
Minimum: 1
Maximum: -1
SpacesInParentheses: false
SpacesInSquareBrackets: false
SpaceBeforeSquareBrackets: false
BitFieldColonSpacing: Both
Standard: Auto
StatementAttributeLikeMacros:
- Q_EMIT
StatementMacros:
- Q_UNUSED
- QT_REQUIRE_VERSION
TabWidth: 8
UseCRLF: false
UseTab: Never
WhitespaceSensitiveMacros:
- STRINGIZE
- PP_STRINGIZE
- BOOST_PP_STRINGIZE
- NS_SWIFT_NAME
- CF_SWIFT_NAME
...

View file

@ -1,10 +1,10 @@
automerge automerge
automerge.h automerge.h
automerge.o automerge.o
*.cmake build/
CMakeCache.txt
CMakeFiles CMakeFiles
CMakePresets.json
Makefile Makefile
DartConfiguration.tcl DartConfiguration.tcl
config.h out/
CMakeCache.txt
Cargo

View file

@ -1,97 +1,297 @@
cmake_minimum_required(VERSION 3.18 FATAL_ERROR) cmake_minimum_required(VERSION 3.23 FATAL_ERROR)
set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake") project(automerge-c VERSION 0.1.0
LANGUAGES C
DESCRIPTION "C bindings for the Automerge Rust library.")
# Parse the library name, project name and project version out of Cargo's TOML file. set(LIBRARY_NAME "automerge")
set(CARGO_LIB_SECTION OFF)
set(LIBRARY_NAME "") set(CMAKE_WINDOWS_EXPORT_ALL_SYMBOLS ON)
set(CARGO_PKG_SECTION OFF)
set(CARGO_PKG_NAME "")
set(CARGO_PKG_VERSION "")
file(READ Cargo.toml TOML_STRING)
string(REPLACE ";" "\\\\;" TOML_STRING "${TOML_STRING}")
string(REPLACE "\n" ";" TOML_LINES "${TOML_STRING}")
foreach(TOML_LINE IN ITEMS ${TOML_LINES})
string(REGEX MATCH "^\\[(lib|package)\\]$" _ ${TOML_LINE})
if(CMAKE_MATCH_1 STREQUAL "lib")
set(CARGO_LIB_SECTION ON)
set(CARGO_PKG_SECTION OFF)
elseif(CMAKE_MATCH_1 STREQUAL "package")
set(CARGO_LIB_SECTION OFF)
set(CARGO_PKG_SECTION ON)
endif()
string(REGEX MATCH "^name += +\"([^\"]+)\"$" _ ${TOML_LINE})
if(CMAKE_MATCH_1 AND (CARGO_LIB_SECTION AND NOT CARGO_PKG_SECTION))
set(LIBRARY_NAME "${CMAKE_MATCH_1}")
elseif(CMAKE_MATCH_1 AND (NOT CARGO_LIB_SECTION AND CARGO_PKG_SECTION))
set(CARGO_PKG_NAME "${CMAKE_MATCH_1}")
endif()
string(REGEX MATCH "^version += +\"([^\"]+)\"$" _ ${TOML_LINE})
if(CMAKE_MATCH_1 AND CARGO_PKG_SECTION)
set(CARGO_PKG_VERSION "${CMAKE_MATCH_1}")
endif()
if(LIBRARY_NAME AND (CARGO_PKG_NAME AND CARGO_PKG_VERSION))
break()
endif()
endforeach()
project(${CARGO_PKG_NAME} VERSION 0.0.1 LANGUAGES C DESCRIPTION "C bindings for the Automerge Rust backend.")
include(CTest)
option(BUILD_SHARED_LIBS "Enable the choice of a shared or static library.") option(BUILD_SHARED_LIBS "Enable the choice of a shared or static library.")
include(CTest)
include(CMakePackageConfigHelpers) include(CMakePackageConfigHelpers)
include(GNUInstallDirs) include(GNUInstallDirs)
set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake")
string(MAKE_C_IDENTIFIER ${PROJECT_NAME} SYMBOL_PREFIX) string(MAKE_C_IDENTIFIER ${PROJECT_NAME} SYMBOL_PREFIX)
string(TOUPPER ${SYMBOL_PREFIX} SYMBOL_PREFIX) string(TOUPPER ${SYMBOL_PREFIX} SYMBOL_PREFIX)
set(CARGO_TARGET_DIR "${CMAKE_CURRENT_BINARY_DIR}/Cargo/target") set(CARGO_TARGET_DIR "${CMAKE_BINARY_DIR}/Cargo/target")
set(CBINDGEN_INCLUDEDIR "${CARGO_TARGET_DIR}/${CMAKE_INSTALL_INCLUDEDIR}") set(CBINDGEN_INCLUDEDIR "${CMAKE_BINARY_DIR}/${CMAKE_INSTALL_INCLUDEDIR}")
set(CBINDGEN_TARGET_DIR "${CBINDGEN_INCLUDEDIR}/${PROJECT_NAME}") set(CBINDGEN_TARGET_DIR "${CBINDGEN_INCLUDEDIR}/${PROJECT_NAME}")
add_subdirectory(src) find_program (
CARGO_CMD
"cargo"
PATHS "$ENV{CARGO_HOME}/bin"
DOC "The Cargo command"
)
# Generate and install the configuration header. if(NOT CARGO_CMD)
message(FATAL_ERROR "Cargo (Rust package manager) not found! "
"Please install it and/or set the CARGO_HOME "
"environment variable to its path.")
endif()
string(TOLOWER "${CMAKE_BUILD_TYPE}" BUILD_TYPE_LOWER)
# In order to build with -Z build-std, we need to pass target explicitly.
# https://doc.rust-lang.org/cargo/reference/unstable.html#build-std
execute_process (
COMMAND rustc -vV
OUTPUT_VARIABLE RUSTC_VERSION
OUTPUT_STRIP_TRAILING_WHITESPACE
)
string(REGEX REPLACE ".*host: ([^ \n]*).*" "\\1"
CARGO_TARGET
${RUSTC_VERSION}
)
if(BUILD_TYPE_LOWER STREQUAL debug)
set(CARGO_BUILD_TYPE "debug")
set(CARGO_FLAG --target=${CARGO_TARGET})
else()
set(CARGO_BUILD_TYPE "release")
if (NOT RUSTC_VERSION MATCHES "nightly")
set(RUSTUP_TOOLCHAIN nightly)
endif()
set(RUSTFLAGS -C\ panic=abort)
set(CARGO_FLAG -Z build-std=std,panic_abort --release --target=${CARGO_TARGET})
endif()
set(CARGO_FEATURES "")
set(CARGO_BINARY_DIR "${CARGO_TARGET_DIR}/${CARGO_TARGET}/${CARGO_BUILD_TYPE}")
set(BINDINGS_NAME "${LIBRARY_NAME}_core")
configure_file(
${CMAKE_MODULE_PATH}/Cargo.toml.in
${CMAKE_SOURCE_DIR}/Cargo.toml
@ONLY
NEWLINE_STYLE LF
)
set(INCLUDE_GUARD_PREFIX "${SYMBOL_PREFIX}")
configure_file(
${CMAKE_MODULE_PATH}/cbindgen.toml.in
${CMAKE_SOURCE_DIR}/cbindgen.toml
@ONLY
NEWLINE_STYLE LF
)
set(CARGO_OUTPUT
${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
${CARGO_BINARY_DIR}/${CMAKE_STATIC_LIBRARY_PREFIX}${BINDINGS_NAME}${CMAKE_STATIC_LIBRARY_SUFFIX}
)
# \note cbindgen's naming behavior isn't fully configurable and it ignores
# `const fn` calls (https://github.com/eqrion/cbindgen/issues/252).
add_custom_command(
OUTPUT
${CARGO_OUTPUT}
COMMAND
# \note cbindgen won't regenerate its output header file after it's been removed but it will after its
# configuration file has been updated.
${CMAKE_COMMAND} -DCONDITION=NOT_EXISTS -P ${CMAKE_SOURCE_DIR}/cmake/file-touch.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h ${CMAKE_SOURCE_DIR}/cbindgen.toml
COMMAND
${CMAKE_COMMAND} -E env CARGO_TARGET_DIR=${CARGO_TARGET_DIR} CBINDGEN_TARGET_DIR=${CBINDGEN_TARGET_DIR} RUSTUP_TOOLCHAIN=${RUSTUP_TOOLCHAIN} RUSTFLAGS=${RUSTFLAGS} ${CARGO_CMD} build ${CARGO_FLAG} ${CARGO_FEATURES}
COMMAND
# Compensate for cbindgen's translation of consecutive uppercase letters to "ScreamingSnakeCase".
${CMAKE_COMMAND} -DMATCH_REGEX=A_M\([^_]+\)_ -DREPLACE_EXPR=AM_\\1_ -P ${CMAKE_SOURCE_DIR}/cmake/file-regex-replace.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
COMMAND
# Compensate for cbindgen ignoring `std:mem::size_of<usize>()` calls.
${CMAKE_COMMAND} -DMATCH_REGEX=USIZE_ -DREPLACE_EXPR=\+${CMAKE_SIZEOF_VOID_P} -P ${CMAKE_SOURCE_DIR}/cmake/file-regex-replace.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
MAIN_DEPENDENCY
src/lib.rs
DEPENDS
src/actor_id.rs
src/byte_span.rs
src/change.rs
src/doc.rs
src/doc/list.rs
src/doc/map.rs
src/doc/utils.rs
src/index.rs
src/item.rs
src/items.rs
src/obj.rs
src/result.rs
src/sync.rs
src/sync/have.rs
src/sync/message.rs
src/sync/state.rs
${CMAKE_SOURCE_DIR}/build.rs
${CMAKE_MODULE_PATH}/Cargo.toml.in
${CMAKE_MODULE_PATH}/cbindgen.toml.in
WORKING_DIRECTORY
${CMAKE_SOURCE_DIR}
COMMENT
"Producing the bindings' artifacts with Cargo..."
VERBATIM
)
add_custom_target(${BINDINGS_NAME}_artifacts ALL
DEPENDS ${CARGO_OUTPUT}
)
add_library(${BINDINGS_NAME} STATIC IMPORTED GLOBAL)
target_include_directories(${BINDINGS_NAME} INTERFACE "${CBINDGEN_INCLUDEDIR}")
set_target_properties(
${BINDINGS_NAME}
PROPERTIES
# \note Cargo writes a debug build into a nested directory instead of
# decorating its name.
DEBUG_POSTFIX ""
DEFINE_SYMBOL ""
IMPORTED_IMPLIB ""
IMPORTED_LOCATION "${CARGO_BINARY_DIR}/${CMAKE_STATIC_LIBRARY_PREFIX}${BINDINGS_NAME}${CMAKE_STATIC_LIBRARY_SUFFIX}"
IMPORTED_NO_SONAME "TRUE"
IMPORTED_SONAME ""
LINKER_LANGUAGE C
PUBLIC_HEADER "${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h"
SOVERSION "${PROJECT_VERSION_MAJOR}"
VERSION "${PROJECT_VERSION}"
# \note Cargo exports all of the symbols automatically.
WINDOWS_EXPORT_ALL_SYMBOLS "TRUE"
)
target_compile_definitions(${BINDINGS_NAME} INTERFACE $<TARGET_PROPERTY:${BINDINGS_NAME},DEFINE_SYMBOL>)
set(UTILS_SUBDIR "utils")
add_custom_command(
OUTPUT
${CBINDGEN_TARGET_DIR}/${UTILS_SUBDIR}/enum_string.h
${CMAKE_BINARY_DIR}/src/${UTILS_SUBDIR}/enum_string.c
COMMAND
${CMAKE_COMMAND} -DPROJECT_NAME=${PROJECT_NAME} -DLIBRARY_NAME=${LIBRARY_NAME} -DSUBDIR=${UTILS_SUBDIR} -P ${CMAKE_SOURCE_DIR}/cmake/enum-string-functions-gen.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h ${CBINDGEN_TARGET_DIR}/${UTILS_SUBDIR}/enum_string.h ${CMAKE_BINARY_DIR}/src/${UTILS_SUBDIR}/enum_string.c
MAIN_DEPENDENCY
${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
DEPENDS
${CMAKE_SOURCE_DIR}/cmake/enum-string-functions-gen.cmake
WORKING_DIRECTORY
${CMAKE_SOURCE_DIR}
COMMENT
"Generating the enum string functions with CMake..."
VERBATIM
)
add_custom_target(${LIBRARY_NAME}_utilities
DEPENDS ${CBINDGEN_TARGET_DIR}/${UTILS_SUBDIR}/enum_string.h
${CMAKE_BINARY_DIR}/src/${UTILS_SUBDIR}/enum_string.c
)
add_library(${LIBRARY_NAME})
target_compile_features(${LIBRARY_NAME} PRIVATE c_std_99)
set(CMAKE_THREAD_PREFER_PTHREAD TRUE)
set(THREADS_PREFER_PTHREAD_FLAG TRUE)
find_package(Threads REQUIRED)
set(LIBRARY_DEPENDENCIES Threads::Threads ${CMAKE_DL_LIBS})
if(WIN32)
list(APPEND LIBRARY_DEPENDENCIES Bcrypt userenv ws2_32)
else()
list(APPEND LIBRARY_DEPENDENCIES m)
endif()
target_link_libraries(${LIBRARY_NAME}
PUBLIC ${BINDINGS_NAME}
${LIBRARY_DEPENDENCIES}
)
# \note An imported library's INTERFACE_INCLUDE_DIRECTORIES property can't
# contain a non-existent path so its build-time include directory
# must be specified for all of its dependent targets instead.
target_include_directories(${LIBRARY_NAME}
PUBLIC "$<BUILD_INTERFACE:${CBINDGEN_INCLUDEDIR};${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}>"
"$<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}>"
)
add_dependencies(${LIBRARY_NAME} ${BINDINGS_NAME}_artifacts)
# Generate the configuration header.
math(EXPR INTEGER_PROJECT_VERSION_MAJOR "${PROJECT_VERSION_MAJOR} * 100000") math(EXPR INTEGER_PROJECT_VERSION_MAJOR "${PROJECT_VERSION_MAJOR} * 100000")
math(EXPR INTEGER_PROJECT_VERSION_MINOR "${PROJECT_VERSION_MINOR} * 100") math(EXPR INTEGER_PROJECT_VERSION_MINOR "${PROJECT_VERSION_MINOR} * 100")
math(EXPR INTEGER_PROJECT_VERSION_PATCH "${PROJECT_VERSION_PATCH}") math(EXPR INTEGER_PROJECT_VERSION_PATCH "${PROJECT_VERSION_PATCH}")
math(EXPR INTEGER_PROJECT_VERSION "${INTEGER_PROJECT_VERSION_MAJOR} + ${INTEGER_PROJECT_VERSION_MINOR} + ${INTEGER_PROJECT_VERSION_PATCH}") math(EXPR INTEGER_PROJECT_VERSION "${INTEGER_PROJECT_VERSION_MAJOR} + \
${INTEGER_PROJECT_VERSION_MINOR} + \
${INTEGER_PROJECT_VERSION_PATCH}")
configure_file( configure_file(
${CMAKE_MODULE_PATH}/config.h.in ${CMAKE_MODULE_PATH}/config.h.in
config.h ${CBINDGEN_TARGET_DIR}/config.h
@ONLY @ONLY
NEWLINE_STYLE LF NEWLINE_STYLE LF
) )
target_sources(${LIBRARY_NAME}
PRIVATE
src/${UTILS_SUBDIR}/result.c
src/${UTILS_SUBDIR}/stack_callback_data.c
src/${UTILS_SUBDIR}/stack.c
src/${UTILS_SUBDIR}/string.c
${CMAKE_BINARY_DIR}/src/${UTILS_SUBDIR}/enum_string.c
PUBLIC
FILE_SET api TYPE HEADERS
BASE_DIRS
${CBINDGEN_INCLUDEDIR}
${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}
FILES
${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
${CBINDGEN_TARGET_DIR}/${UTILS_SUBDIR}/enum_string.h
${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/result.h
${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/stack_callback_data.h
${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/stack.h
${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/string.h
INTERFACE
FILE_SET config TYPE HEADERS
BASE_DIRS
${CBINDGEN_INCLUDEDIR}
FILES
${CBINDGEN_TARGET_DIR}/config.h
)
install( install(
FILES ${CMAKE_BINARY_DIR}/config.h TARGETS ${LIBRARY_NAME}
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME} EXPORT ${PROJECT_NAME}-config
FILE_SET api
FILE_SET config
)
# \note Install the Cargo-built core bindings to enable direct linkage.
install(
FILES $<TARGET_PROPERTY:${BINDINGS_NAME},IMPORTED_LOCATION>
DESTINATION ${CMAKE_INSTALL_LIBDIR}
)
install(EXPORT ${PROJECT_NAME}-config
FILE ${PROJECT_NAME}-config.cmake
NAMESPACE "${PROJECT_NAME}::"
DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/${LIB}
) )
if(BUILD_TESTING) if(BUILD_TESTING)
@ -100,42 +300,6 @@ if(BUILD_TESTING)
enable_testing() enable_testing()
endif() endif()
add_subdirectory(docs)
add_subdirectory(examples EXCLUDE_FROM_ALL) add_subdirectory(examples EXCLUDE_FROM_ALL)
# Generate and install .cmake files
set(PROJECT_CONFIG_NAME "${PROJECT_NAME}-config")
set(PROJECT_CONFIG_VERSION_NAME "${PROJECT_CONFIG_NAME}-version")
write_basic_package_version_file(
${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_CONFIG_VERSION_NAME}.cmake
VERSION ${PROJECT_VERSION}
COMPATIBILITY ExactVersion
)
# The namespace label starts with the title-cased library name.
string(SUBSTRING ${LIBRARY_NAME} 0 1 NS_FIRST)
string(SUBSTRING ${LIBRARY_NAME} 1 -1 NS_REST)
string(TOUPPER ${NS_FIRST} NS_FIRST)
string(TOLOWER ${NS_REST} NS_REST)
string(CONCAT NAMESPACE ${NS_FIRST} ${NS_REST} "::")
# \note CMake doesn't automate the exporting of an imported library's targets
# so the package configuration script must do it.
configure_package_config_file(
${CMAKE_MODULE_PATH}/${PROJECT_CONFIG_NAME}.cmake.in
${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_CONFIG_NAME}.cmake
INSTALL_DESTINATION ${CMAKE_INSTALL_LIBDIR}/cmake/${PROJECT_NAME}
)
install(
FILES
${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_CONFIG_NAME}.cmake
${CMAKE_CURRENT_BINARY_DIR}/${PROJECT_CONFIG_VERSION_NAME}.cmake
DESTINATION
${CMAKE_INSTALL_LIBDIR}/cmake/${PROJECT_NAME}
)

View file

@ -7,8 +7,8 @@ license = "MIT"
rust-version = "1.57.0" rust-version = "1.57.0"
[lib] [lib]
name = "automerge" name = "automerge_core"
crate-type = ["cdylib", "staticlib"] crate-type = ["staticlib"]
bench = false bench = false
doc = false doc = false

View file

@ -1,22 +1,29 @@
automerge-c exposes an API to C that can either be used directly or as a basis # Overview
for other language bindings that have good support for calling into C functions.
# Building automerge-c exposes a C API that can either be used directly or as the basis
for other language bindings that have good support for calling C functions.
See the main README for instructions on getting your environment set up, then # Installing
you can use `./scripts/ci/cmake-build Release static` to build automerge-c.
It will output two files: See the main README for instructions on getting your environment set up and then
you can build the automerge-c library and install its constituent files within
a root directory of your choosing (e.g. "/usr/local") like so:
```shell
cmake -E make_directory automerge-c/build
cmake -S automerge-c -B automerge-c/build
cmake --build automerge-c/build
cmake --install automerge-c/build --prefix "/usr/local"
```
Installation is important because the name, location and structure of CMake's
out-of-source build subdirectory is subject to change based on the platform and
the release version; generated headers like `automerge-c/config.h` and
`automerge-c/utils/enum_string.h` are only sure to be found within their
installed locations.
- ./build/Cargo/target/include/automerge-c/automerge.h It's not obvious because they are versioned but the `Cargo.toml` and
- ./build/Cargo/target/release/libautomerge.a `cbindgen.toml` configuration files are also generated in order to ensure that
the project name, project version and library name that they contain match those
To use these in your application you must arrange for your C compiler to find specified within the top-level `CMakeLists.txt` file.
these files, either by moving them to the right location on your computer, or
by configuring the compiler to reference these directories.
- `export LDFLAGS=-L./build/Cargo/target/release -lautomerge`
- `export CFLAGS=-I./build/Cargo/target/include`
If you'd like to cross compile the library for different platforms you can do so If you'd like to cross compile the library for different platforms you can do so
using [cross](https://github.com/cross-rs/cross). For example: using [cross](https://github.com/cross-rs/cross). For example:
@ -25,134 +32,176 @@ using [cross](https://github.com/cross-rs/cross). For example:
This will output a shared library in the directory `rust/target/aarch64-unknown-linux-gnu/release/`. This will output a shared library in the directory `rust/target/aarch64-unknown-linux-gnu/release/`.
You can replace `aarch64-unknown-linux-gnu` with any [cross supported targets](https://github.com/cross-rs/cross#supported-targets). The targets below are known to work, though other targets are expected to work too: You can replace `aarch64-unknown-linux-gnu` with any
[cross supported targets](https://github.com/cross-rs/cross#supported-targets).
The targets below are known to work, though other targets are expected to work
too:
- `x86_64-apple-darwin` - `x86_64-apple-darwin`
- `aarch64-apple-darwin` - `aarch64-apple-darwin`
- `x86_64-unknown-linux-gnu` - `x86_64-unknown-linux-gnu`
- `aarch64-unknown-linux-gnu` - `aarch64-unknown-linux-gnu`
As a caveat, the header file is currently 32/64-bit dependant. You can re-use it As a caveat, CMake generates the `automerge.h` header file in terms of the
for all 64-bit architectures, but you must generate a specific header for 32-bit processor architecture of the computer on which it was built so, for example,
targets. don't use a header generated for a 64-bit processor if your target is a 32-bit
processor.
# Usage # Usage
For full reference, read through `automerge.h`, or to get started quickly look You can build and view the C API's HTML reference documentation like so:
at the ```shell
cmake -E make_directory automerge-c/build
cmake -S automerge-c -B automerge-c/build
cmake --build automerge-c/build --target automerge_docs
firefox automerge-c/build/src/html/index.html
```
To get started quickly, look at the
[examples](https://github.com/automerge/automerge-rs/tree/main/rust/automerge-c/examples). [examples](https://github.com/automerge/automerge-rs/tree/main/rust/automerge-c/examples).
Almost all operations in automerge-c act on an AMdoc struct which you can get Almost all operations in automerge-c act on an Automerge document
from `AMcreate()` or `AMload()`. Operations on a given doc are not thread safe (`AMdoc` struct) which is structurally similar to a JSON document.
so you must use a mutex or similar to avoid calling more than one function with
the same AMdoc pointer concurrently.
As with all functions that either allocate memory, or could fail if given You can get a document by calling either `AMcreate()` or `AMload()`. Operations
invalid input, `AMcreate()` returns an `AMresult`. The `AMresult` contains the on a given document are not thread-safe so you must use a mutex or similar to
returned doc (or error message), and must be freed with `AMfree()` after you are avoid calling more than one function on the same one concurrently.
done to avoid leaking memory.
A C API function that could succeed or fail returns a result (`AMresult` struct)
containing a status code (`AMstatus` enum) and either a sequence of at least one
item (`AMitem` struct) or a read-only view onto a UTF-8 error message string
(`AMbyteSpan` struct).
An item contains up to three components: an index within its parent object
(`AMbyteSpan` struct or `size_t`), a unique identifier (`AMobjId` struct) and a
value.
The result of a successful function call that doesn't produce any values will
contain a single item that is void (`AM_VAL_TYPE_VOID`).
A returned result **must** be passed to `AMresultFree()` once the item(s) or
error message it contains is no longer needed in order to avoid a memory leak.
``` ```
#include <automerge-c/automerge.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h>
#include <automerge-c/automerge.h>
#include <automerge-c/utils/string.h>
int main(int argc, char** argv) { int main(int argc, char** argv) {
AMresult *docResult = AMcreate(NULL); AMresult *docResult = AMcreate(NULL);
if (AMresultStatus(docResult) != AM_STATUS_OK) { if (AMresultStatus(docResult) != AM_STATUS_OK) {
printf("failed to create doc: %s", AMerrorMessage(docResult).src); char* const err_msg = AMstrdup(AMresultError(docResult), NULL);
printf("failed to create doc: %s", err_msg);
free(err_msg);
goto cleanup; goto cleanup;
} }
AMdoc *doc = AMresultValue(docResult).doc; AMdoc *doc;
AMitemToDoc(AMresultItem(docResult), &doc);
// useful code goes here! // useful code goes here!
cleanup: cleanup:
AMfree(docResult); AMresultFree(docResult);
} }
``` ```
If you are writing code in C directly, you can use the `AMpush()` helper If you are writing an application in C, the `AMstackItem()`, `AMstackItems()`
function to reduce the boilerplate of error handling and freeing for you (see and `AMstackResult()` functions enable the lifetimes of anonymous results to be
examples/quickstart.c). centrally managed and allow the same validation logic to be reused without
relying upon the `goto` statement (see examples/quickstart.c).
If you are wrapping automerge-c in another language, particularly one that has a If you are wrapping automerge-c in another language, particularly one that has a
garbage collector, you can call `AMfree` within a finalizer to ensure that memory garbage collector, you can call the `AMresultFree()` function within a finalizer
is reclaimed when it is no longer needed. to ensure that memory is reclaimed when it is no longer needed.
An AMdoc wraps an automerge document which are very similar to JSON documents. Automerge documents consist of a mutable root which is always a map from string
Automerge documents consist of a mutable root, which is always a map from string keys to values. A value can be one of the following types:
keys to values. Values can have the following types:
- A number of type double / int64_t / uint64_t - A number of type double / int64_t / uint64_t
- An explicit true / false / nul - An explicit true / false / null
- An immutable utf-8 string (AMbyteSpan) - An immutable UTF-8 string (`AMbyteSpan`).
- An immutable array of arbitrary bytes (AMbyteSpan) - An immutable array of arbitrary bytes (`AMbyteSpan`).
- A mutable map from string keys to values (AMmap) - A mutable map from string keys to values.
- A mutable list of values (AMlist) - A mutable list of values.
- A mutable string (AMtext) - A mutable UTF-8 string.
If you read from a location in the document with no value a value with If you read from a location in the document with no value, an item with type
`.tag == AM_VALUE_VOID` will be returned, but you cannot write such a value explicitly. `AM_VAL_TYPE_VOID` will be returned, but you cannot write such a value
explicitly.
Under the hood, automerge references mutable objects by the internal object id, Under the hood, automerge references a mutable object by its object identifier
and `AM_ROOT` is always the object id of the root value. where `AM_ROOT` signifies a document's root map object.
There is a function to put each type of value into either a map or a list, and a There are functions to put each type of value into either a map or a list, and
function to read the current value from a list. As (in general) collaborators functions to read the current or a historical value from a map or a list. As (in general) collaborators
may edit the document at any time, you cannot guarantee that the type of the may edit the document at any time, you cannot guarantee that the type of the
value at a given part of the document will stay the same. As a result reading value at a given part of the document will stay the same. As a result, reading
from the document will return an `AMvalue` union that you can inspect to from the document will return an `AMitem` struct that you can inspect to
determine its type. determine the type of value that it contains.
Strings in automerge-c are represented using an `AMbyteSpan` which contains a Strings in automerge-c are represented using an `AMbyteSpan` which contains a
pointer and a length. Strings must be valid utf-8 and may contain null bytes. pointer and a length. Strings must be valid UTF-8 and may contain NUL (`0`)
As a convenience you can use `AMstr()` to get the representation of a characters.
null-terminated C string as an `AMbyteSpan`. For your convenience, you can call `AMstr()` to get the `AMbyteSpan` struct
equivalent of a null-terminated byte string or `AMstrdup()` to get the
representation of an `AMbyteSpan` struct as a null-terminated byte string
wherein its NUL characters have been removed/replaced as you choose.
Putting all of that together, to read and write from the root of the document Putting all of that together, to read and write from the root of the document
you can do this: you can do this:
``` ```
#include <automerge-c/automerge.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h>
#include <automerge-c/automerge.h>
#include <automerge-c/utils/string.h>
int main(int argc, char** argv) { int main(int argc, char** argv) {
// ...previous example... // ...previous example...
AMdoc *doc = AMresultValue(docResult).doc; AMdoc *doc;
AMitemToDoc(AMresultItem(docResult), &doc);
AMresult *putResult = AMmapPutStr(doc, AM_ROOT, AMstr("key"), AMstr("value")); AMresult *putResult = AMmapPutStr(doc, AM_ROOT, AMstr("key"), AMstr("value"));
if (AMresultStatus(putResult) != AM_STATUS_OK) { if (AMresultStatus(putResult) != AM_STATUS_OK) {
printf("failed to put: %s", AMerrorMessage(putResult).src); char* const err_msg = AMstrdup(AMresultError(putResult), NULL);
printf("failed to put: %s", err_msg);
free(err_msg);
goto cleanup; goto cleanup;
} }
AMresult *getResult = AMmapGet(doc, AM_ROOT, AMstr("key"), NULL); AMresult *getResult = AMmapGet(doc, AM_ROOT, AMstr("key"), NULL);
if (AMresultStatus(getResult) != AM_STATUS_OK) { if (AMresultStatus(getResult) != AM_STATUS_OK) {
printf("failed to get: %s", AMerrorMessage(getResult).src); char* const err_msg = AMstrdup(AMresultError(putResult), NULL);
printf("failed to get: %s", err_msg);
free(err_msg);
goto cleanup; goto cleanup;
} }
AMvalue got = AMresultValue(getResult); AMbyteSpan got;
if (got.tag != AM_VALUE_STR) { if (AMitemToStr(AMresultItem(getResult), &got)) {
char* const c_str = AMstrdup(got, NULL);
printf("Got %zu-character string \"%s\"", got.count, c_str);
free(c_str);
} else {
printf("expected to read a string!"); printf("expected to read a string!");
goto cleanup; goto cleanup;
} }
printf("Got %zu-character string `%s`", got.str.count, got.str.src);
cleanup: cleanup:
AMfree(getResult); AMresultFree(getResult);
AMfree(putResult); AMresultFree(putResult);
AMfree(docResult); AMresultFree(docResult);
} }
``` ```
Functions that do not return an `AMresult` (for example `AMmapItemValue()`) do Functions that do not return an `AMresult` (for example `AMitemKey()`) do
not allocate memory, but continue to reference memory that was previously not allocate memory but rather reference memory that was previously
allocated. It's thus important to keep the original `AMresult` alive (in this allocated. It's therefore important to keep the original `AMresult` alive (in
case the one returned by `AMmapRange()`) until after you are done with the return this case the one returned by `AMmapRange()`) until after you are finished with
values of these functions. the items that it contains. However, the memory for an individual `AMitem` can
be shared with a new `AMresult` by calling `AMitemResult()` on it. In other
words, a select group of items can be filtered out of a collection and only each
one's corresponding `AMresult` must be kept alive from that point forward; the
originating collection's `AMresult` can be safely freed.
Beyond that, good luck! Beyond that, good luck!

View file

@ -1,7 +1,7 @@
after_includes = """\n after_includes = """\n
/** /**
* \\defgroup enumerations Public Enumerations * \\defgroup enumerations Public Enumerations
Symbolic names for integer constants. * Symbolic names for integer constants.
*/ */
/** /**
@ -12,21 +12,23 @@ after_includes = """\n
#define AM_ROOT NULL #define AM_ROOT NULL
/** /**
* \\memberof AMchangeHash * \\memberof AMdoc
* \\def AM_CHANGE_HASH_SIZE * \\def AM_CHANGE_HASH_SIZE
* \\brief The count of bytes in a change hash. * \\brief The count of bytes in a change hash.
*/ */
#define AM_CHANGE_HASH_SIZE 32 #define AM_CHANGE_HASH_SIZE 32
""" """
autogen_warning = "/* Warning, this file is autogenerated by cbindgen. Don't modify this manually. */" autogen_warning = """
/**
* \\file
* \\brief All constants, functions and types in the core Automerge C API.
*
* \\warning This file is auto-generated by cbindgen.
*/
"""
documentation = true documentation = true
documentation_style = "doxy" documentation_style = "doxy"
header = """ include_guard = "AUTOMERGE_C_H"
/** \\file
* All constants, functions and types in the Automerge library's C API.
*/
"""
include_guard = "AUTOMERGE_H"
includes = [] includes = []
language = "C" language = "C"
line_length = 140 line_length = 140

View file

@ -0,0 +1,22 @@
[package]
name = "@PROJECT_NAME@"
version = "@PROJECT_VERSION@"
authors = ["Orion Henry <orion.henry@gmail.com>", "Jason Kankiewicz <jason.kankiewicz@gmail.com>"]
edition = "2021"
license = "MIT"
rust-version = "1.57.0"
[lib]
name = "@BINDINGS_NAME@"
crate-type = ["staticlib"]
bench = false
doc = false
[dependencies]
@LIBRARY_NAME@ = { path = "../@LIBRARY_NAME@" }
hex = "^0.4.3"
libc = "^0.2"
smol_str = "^0.1.21"
[build-dependencies]
cbindgen = "^0.24"

View file

@ -0,0 +1,48 @@
after_includes = """\n
/**
* \\defgroup enumerations Public Enumerations
* Symbolic names for integer constants.
*/
/**
* \\memberof AMdoc
* \\def AM_ROOT
* \\brief The root object of a document.
*/
#define AM_ROOT NULL
/**
* \\memberof AMdoc
* \\def AM_CHANGE_HASH_SIZE
* \\brief The count of bytes in a change hash.
*/
#define AM_CHANGE_HASH_SIZE 32
"""
autogen_warning = """
/**
* \\file
* \\brief All constants, functions and types in the core Automerge C API.
*
* \\warning This file is auto-generated by cbindgen.
*/
"""
documentation = true
documentation_style = "doxy"
include_guard = "@INCLUDE_GUARD_PREFIX@_H"
includes = []
language = "C"
line_length = 140
no_includes = true
style = "both"
sys_includes = ["stdbool.h", "stddef.h", "stdint.h", "time.h"]
usize_is_size_t = true
[enum]
derive_const_casts = true
enum_class = true
must_use = "MUST_USE_ENUM"
prefix_with_name = true
rename_variants = "ScreamingSnakeCase"
[export]
item_types = ["constants", "enums", "functions", "opaque", "structs", "typedefs"]

View file

@ -1,14 +1,35 @@
#ifndef @SYMBOL_PREFIX@_CONFIG_H #ifndef @INCLUDE_GUARD_PREFIX@_CONFIG_H
#define @SYMBOL_PREFIX@_CONFIG_H #define @INCLUDE_GUARD_PREFIX@_CONFIG_H
/**
/* This header is auto-generated by CMake. */ * \file
* \brief Configuration pararameters defined by the build system.
*
* \warning This file is auto-generated by CMake.
*/
/**
* \def @SYMBOL_PREFIX@_VERSION
* \brief Denotes a semantic version of the form {MAJOR}{MINOR}{PATCH} as three,
* two-digit decimal numbers without leading zeros (e.g. 100 is 0.1.0).
*/
#define @SYMBOL_PREFIX@_VERSION @INTEGER_PROJECT_VERSION@ #define @SYMBOL_PREFIX@_VERSION @INTEGER_PROJECT_VERSION@
/**
* \def @SYMBOL_PREFIX@_MAJOR_VERSION
* \brief Denotes a semantic major version as a decimal number.
*/
#define @SYMBOL_PREFIX@_MAJOR_VERSION (@SYMBOL_PREFIX@_VERSION / 100000) #define @SYMBOL_PREFIX@_MAJOR_VERSION (@SYMBOL_PREFIX@_VERSION / 100000)
/**
* \def @SYMBOL_PREFIX@_MINOR_VERSION
* \brief Denotes a semantic minor version as a decimal number.
*/
#define @SYMBOL_PREFIX@_MINOR_VERSION ((@SYMBOL_PREFIX@_VERSION / 100) % 1000) #define @SYMBOL_PREFIX@_MINOR_VERSION ((@SYMBOL_PREFIX@_VERSION / 100) % 1000)
/**
* \def @SYMBOL_PREFIX@_PATCH_VERSION
* \brief Denotes a semantic patch version as a decimal number.
*/
#define @SYMBOL_PREFIX@_PATCH_VERSION (@SYMBOL_PREFIX@_VERSION % 100) #define @SYMBOL_PREFIX@_PATCH_VERSION (@SYMBOL_PREFIX@_VERSION % 100)
#endif /* @SYMBOL_PREFIX@_CONFIG_H */ #endif /* @INCLUDE_GUARD_PREFIX@_CONFIG_H */

View file

@ -0,0 +1,183 @@
# This CMake script is used to generate a header and a source file for utility
# functions that convert the tags of generated enum types into strings and
# strings into the tags of generated enum types.
cmake_minimum_required(VERSION 3.23 FATAL_ERROR)
# Seeks the starting line of the source enum's declaration.
macro(seek_enum_mode)
if (line MATCHES "^(typedef[ \t]+)?enum ")
string(REGEX REPLACE "^enum ([0-9a-zA-Z_]+).*$" "\\1" enum_name "${line}")
set(mode "read_tags")
endif()
endmacro()
# Scans the input for the current enum's tags.
macro(read_tags_mode)
if(line MATCHES "^}")
set(mode "generate")
elseif(line MATCHES "^[A-Z0-9_]+.*$")
string(REGEX REPLACE "^([A-Za-z0-9_]+).*$" "\\1" tmp "${line}")
list(APPEND enum_tags "${tmp}")
endif()
endmacro()
macro(write_header_file)
# Generate a to-string function declaration.
list(APPEND header_body
"/**\n"
" * \\ingroup enumerations\n"
" * \\brief Gets the string representation of an `${enum_name}` enum tag.\n"
" *\n"
" * \\param[in] tag An `${enum_name}` enum tag.\n"
" * \\return A null-terminated byte string.\n"
" */\n"
"char const* ${enum_name}ToString(${enum_name} const tag)\;\n"
"\n")
# Generate a from-string function declaration.
list(APPEND header_body
"/**\n"
" * \\ingroup enumerations\n"
" * \\brief Gets an `${enum_name}` enum tag from its string representation.\n"
" *\n"
" * \\param[out] dest An `${enum_name}` enum tag pointer.\n"
" * \\param[in] src A null-terminated byte string.\n"
" * \\return `true` if \\p src matches the string representation of an\n"
" * `${enum_name}` enum tag, `false` otherwise.\n"
" */\n"
"bool ${enum_name}FromString(${enum_name}* dest, char const* const src)\;\n"
"\n")
endmacro()
macro(write_source_file)
# Generate a to-string function implementation.
list(APPEND source_body
"char const* ${enum_name}ToString(${enum_name} const tag) {\n"
" switch (tag) {\n"
" default:\n"
" return \"???\"\;\n")
foreach(label IN LISTS enum_tags)
list(APPEND source_body
" case ${label}:\n"
" return \"${label}\"\;\n")
endforeach()
list(APPEND source_body
" }\n"
"}\n"
"\n")
# Generate a from-string function implementation.
list(APPEND source_body
"bool ${enum_name}FromString(${enum_name}* dest, char const* const src) {\n")
foreach(label IN LISTS enum_tags)
list(APPEND source_body
" if (!strcmp(src, \"${label}\")) {\n"
" *dest = ${label}\;\n"
" return true\;\n"
" }\n")
endforeach()
list(APPEND source_body
" return false\;\n"
"}\n"
"\n")
endmacro()
function(main)
set(header_body "")
# File header and includes.
list(APPEND header_body
"#ifndef ${include_guard}\n"
"#define ${include_guard}\n"
"/**\n"
" * \\file\n"
" * \\brief Utility functions for converting enum tags into null-terminated\n"
" * byte strings and vice versa.\n"
" *\n"
" * \\warning This file is auto-generated by CMake.\n"
" */\n"
"\n"
"#include <stdbool.h>\n"
"\n"
"#include <${library_include}>\n"
"\n")
set(source_body "")
# File includes.
list(APPEND source_body
"/** \\warning This file is auto-generated by CMake. */\n"
"\n"
"#include \"stdio.h\"\n"
"#include \"string.h\"\n"
"\n"
"#include <${header_include}>\n"
"\n")
set(enum_name "")
set(enum_tags "")
set(mode "seek_enum")
file(STRINGS "${input_path}" lines)
foreach(line IN LISTS lines)
string(REGEX REPLACE "^(.+)(//.*)?" "\\1" line "${line}")
string(STRIP "${line}" line)
if(mode STREQUAL "seek_enum")
seek_enum_mode()
elseif(mode STREQUAL "read_tags")
read_tags_mode()
else()
# The end of the enum declaration was reached.
if(NOT enum_name)
# The end of the file was reached.
return()
endif()
if(NOT enum_tags)
message(FATAL_ERROR "No tags found for `${enum_name}`.")
endif()
string(TOLOWER "${enum_name}" output_stem_prefix)
string(CONCAT output_stem "${output_stem_prefix}" "_string")
cmake_path(REPLACE_EXTENSION output_stem "h" OUTPUT_VARIABLE output_header_basename)
write_header_file()
write_source_file()
set(enum_name "")
set(enum_tags "")
set(mode "seek_enum")
endif()
endforeach()
# File footer.
list(APPEND header_body
"#endif /* ${include_guard} */\n")
message(STATUS "Generating header file \"${output_header_path}\"...")
file(WRITE "${output_header_path}" ${header_body})
message(STATUS "Generating source file \"${output_source_path}\"...")
file(WRITE "${output_source_path}" ${source_body})
endfunction()
if(NOT DEFINED PROJECT_NAME)
message(FATAL_ERROR "Variable PROJECT_NAME is not defined.")
elseif(NOT DEFINED LIBRARY_NAME)
message(FATAL_ERROR "Variable LIBRARY_NAME is not defined.")
elseif(NOT DEFINED SUBDIR)
message(FATAL_ERROR "Variable SUBDIR is not defined.")
elseif(${CMAKE_ARGC} LESS 9)
message(FATAL_ERROR "Too few arguments.")
elseif(${CMAKE_ARGC} GREATER 10)
message(FATAL_ERROR "Too many arguments.")
elseif(NOT EXISTS ${CMAKE_ARGV5})
message(FATAL_ERROR "Input header \"${CMAKE_ARGV7}\" not found.")
endif()
cmake_path(CONVERT "${CMAKE_ARGV7}" TO_CMAKE_PATH_LIST input_path NORMALIZE)
cmake_path(CONVERT "${CMAKE_ARGV8}" TO_CMAKE_PATH_LIST output_header_path NORMALIZE)
cmake_path(CONVERT "${CMAKE_ARGV9}" TO_CMAKE_PATH_LIST output_source_path NORMALIZE)
string(TOLOWER "${PROJECT_NAME}" project_root)
cmake_path(CONVERT "${SUBDIR}" TO_CMAKE_PATH_LIST project_subdir NORMALIZE)
string(TOLOWER "${project_subdir}" project_subdir)
string(TOLOWER "${LIBRARY_NAME}" library_stem)
cmake_path(REPLACE_EXTENSION library_stem "h" OUTPUT_VARIABLE library_basename)
string(JOIN "/" library_include "${project_root}" "${library_basename}")
string(TOUPPER "${PROJECT_NAME}" project_name_upper)
string(TOUPPER "${project_subdir}" include_guard_infix)
string(REGEX REPLACE "/" "_" include_guard_infix "${include_guard_infix}")
string(REGEX REPLACE "-" "_" include_guard_prefix "${project_name_upper}")
string(JOIN "_" include_guard_prefix "${include_guard_prefix}" "${include_guard_infix}")
string(JOIN "/" output_header_prefix "${project_root}" "${project_subdir}")
cmake_path(GET output_header_path STEM output_header_stem)
string(TOUPPER "${output_header_stem}" include_guard_stem)
string(JOIN "_" include_guard "${include_guard_prefix}" "${include_guard_stem}" "H")
cmake_path(GET output_header_path FILENAME output_header_basename)
string(JOIN "/" header_include "${output_header_prefix}" "${output_header_basename}")
main()

View file

@ -1,4 +1,6 @@
cmake_minimum_required(VERSION 3.18 FATAL_ERROR) # This CMake script is used to perform string substitutions within a generated
# file.
cmake_minimum_required(VERSION 3.23 FATAL_ERROR)
if(NOT DEFINED MATCH_REGEX) if(NOT DEFINED MATCH_REGEX)
message(FATAL_ERROR "Variable \"MATCH_REGEX\" is not defined.") message(FATAL_ERROR "Variable \"MATCH_REGEX\" is not defined.")

View file

@ -1,4 +1,6 @@
cmake_minimum_required(VERSION 3.18 FATAL_ERROR) # This CMake script is used to force Cargo to regenerate the header file for the
# core bindings after the out-of-source build directory has been cleaned.
cmake_minimum_required(VERSION 3.23 FATAL_ERROR)
if(NOT DEFINED CONDITION) if(NOT DEFINED CONDITION)
message(FATAL_ERROR "Variable \"CONDITION\" is not defined.") message(FATAL_ERROR "Variable \"CONDITION\" is not defined.")

View file

@ -0,0 +1,35 @@
find_package(Doxygen OPTIONAL_COMPONENTS dot)
if(DOXYGEN_FOUND)
set(DOXYGEN_ALIASES "installed_headerfile=\\headerfile ${LIBRARY_NAME}.h <${PROJECT_NAME}/${LIBRARY_NAME}.h>")
set(DOXYGEN_GENERATE_LATEX YES)
set(DOXYGEN_PDF_HYPERLINKS YES)
set(DOXYGEN_PROJECT_LOGO "${CMAKE_CURRENT_SOURCE_DIR}/img/brandmark.png")
set(DOXYGEN_SORT_BRIEF_DOCS YES)
set(DOXYGEN_USE_MDFILE_AS_MAINPAGE "${CMAKE_SOURCE_DIR}/README.md")
doxygen_add_docs(
${LIBRARY_NAME}_docs
"${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h"
"${CBINDGEN_TARGET_DIR}/config.h"
"${CBINDGEN_TARGET_DIR}/${UTILS_SUBDIR}/enum_string.h"
"${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/result.h"
"${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/stack_callback_data.h"
"${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/stack.h"
"${CMAKE_SOURCE_DIR}/${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}/${UTILS_SUBDIR}/string.h"
"${CMAKE_SOURCE_DIR}/README.md"
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
COMMENT "Producing documentation with Doxygen..."
)
# \note A Doxygen input file isn't a file-level dependency so the Doxygen
# command must instead depend upon a target that either outputs the
# file or depends upon it also or it will just output an error message
# when it can't be found.
add_dependencies(${LIBRARY_NAME}_docs ${BINDINGS_NAME}_artifacts ${LIBRARY_NAME}_utilities)
endif()

View file

Before

Width:  |  Height:  |  Size: 1.4 KiB

After

Width:  |  Height:  |  Size: 1.4 KiB

View file

@ -1,41 +1,39 @@
cmake_minimum_required(VERSION 3.18 FATAL_ERROR)
add_executable( add_executable(
example_quickstart ${LIBRARY_NAME}_quickstart
quickstart.c quickstart.c
) )
set_target_properties(example_quickstart PROPERTIES LINKER_LANGUAGE C) set_target_properties(${LIBRARY_NAME}_quickstart PROPERTIES LINKER_LANGUAGE C)
# \note An imported library's INTERFACE_INCLUDE_DIRECTORIES property can't # \note An imported library's INTERFACE_INCLUDE_DIRECTORIES property can't
# contain a non-existent path so its build-time include directory # contain a non-existent path so its build-time include directory
# must be specified for all of its dependent targets instead. # must be specified for all of its dependent targets instead.
target_include_directories( target_include_directories(
example_quickstart ${LIBRARY_NAME}_quickstart
PRIVATE "$<BUILD_INTERFACE:${CBINDGEN_INCLUDEDIR}>" PRIVATE "$<BUILD_INTERFACE:${CBINDGEN_INCLUDEDIR}>"
) )
target_link_libraries(example_quickstart PRIVATE ${LIBRARY_NAME}) target_link_libraries(${LIBRARY_NAME}_quickstart PRIVATE ${LIBRARY_NAME})
add_dependencies(example_quickstart ${LIBRARY_NAME}_artifacts) add_dependencies(${LIBRARY_NAME}_quickstart ${BINDINGS_NAME}_artifacts)
if(BUILD_SHARED_LIBS AND WIN32) if(BUILD_SHARED_LIBS AND WIN32)
add_custom_command( add_custom_command(
TARGET example_quickstart TARGET ${LIBRARY_NAME}_quickstart
POST_BUILD POST_BUILD
COMMAND ${CMAKE_COMMAND} -E copy_if_different COMMAND ${CMAKE_COMMAND} -E copy_if_different
${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_${CMAKE_BUILD_TYPE}_POSTFIX}${CMAKE_SHARED_LIBRARY_SUFFIX} ${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_${CMAKE_BUILD_TYPE}_POSTFIX}${CMAKE_SHARED_LIBRARY_SUFFIX}
${CMAKE_CURRENT_BINARY_DIR} ${CMAKE_BINARY_DIR}
COMMENT "Copying the DLL built by Cargo into the examples directory..." COMMENT "Copying the DLL built by Cargo into the examples directory..."
VERBATIM VERBATIM
) )
endif() endif()
add_custom_command( add_custom_command(
TARGET example_quickstart TARGET ${LIBRARY_NAME}_quickstart
POST_BUILD POST_BUILD
COMMAND COMMAND
example_quickstart ${LIBRARY_NAME}_quickstart
COMMENT COMMENT
"Running the example quickstart..." "Running the example quickstart..."
VERBATIM VERBATIM

View file

@ -5,5 +5,5 @@
```shell ```shell
cmake -E make_directory automerge-c/build cmake -E make_directory automerge-c/build
cmake -S automerge-c -B automerge-c/build cmake -S automerge-c -B automerge-c/build
cmake --build automerge-c/build --target example_quickstart cmake --build automerge-c/build --target automerge_quickstart
``` ```

View file

@ -3,152 +3,127 @@
#include <string.h> #include <string.h>
#include <automerge-c/automerge.h> #include <automerge-c/automerge.h>
#include <automerge-c/utils/enum_string.h>
#include <automerge-c/utils/stack.h>
#include <automerge-c/utils/stack_callback_data.h>
#include <automerge-c/utils/string.h>
static void abort_cb(AMresultStack**, uint8_t); static bool abort_cb(AMstack**, void*);
/** /**
* \brief Based on https://automerge.github.io/docs/quickstart * \brief Based on https://automerge.github.io/docs/quickstart
*/ */
int main(int argc, char** argv) { int main(int argc, char** argv) {
AMresultStack* stack = NULL; AMstack* stack = NULL;
AMdoc* const doc1 = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, abort_cb).doc; AMdoc* doc1;
AMobjId const* const cards = AMpush(&stack, AMitemToDoc(AMstackItem(&stack, AMcreate(NULL), abort_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc1);
AMmapPutObject(doc1, AM_ROOT, AMstr("cards"), AM_OBJ_TYPE_LIST), AMobjId const* const cards =
AM_VALUE_OBJ_ID, AMitemObjId(AMstackItem(&stack, AMmapPutObject(doc1, AM_ROOT, AMstr("cards"), AM_OBJ_TYPE_LIST), abort_cb,
abort_cb).obj_id; AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
AMobjId const* const card1 = AMpush(&stack, AMobjId const* const card1 =
AMlistPutObject(doc1, cards, SIZE_MAX, true, AM_OBJ_TYPE_MAP), AMitemObjId(AMstackItem(&stack, AMlistPutObject(doc1, cards, SIZE_MAX, true, AM_OBJ_TYPE_MAP), abort_cb,
AM_VALUE_OBJ_ID, AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
abort_cb).obj_id; AMstackItem(NULL, AMmapPutStr(doc1, card1, AMstr("title"), AMstr("Rewrite everything in Clojure")), abort_cb,
AMfree(AMmapPutStr(doc1, card1, AMstr("title"), AMstr("Rewrite everything in Clojure"))); AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMmapPutBool(doc1, card1, AMstr("done"), false)); AMstackItem(NULL, AMmapPutBool(doc1, card1, AMstr("done"), false), abort_cb, AMexpect(AM_VAL_TYPE_VOID));
AMobjId const* const card2 = AMpush(&stack, AMobjId const* const card2 =
AMlistPutObject(doc1, cards, SIZE_MAX, true, AM_OBJ_TYPE_MAP), AMitemObjId(AMstackItem(&stack, AMlistPutObject(doc1, cards, SIZE_MAX, true, AM_OBJ_TYPE_MAP), abort_cb,
AM_VALUE_OBJ_ID, AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
abort_cb).obj_id; AMstackItem(NULL, AMmapPutStr(doc1, card2, AMstr("title"), AMstr("Rewrite everything in Haskell")), abort_cb,
AMfree(AMmapPutStr(doc1, card2, AMstr("title"), AMstr("Rewrite everything in Haskell"))); AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMmapPutBool(doc1, card2, AMstr("done"), false)); AMstackItem(NULL, AMmapPutBool(doc1, card2, AMstr("done"), false), abort_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMcommit(doc1, AMstr("Add card"), NULL)); AMstackItem(NULL, AMcommit(doc1, AMstr("Add card"), NULL), abort_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMdoc* doc2 = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, abort_cb).doc; AMdoc* doc2;
AMfree(AMmerge(doc2, doc1)); AMitemToDoc(AMstackItem(&stack, AMcreate(NULL), abort_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc2);
AMstackItem(NULL, AMmerge(doc2, doc1), abort_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMbyteSpan const binary = AMpush(&stack, AMsave(doc1), AM_VALUE_BYTES, abort_cb).bytes; AMbyteSpan binary;
doc2 = AMpush(&stack, AMload(binary.src, binary.count), AM_VALUE_DOC, abort_cb).doc; AMitemToBytes(AMstackItem(&stack, AMsave(doc1), abort_cb, AMexpect(AM_VAL_TYPE_BYTES)), &binary);
AMitemToDoc(AMstackItem(&stack, AMload(binary.src, binary.count), abort_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc2);
AMfree(AMmapPutBool(doc1, card1, AMstr("done"), true)); AMstackItem(NULL, AMmapPutBool(doc1, card1, AMstr("done"), true), abort_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMcommit(doc1, AMstr("Mark card as done"), NULL)); AMstackItem(NULL, AMcommit(doc1, AMstr("Mark card as done"), NULL), abort_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMfree(AMlistDelete(doc2, cards, 0)); AMstackItem(NULL, AMlistDelete(doc2, cards, 0), abort_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMcommit(doc2, AMstr("Delete card"), NULL)); AMstackItem(NULL, AMcommit(doc2, AMstr("Delete card"), NULL), abort_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMfree(AMmerge(doc1, doc2)); AMstackItem(NULL, AMmerge(doc1, doc2), abort_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMchanges changes = AMpush(&stack, AMgetChanges(doc1, NULL), AM_VALUE_CHANGES, abort_cb).changes; AMitems changes = AMstackItems(&stack, AMgetChanges(doc1, NULL), abort_cb, AMexpect(AM_VAL_TYPE_CHANGE));
AMchange const* change = NULL; AMitem* item = NULL;
while ((change = AMchangesNext(&changes, 1)) != NULL) { while ((item = AMitemsNext(&changes, 1)) != NULL) {
AMbyteSpan const change_hash = AMchangeHash(change); AMchange const* change;
AMchangeHashes const heads = AMpush(&stack, AMitemToChange(item, &change);
AMchangeHashesInit(&change_hash, 1), AMitems const heads = AMstackItems(&stack, AMitemFromChangeHash(AMchangeHash(change)), abort_cb,
AM_VALUE_CHANGE_HASHES, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
abort_cb).change_hashes; char* const c_msg = AMstrdup(AMchangeMessage(change), NULL);
AMbyteSpan const msg = AMchangeMessage(change); printf("%s %zu\n", c_msg, AMobjSize(doc1, cards, &heads));
char* const c_msg = calloc(1, msg.count + 1);
strncpy(c_msg, msg.src, msg.count);
printf("%s %ld\n", c_msg, AMobjSize(doc1, cards, &heads));
free(c_msg); free(c_msg);
} }
AMfreeStack(&stack); AMstackFree(&stack);
} }
static char const* discriminant_suffix(AMvalueVariant const);
/** /**
* \brief Prints an error message to `stderr`, deallocates all results in the * \brief Examines the result at the top of the given stack and, if it's
* given stack and exits. * invalid, prints an error message to `stderr`, deallocates all results
* in the stack and exits.
* *
* \param[in,out] stack A pointer to a pointer to an `AMresultStack` struct. * \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] discriminant An `AMvalueVariant` enum tag. * \param[in] data A pointer to an owned `AMstackCallbackData` struct or `NULL`.
* \pre \p stack` != NULL`. * \return `true` if the top `AMresult` in \p stack is valid, `false` otherwise.
* \post `*stack == NULL`. * \pre \p stack `!= NULL`.
*/ */
static void abort_cb(AMresultStack** stack, uint8_t discriminant) { static bool abort_cb(AMstack** stack, void* data) {
static char buffer[512] = {0}; static char buffer[512] = {0};
char const* suffix = NULL; char const* suffix = NULL;
if (!stack) { if (!stack) {
suffix = "Stack*"; suffix = "Stack*";
} } else if (!*stack) {
else if (!*stack) {
suffix = "Stack"; suffix = "Stack";
} } else if (!(*stack)->result) {
else if (!(*stack)->result) {
suffix = ""; suffix = "";
} }
if (suffix) { if (suffix) {
fprintf(stderr, "Null `AMresult%s*`.", suffix); fprintf(stderr, "Null `AMresult%s*`.\n", suffix);
AMfreeStack(stack); AMstackFree(stack);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
return; return false;
} }
AMstatus const status = AMresultStatus((*stack)->result); AMstatus const status = AMresultStatus((*stack)->result);
switch (status) { switch (status) {
case AM_STATUS_ERROR: strcpy(buffer, "Error"); break; case AM_STATUS_ERROR:
case AM_STATUS_INVALID_RESULT: strcpy(buffer, "Invalid result"); break; strcpy(buffer, "Error");
case AM_STATUS_OK: break; break;
default: sprintf(buffer, "Unknown `AMstatus` tag %d", status); case AM_STATUS_INVALID_RESULT:
strcpy(buffer, "Invalid result");
break;
case AM_STATUS_OK:
break;
default:
sprintf(buffer, "Unknown `AMstatus` tag %d", status);
} }
if (buffer[0]) { if (buffer[0]) {
AMbyteSpan const msg = AMerrorMessage((*stack)->result); char* const c_msg = AMstrdup(AMresultError((*stack)->result), NULL);
char* const c_msg = calloc(1, msg.count + 1); fprintf(stderr, "%s; %s.\n", buffer, c_msg);
strncpy(c_msg, msg.src, msg.count);
fprintf(stderr, "%s; %s.", buffer, c_msg);
free(c_msg); free(c_msg);
AMfreeStack(stack); AMstackFree(stack);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
return; return false;
} }
AMvalue const value = AMresultValue((*stack)->result); if (data) {
fprintf(stderr, "Unexpected tag `AM_VALUE_%s` (%d); expected `AM_VALUE_%s`.", AMstackCallbackData* sc_data = (AMstackCallbackData*)data;
discriminant_suffix(value.tag), AMvalType const tag = AMitemValType(AMresultItem((*stack)->result));
value.tag, if (tag != sc_data->bitmask) {
discriminant_suffix(discriminant)); fprintf(stderr, "Unexpected tag `%s` (%d) instead of `%s` at %s:%d.\n", AMvalTypeToString(tag), tag,
AMfreeStack(stack); AMvalTypeToString(sc_data->bitmask), sc_data->file, sc_data->line);
exit(EXIT_FAILURE); free(sc_data);
} AMstackFree(stack);
exit(EXIT_FAILURE);
/** return false;
* \brief Gets the suffix for a discriminant's corresponding string }
* representation.
*
* \param[in] discriminant An `AMvalueVariant` enum tag.
* \return A UTF-8 string.
*/
static char const* discriminant_suffix(AMvalueVariant const discriminant) {
char const* suffix = NULL;
switch (discriminant) {
case AM_VALUE_ACTOR_ID: suffix = "ACTOR_ID"; break;
case AM_VALUE_BOOLEAN: suffix = "BOOLEAN"; break;
case AM_VALUE_BYTES: suffix = "BYTES"; break;
case AM_VALUE_CHANGE_HASHES: suffix = "CHANGE_HASHES"; break;
case AM_VALUE_CHANGES: suffix = "CHANGES"; break;
case AM_VALUE_COUNTER: suffix = "COUNTER"; break;
case AM_VALUE_DOC: suffix = "DOC"; break;
case AM_VALUE_F64: suffix = "F64"; break;
case AM_VALUE_INT: suffix = "INT"; break;
case AM_VALUE_LIST_ITEMS: suffix = "LIST_ITEMS"; break;
case AM_VALUE_MAP_ITEMS: suffix = "MAP_ITEMS"; break;
case AM_VALUE_NULL: suffix = "NULL"; break;
case AM_VALUE_OBJ_ID: suffix = "OBJ_ID"; break;
case AM_VALUE_OBJ_ITEMS: suffix = "OBJ_ITEMS"; break;
case AM_VALUE_STR: suffix = "STR"; break;
case AM_VALUE_STRS: suffix = "STRINGS"; break;
case AM_VALUE_SYNC_MESSAGE: suffix = "SYNC_MESSAGE"; break;
case AM_VALUE_SYNC_STATE: suffix = "SYNC_STATE"; break;
case AM_VALUE_TIMESTAMP: suffix = "TIMESTAMP"; break;
case AM_VALUE_UINT: suffix = "UINT"; break;
case AM_VALUE_VOID: suffix = "VOID"; break;
default: suffix = "...";
} }
return suffix; free(data);
return true;
} }

View file

@ -0,0 +1,30 @@
#ifndef AUTOMERGE_C_UTILS_RESULT_H
#define AUTOMERGE_C_UTILS_RESULT_H
/**
* \file
* \brief Utility functions for use with `AMresult` structs.
*/
#include <stdarg.h>
#include <automerge-c/automerge.h>
/**
* \brief Transfers the items within an arbitrary list of results into a
* new result in their order of specification.
* \param[in] count The count of subsequent arguments.
* \param[in] ... A \p count list of arguments, each of which is a pointer to
* an `AMresult` struct whose items will be transferred out of it
* and which is subsequently freed.
* \return A pointer to an `AMresult` struct or `NULL`.
* \pre `𝑥 ` \p ... `, AMresultStatus(𝑥) == AM_STATUS_OK`
* \post `(𝑥 ` \p ... `, AMresultStatus(𝑥) != AM_STATUS_OK) -> NULL`
* \attention All `AMresult` struct pointer arguments are passed to
* `AMresultFree()` regardless of success; use `AMresultCat()`
* instead if you wish to pass them to `AMresultFree()` yourself.
* \warning The returned `AMresult` struct pointer must be passed to
* `AMresultFree()` in order to avoid a memory leak.
*/
AMresult* AMresultFrom(int count, ...);
#endif /* AUTOMERGE_C_UTILS_RESULT_H */

View file

@ -0,0 +1,130 @@
#ifndef AUTOMERGE_C_UTILS_STACK_H
#define AUTOMERGE_C_UTILS_STACK_H
/**
* \file
* \brief Utility data structures and functions for hiding `AMresult` structs,
* managing their lifetimes, and automatically applying custom
* validation logic to the `AMitem` structs that they contain.
*
* \note The `AMstack` struct and its related functions drastically reduce the
* need for boilerplate code and/or `goto` statement usage within a C
* application but a higher-level programming language offers even better
* ways to do the same things.
*/
#include <automerge-c/automerge.h>
/**
* \struct AMstack
* \brief A node in a singly-linked list of result pointers.
*/
typedef struct AMstack {
/** A result to be deallocated. */
AMresult* result;
/** The previous node in the singly-linked list or `NULL`. */
struct AMstack* prev;
} AMstack;
/**
* \memberof AMstack
* \brief The prototype of a function that examines the result at the top of
* the given stack in terms of some arbitrary data.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] data A pointer to arbitrary data or `NULL`.
* \return `true` if the top `AMresult` struct in \p stack is valid, `false`
* otherwise.
* \pre \p stack `!= NULL`.
*/
typedef bool (*AMstackCallback)(AMstack** stack, void* data);
/**
* \memberof AMstack
* \brief Deallocates the storage for a stack of results.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \pre \p stack `!= NULL`
* \post `*stack == NULL`
*/
void AMstackFree(AMstack** stack);
/**
* \memberof AMstack
* \brief Gets a result from the stack after removing it.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] result A pointer to the `AMresult` to be popped or `NULL` to
* select the top result in \p stack.
* \return A pointer to an `AMresult` struct or `NULL`.
* \pre \p stack `!= NULL`
* \warning The returned `AMresult` struct pointer must be passed to
* `AMresultFree()` in order to avoid a memory leak.
*/
AMresult* AMstackPop(AMstack** stack, AMresult const* result);
/**
* \memberof AMstack
* \brief Pushes the given result onto the given stack, calls the given
* callback with the given data to validate it and then either gets the
* result if it's valid or gets `NULL` instead.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] result A pointer to an `AMresult` struct.
* \param[in] callback A pointer to a function with the same signature as
* `AMstackCallback()` or `NULL`.
* \param[in] data A pointer to arbitrary data or `NULL` which is passed to
* \p callback.
* \return \p result or `NULL`.
* \warning If \p stack `== NULL` then \p result is deallocated in order to
* avoid a memory leak.
*/
AMresult* AMstackResult(AMstack** stack, AMresult* result, AMstackCallback callback, void* data);
/**
* \memberof AMstack
* \brief Pushes the given result onto the given stack, calls the given
* callback with the given data to validate it and then either gets the
* first item in the sequence of items within that result if it's valid
* or gets `NULL` instead.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] result A pointer to an `AMresult` struct.
* \param[in] callback A pointer to a function with the same signature as
* `AMstackCallback()` or `NULL`.
* \param[in] data A pointer to arbitrary data or `NULL` which is passed to
* \p callback.
* \return A pointer to an `AMitem` struct or `NULL`.
* \warning If \p stack `== NULL` then \p result is deallocated in order to
* avoid a memory leak.
*/
AMitem* AMstackItem(AMstack** stack, AMresult* result, AMstackCallback callback, void* data);
/**
* \memberof AMstack
* \brief Pushes the given result onto the given stack, calls the given
* callback with the given data to validate it and then either gets an
* `AMitems` struct over the sequence of items within that result if it's
* valid or gets an empty `AMitems` instead.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] result A pointer to an `AMresult` struct.
* \param[in] callback A pointer to a function with the same signature as
* `AMstackCallback()` or `NULL`.
* \param[in] data A pointer to arbitrary data or `NULL` which is passed to
* \p callback.
* \return An `AMitems` struct.
* \warning If \p stack `== NULL` then \p result is deallocated immediately
* in order to avoid a memory leak.
*/
AMitems AMstackItems(AMstack** stack, AMresult* result, AMstackCallback callback, void* data);
/**
* \memberof AMstack
* \brief Gets the count of results that have been pushed onto the stack.
*
* \param[in,out] stack A pointer to an `AMstack` struct.
* \return A 64-bit unsigned integer.
*/
size_t AMstackSize(AMstack const* const stack);
#endif /* AUTOMERGE_C_UTILS_STACK_H */

View file

@ -0,0 +1,53 @@
#ifndef AUTOMERGE_C_UTILS_PUSH_CALLBACK_DATA_H
#define AUTOMERGE_C_UTILS_PUSH_CALLBACK_DATA_H
/**
* \file
* \brief Utility data structures, functions and macros for supplying
* parameters to the custom validation logic applied to `AMitem`
* structs.
*/
#include <automerge-c/automerge.h>
/**
* \struct AMstackCallbackData
* \brief A data structure for passing the parameters of an item value test
* to an implementation of the `AMstackCallback` function prototype.
*/
typedef struct {
/** A bitmask of `AMvalType` tags. */
AMvalType bitmask;
/** A null-terminated file path string. */
char const* file;
/** The ordinal number of a line within a file. */
int line;
} AMstackCallbackData;
/**
* \memberof AMstackCallbackData
* \brief Allocates a new `AMstackCallbackData` struct and initializes its
* members from their corresponding arguments.
*
* \param[in] bitmask A bitmask of `AMvalType` tags.
* \param[in] file A null-terminated file path string.
* \param[in] line The ordinal number of a line within a file.
* \return A pointer to a disowned `AMstackCallbackData` struct.
* \warning The returned pointer must be passed to `free()` to avoid a memory
* leak.
*/
AMstackCallbackData* AMstackCallbackDataInit(AMvalType const bitmask, char const* const file, int const line);
/**
* \memberof AMstackCallbackData
* \def AMexpect
* \brief Allocates a new `AMstackCallbackData` struct and initializes it from
* an `AMvalueType` bitmask.
*
* \param[in] bitmask A bitmask of `AMvalType` tags.
* \return A pointer to a disowned `AMstackCallbackData` struct.
* \warning The returned pointer must be passed to `free()` to avoid a memory
* leak.
*/
#define AMexpect(bitmask) AMstackCallbackDataInit(bitmask, __FILE__, __LINE__)
#endif /* AUTOMERGE_C_UTILS_PUSH_CALLBACK_DATA_H */

View file

@ -0,0 +1,29 @@
#ifndef AUTOMERGE_C_UTILS_STRING_H
#define AUTOMERGE_C_UTILS_STRING_H
/**
* \file
* \brief Utility functions for use with `AMbyteSpan` structs that provide
* UTF-8 string views.
*/
#include <automerge-c/automerge.h>
/**
* \memberof AMbyteSpan
* \brief Returns a pointer to a null-terminated byte string which is a
* duplicate of the given UTF-8 string view except for the substitution
* of its NUL (0) characters with the specified null-terminated byte
* string.
*
* \param[in] str A UTF-8 string view as an `AMbyteSpan` struct.
* \param[in] nul A null-terminated byte string to substitute for NUL characters
* or `NULL` to substitute `"\\0"` for NUL characters.
* \return A disowned null-terminated byte string.
* \pre \p str.src `!= NULL`
* \pre \p str.count `<= sizeof(`\p str.src `)`
* \warning The returned pointer must be passed to `free()` to avoid a memory
* leak.
*/
char* AMstrdup(AMbyteSpan const str, char const* nul);
#endif /* AUTOMERGE_C_UTILS_STRING_H */

View file

@ -1,250 +0,0 @@
cmake_minimum_required(VERSION 3.18 FATAL_ERROR)
find_program (
CARGO_CMD
"cargo"
PATHS "$ENV{CARGO_HOME}/bin"
DOC "The Cargo command"
)
if(NOT CARGO_CMD)
message(FATAL_ERROR "Cargo (Rust package manager) not found! Install it and/or set the CARGO_HOME environment variable.")
endif()
string(TOLOWER "${CMAKE_BUILD_TYPE}" BUILD_TYPE_LOWER)
if(BUILD_TYPE_LOWER STREQUAL debug)
set(CARGO_BUILD_TYPE "debug")
set(CARGO_FLAG "")
else()
set(CARGO_BUILD_TYPE "release")
set(CARGO_FLAG "--release")
endif()
set(CARGO_FEATURES "")
set(CARGO_CURRENT_BINARY_DIR "${CARGO_TARGET_DIR}/${CARGO_BUILD_TYPE}")
set(
CARGO_OUTPUT
${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}
${CARGO_CURRENT_BINARY_DIR}/${CMAKE_STATIC_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_STATIC_LIBRARY_SUFFIX}
)
if(WIN32)
# \note The basename of an import library output by Cargo is the filename
# of its corresponding shared library.
list(APPEND CARGO_OUTPUT ${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}${CMAKE_STATIC_LIBRARY_SUFFIX})
endif()
add_custom_command(
OUTPUT
${CARGO_OUTPUT}
COMMAND
# \note cbindgen won't regenerate its output header file after it's
# been removed but it will after its configuration file has been
# updated.
${CMAKE_COMMAND} -DCONDITION=NOT_EXISTS -P ${CMAKE_SOURCE_DIR}/cmake/file_touch.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h ${CMAKE_SOURCE_DIR}/cbindgen.toml
COMMAND
${CMAKE_COMMAND} -E env CARGO_TARGET_DIR=${CARGO_TARGET_DIR} CBINDGEN_TARGET_DIR=${CBINDGEN_TARGET_DIR} ${CARGO_CMD} build ${CARGO_FLAG} ${CARGO_FEATURES}
MAIN_DEPENDENCY
lib.rs
DEPENDS
actor_id.rs
byte_span.rs
change_hashes.rs
change.rs
changes.rs
doc.rs
doc/list.rs
doc/list/item.rs
doc/list/items.rs
doc/map.rs
doc/map/item.rs
doc/map/items.rs
doc/utils.rs
obj.rs
obj/item.rs
obj/items.rs
result.rs
result_stack.rs
strs.rs
sync.rs
sync/have.rs
sync/haves.rs
sync/message.rs
sync/state.rs
${CMAKE_SOURCE_DIR}/build.rs
${CMAKE_SOURCE_DIR}/Cargo.toml
${CMAKE_SOURCE_DIR}/cbindgen.toml
WORKING_DIRECTORY
${CMAKE_SOURCE_DIR}
COMMENT
"Producing the library artifacts with Cargo..."
VERBATIM
)
add_custom_target(
${LIBRARY_NAME}_artifacts ALL
DEPENDS ${CARGO_OUTPUT}
)
# \note cbindgen's naming behavior isn't fully configurable and it ignores
# `const fn` calls (https://github.com/eqrion/cbindgen/issues/252).
add_custom_command(
TARGET ${LIBRARY_NAME}_artifacts
POST_BUILD
COMMAND
# Compensate for cbindgen's variant struct naming.
${CMAKE_COMMAND} -DMATCH_REGEX=AM\([^_]+_[^_]+\)_Body -DREPLACE_EXPR=AM\\1 -P ${CMAKE_SOURCE_DIR}/cmake/file_regex_replace.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
COMMAND
# Compensate for cbindgen's union tag enum type naming.
${CMAKE_COMMAND} -DMATCH_REGEX=AM\([^_]+\)_Tag -DREPLACE_EXPR=AM\\1Variant -P ${CMAKE_SOURCE_DIR}/cmake/file_regex_replace.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
COMMAND
# Compensate for cbindgen's translation of consecutive uppercase letters to "ScreamingSnakeCase".
${CMAKE_COMMAND} -DMATCH_REGEX=A_M\([^_]+\)_ -DREPLACE_EXPR=AM_\\1_ -P ${CMAKE_SOURCE_DIR}/cmake/file_regex_replace.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
COMMAND
# Compensate for cbindgen ignoring `std:mem::size_of<usize>()` calls.
${CMAKE_COMMAND} -DMATCH_REGEX=USIZE_ -DREPLACE_EXPR=\+${CMAKE_SIZEOF_VOID_P} -P ${CMAKE_SOURCE_DIR}/cmake/file_regex_replace.cmake -- ${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h
WORKING_DIRECTORY
${CMAKE_SOURCE_DIR}
COMMENT
"Compensating for cbindgen deficits..."
VERBATIM
)
if(BUILD_SHARED_LIBS)
if(WIN32)
set(LIBRARY_DESTINATION "${CMAKE_INSTALL_BINDIR}")
else()
set(LIBRARY_DESTINATION "${CMAKE_INSTALL_LIBDIR}")
endif()
set(LIBRARY_DEFINE_SYMBOL "${SYMBOL_PREFIX}_EXPORTS")
# \note The basename of an import library output by Cargo is the filename
# of its corresponding shared library.
set(LIBRARY_IMPLIB "${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}${CMAKE_STATIC_LIBRARY_SUFFIX}")
set(LIBRARY_LOCATION "${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_SHARED_LIBRARY_SUFFIX}")
set(LIBRARY_NO_SONAME "${WIN32}")
set(LIBRARY_SONAME "${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_${CMAKE_BUILD_TYPE}_POSTFIX}${CMAKE_SHARED_LIBRARY_SUFFIX}")
set(LIBRARY_TYPE "SHARED")
else()
set(LIBRARY_DEFINE_SYMBOL "")
set(LIBRARY_DESTINATION "${CMAKE_INSTALL_LIBDIR}")
set(LIBRARY_IMPLIB "")
set(LIBRARY_LOCATION "${CARGO_CURRENT_BINARY_DIR}/${CMAKE_STATIC_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_STATIC_LIBRARY_SUFFIX}")
set(LIBRARY_NO_SONAME "TRUE")
set(LIBRARY_SONAME "")
set(LIBRARY_TYPE "STATIC")
endif()
add_library(${LIBRARY_NAME} ${LIBRARY_TYPE} IMPORTED GLOBAL)
set_target_properties(
${LIBRARY_NAME}
PROPERTIES
# \note Cargo writes a debug build into a nested directory instead of
# decorating its name.
DEBUG_POSTFIX ""
DEFINE_SYMBOL "${LIBRARY_DEFINE_SYMBOL}"
IMPORTED_IMPLIB "${LIBRARY_IMPLIB}"
IMPORTED_LOCATION "${LIBRARY_LOCATION}"
IMPORTED_NO_SONAME "${LIBRARY_NO_SONAME}"
IMPORTED_SONAME "${LIBRARY_SONAME}"
LINKER_LANGUAGE C
PUBLIC_HEADER "${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h"
SOVERSION "${PROJECT_VERSION_MAJOR}"
VERSION "${PROJECT_VERSION}"
# \note Cargo exports all of the symbols automatically.
WINDOWS_EXPORT_ALL_SYMBOLS "TRUE"
)
target_compile_definitions(${LIBRARY_NAME} INTERFACE $<TARGET_PROPERTY:${LIBRARY_NAME},DEFINE_SYMBOL>)
target_include_directories(
${LIBRARY_NAME}
INTERFACE
"$<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}>"
)
set(CMAKE_THREAD_PREFER_PTHREAD TRUE)
set(THREADS_PREFER_PTHREAD_FLAG TRUE)
find_package(Threads REQUIRED)
set(LIBRARY_DEPENDENCIES Threads::Threads ${CMAKE_DL_LIBS})
if(WIN32)
list(APPEND LIBRARY_DEPENDENCIES Bcrypt userenv ws2_32)
else()
list(APPEND LIBRARY_DEPENDENCIES m)
endif()
target_link_libraries(${LIBRARY_NAME} INTERFACE ${LIBRARY_DEPENDENCIES})
install(
FILES $<TARGET_PROPERTY:${LIBRARY_NAME},IMPORTED_IMPLIB>
TYPE LIB
# \note The basename of an import library output by Cargo is the filename
# of its corresponding shared library.
RENAME "${CMAKE_STATIC_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_${CMAKE_BUILD_TYPE}_POSTFIX}${CMAKE_STATIC_LIBRARY_SUFFIX}"
OPTIONAL
)
set(LIBRARY_FILE_NAME "${CMAKE_${LIBRARY_TYPE}_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_${CMAKE_BUILD_TYPE}_POSTFIX}${CMAKE_${LIBRARY_TYPE}_LIBRARY_SUFFIX}")
install(
FILES $<TARGET_PROPERTY:${LIBRARY_NAME},IMPORTED_LOCATION>
RENAME "${LIBRARY_FILE_NAME}"
DESTINATION ${LIBRARY_DESTINATION}
)
install(
FILES $<TARGET_PROPERTY:${LIBRARY_NAME},PUBLIC_HEADER>
DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}
)
find_package(Doxygen OPTIONAL_COMPONENTS dot)
if(DOXYGEN_FOUND)
set(DOXYGEN_ALIASES "installed_headerfile=\\headerfile ${LIBRARY_NAME}.h <${PROJECT_NAME}/${LIBRARY_NAME}.h>")
set(DOXYGEN_GENERATE_LATEX YES)
set(DOXYGEN_PDF_HYPERLINKS YES)
set(DOXYGEN_PROJECT_LOGO "${CMAKE_SOURCE_DIR}/img/brandmark.png")
set(DOXYGEN_SORT_BRIEF_DOCS YES)
set(DOXYGEN_USE_MDFILE_AS_MAINPAGE "${CMAKE_SOURCE_DIR}/README.md")
doxygen_add_docs(
${LIBRARY_NAME}_docs
"${CBINDGEN_TARGET_DIR}/${LIBRARY_NAME}.h"
"${CMAKE_SOURCE_DIR}/README.md"
USE_STAMP_FILE
WORKING_DIRECTORY ${CMAKE_SOURCE_DIR}
COMMENT "Producing documentation with Doxygen..."
)
# \note A Doxygen input file isn't a file-level dependency so the Doxygen
# command must instead depend upon a target that outputs the file or
# it will just output an error message when it can't be found.
add_dependencies(${LIBRARY_NAME}_docs ${LIBRARY_NAME}_artifacts)
endif()

View file

@ -1,4 +1,5 @@
use automerge as am; use automerge as am;
use libc::c_int;
use std::cell::RefCell; use std::cell::RefCell;
use std::cmp::Ordering; use std::cmp::Ordering;
use std::str::FromStr; use std::str::FromStr;
@ -11,7 +12,7 @@ macro_rules! to_actor_id {
let handle = $handle.as_ref(); let handle = $handle.as_ref();
match handle { match handle {
Some(b) => b, Some(b) => b,
None => return AMresult::err("Invalid AMactorId pointer").into(), None => return AMresult::error("Invalid `AMactorId*`").into(),
} }
}}; }};
} }
@ -57,11 +58,11 @@ impl AsRef<am::ActorId> for AMactorId {
} }
/// \memberof AMactorId /// \memberof AMactorId
/// \brief Gets the value of an actor identifier as a sequence of bytes. /// \brief Gets the value of an actor identifier as an array of bytes.
/// ///
/// \param[in] actor_id A pointer to an `AMactorId` struct. /// \param[in] actor_id A pointer to an `AMactorId` struct.
/// \pre \p actor_id `!= NULL`. /// \return An `AMbyteSpan` struct for an array of bytes.
/// \return An `AMbyteSpan` struct. /// \pre \p actor_id `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -82,8 +83,8 @@ pub unsafe extern "C" fn AMactorIdBytes(actor_id: *const AMactorId) -> AMbyteSpa
/// \return `-1` if \p actor_id1 `<` \p actor_id2, `0` if /// \return `-1` if \p actor_id1 `<` \p actor_id2, `0` if
/// \p actor_id1 `==` \p actor_id2 and `1` if /// \p actor_id1 `==` \p actor_id2 and `1` if
/// \p actor_id1 `>` \p actor_id2. /// \p actor_id1 `>` \p actor_id2.
/// \pre \p actor_id1 `!= NULL`. /// \pre \p actor_id1 `!= NULL`
/// \pre \p actor_id2 `!= NULL`. /// \pre \p actor_id2 `!= NULL`
/// \internal /// \internal
/// ///
/// #Safety /// #Safety
@ -93,7 +94,7 @@ pub unsafe extern "C" fn AMactorIdBytes(actor_id: *const AMactorId) -> AMbyteSpa
pub unsafe extern "C" fn AMactorIdCmp( pub unsafe extern "C" fn AMactorIdCmp(
actor_id1: *const AMactorId, actor_id1: *const AMactorId,
actor_id2: *const AMactorId, actor_id2: *const AMactorId,
) -> isize { ) -> c_int {
match (actor_id1.as_ref(), actor_id2.as_ref()) { match (actor_id1.as_ref(), actor_id2.as_ref()) {
(Some(actor_id1), Some(actor_id2)) => match actor_id1.as_ref().cmp(actor_id2.as_ref()) { (Some(actor_id1), Some(actor_id2)) => match actor_id1.as_ref().cmp(actor_id2.as_ref()) {
Ordering::Less => -1, Ordering::Less => -1,
@ -101,65 +102,69 @@ pub unsafe extern "C" fn AMactorIdCmp(
Ordering::Greater => 1, Ordering::Greater => 1,
}, },
(None, Some(_)) => -1, (None, Some(_)) => -1,
(Some(_), None) => 1,
(None, None) => 0, (None, None) => 0,
(Some(_), None) => 1,
} }
} }
/// \memberof AMactorId /// \memberof AMactorId
/// \brief Allocates a new actor identifier and initializes it with a random /// \brief Allocates a new actor identifier and initializes it from a random
/// UUID. /// UUID value.
/// ///
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_ACTOR_ID` item.
/// `AMactorId` struct. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMactorIdInit() -> *mut AMresult { pub unsafe extern "C" fn AMactorIdInit() -> *mut AMresult {
to_result(Ok::<am::ActorId, am::AutomergeError>(am::ActorId::random())) to_result(Ok::<am::ActorId, am::AutomergeError>(am::ActorId::random()))
} }
/// \memberof AMactorId /// \memberof AMactorId
/// \brief Allocates a new actor identifier and initializes it from a sequence /// \brief Allocates a new actor identifier and initializes it from an array of
/// of bytes. /// bytes value.
/// ///
/// \param[in] src A pointer to a contiguous sequence of bytes. /// \param[in] src A pointer to an array of bytes.
/// \param[in] count The number of bytes to copy from \p src. /// \param[in] count The count of bytes to copy from the array pointed to by
/// \pre `0 <` \p count `<= sizeof(`\p src`)`. /// \p src.
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_ACTOR_ID` item.
/// `AMactorId` struct. /// \pre \p src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `sizeof(`\p src `) > 0`
/// in order to prevent a memory leak. /// \pre \p count `<= sizeof(`\p src `)`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// src must be a byte array of size `>= count` /// src must be a byte array of length `>= count`
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMactorIdInitBytes(src: *const u8, count: usize) -> *mut AMresult { pub unsafe extern "C" fn AMactorIdFromBytes(src: *const u8, count: usize) -> *mut AMresult {
let slice = std::slice::from_raw_parts(src, count); if !src.is_null() {
to_result(Ok::<am::ActorId, am::InvalidActorId>(am::ActorId::from( let value = std::slice::from_raw_parts(src, count);
slice, to_result(Ok::<am::ActorId, am::InvalidActorId>(am::ActorId::from(
))) value,
)))
} else {
AMresult::error("Invalid uint8_t*").into()
}
} }
/// \memberof AMactorId /// \memberof AMactorId
/// \brief Allocates a new actor identifier and initializes it from a /// \brief Allocates a new actor identifier and initializes it from a
/// hexadecimal string. /// hexadecimal UTF-8 string view value.
/// ///
/// \param[in] hex_str A UTF-8 string view as an `AMbyteSpan` struct. /// \param[in] value A UTF-8 string view as an `AMbyteSpan` struct.
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_ACTOR_ID` item.
/// `AMactorId` struct. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// hex_str must be a valid pointer to an AMbyteSpan /// hex_str must be a valid pointer to an AMbyteSpan
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMactorIdInitStr(hex_str: AMbyteSpan) -> *mut AMresult { pub unsafe extern "C" fn AMactorIdFromStr(value: AMbyteSpan) -> *mut AMresult {
use am::AutomergeError::InvalidActorId; use am::AutomergeError::InvalidActorId;
to_result(match (&hex_str).try_into() { to_result(match (&value).try_into() {
Ok(s) => match am::ActorId::from_str(s) { Ok(s) => match am::ActorId::from_str(s) {
Ok(actor_id) => Ok(actor_id), Ok(actor_id) => Ok(actor_id),
Err(_) => Err(InvalidActorId(String::from(s))), Err(_) => Err(InvalidActorId(String::from(s))),
@ -169,11 +174,12 @@ pub unsafe extern "C" fn AMactorIdInitStr(hex_str: AMbyteSpan) -> *mut AMresult
} }
/// \memberof AMactorId /// \memberof AMactorId
/// \brief Gets the value of an actor identifier as a hexadecimal string. /// \brief Gets the value of an actor identifier as a UTF-8 hexadecimal string
/// view.
/// ///
/// \param[in] actor_id A pointer to an `AMactorId` struct. /// \param[in] actor_id A pointer to an `AMactorId` struct.
/// \pre \p actor_id `!= NULL`.
/// \return A UTF-8 string view as an `AMbyteSpan` struct. /// \return A UTF-8 string view as an `AMbyteSpan` struct.
/// \pre \p actor_id `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety

View file

@ -1,14 +1,17 @@
use automerge as am; use automerge as am;
use libc::strlen; use std::cmp::Ordering;
use std::convert::TryFrom; use std::convert::TryFrom;
use std::os::raw::c_char; use std::os::raw::c_char;
use libc::{c_int, strlen};
use smol_str::SmolStr;
macro_rules! to_str { macro_rules! to_str {
($span:expr) => {{ ($byte_span:expr) => {{
let result: Result<&str, am::AutomergeError> = (&$span).try_into(); let result: Result<&str, am::AutomergeError> = (&$byte_span).try_into();
match result { match result {
Ok(s) => s, Ok(s) => s,
Err(e) => return AMresult::err(&e.to_string()).into(), Err(e) => return AMresult::error(&e.to_string()).into(),
} }
}}; }};
} }
@ -17,16 +20,17 @@ pub(crate) use to_str;
/// \struct AMbyteSpan /// \struct AMbyteSpan
/// \installed_headerfile /// \installed_headerfile
/// \brief A view onto a contiguous sequence of bytes. /// \brief A view onto an array of bytes.
#[repr(C)] #[repr(C)]
pub struct AMbyteSpan { pub struct AMbyteSpan {
/// A pointer to an array of bytes. /// A pointer to the first byte of an array of bytes.
/// \attention <b>NEVER CALL `free()` ON \p src!</b> /// \warning \p src is only valid until the array of bytes to which it
/// \warning \p src is only valid until the `AMfree()` function is called /// points is freed.
/// on the `AMresult` struct that stores the array of bytes to /// \note If the `AMbyteSpan` came from within an `AMitem` struct then
/// which it points. /// \p src will be freed when the pointer to the `AMresult` struct
/// containing the `AMitem` struct is passed to `AMresultFree()`.
pub src: *const u8, pub src: *const u8,
/// The number of bytes in the array. /// The count of bytes in the array.
pub count: usize, pub count: usize,
} }
@ -52,9 +56,7 @@ impl PartialEq for AMbyteSpan {
} else if self.src == other.src { } else if self.src == other.src {
return true; return true;
} }
let slice = unsafe { std::slice::from_raw_parts(self.src, self.count) }; <&[u8]>::from(self) == <&[u8]>::from(other)
let other_slice = unsafe { std::slice::from_raw_parts(other.src, other.count) };
slice == other_slice
} }
} }
@ -72,10 +74,15 @@ impl From<&am::ActorId> for AMbyteSpan {
impl From<&mut am::ActorId> for AMbyteSpan { impl From<&mut am::ActorId> for AMbyteSpan {
fn from(actor: &mut am::ActorId) -> Self { fn from(actor: &mut am::ActorId) -> Self {
let slice = actor.to_bytes(); actor.as_ref().into()
}
}
impl From<&am::ChangeHash> for AMbyteSpan {
fn from(change_hash: &am::ChangeHash) -> Self {
Self { Self {
src: slice.as_ptr(), src: change_hash.0.as_ptr(),
count: slice.len(), count: change_hash.0.len(),
} }
} }
} }
@ -93,12 +100,9 @@ impl From<*const c_char> for AMbyteSpan {
} }
} }
impl From<&am::ChangeHash> for AMbyteSpan { impl From<&SmolStr> for AMbyteSpan {
fn from(change_hash: &am::ChangeHash) -> Self { fn from(smol_str: &SmolStr) -> Self {
Self { smol_str.as_bytes().into()
src: change_hash.0.as_ptr(),
count: change_hash.0.len(),
}
} }
} }
@ -111,13 +115,39 @@ impl From<&[u8]> for AMbyteSpan {
} }
} }
impl From<&AMbyteSpan> for &[u8] {
fn from(byte_span: &AMbyteSpan) -> Self {
unsafe { std::slice::from_raw_parts(byte_span.src, byte_span.count) }
}
}
impl From<&AMbyteSpan> for Vec<u8> {
fn from(byte_span: &AMbyteSpan) -> Self {
<&[u8]>::from(byte_span).to_vec()
}
}
impl TryFrom<&AMbyteSpan> for am::ChangeHash {
type Error = am::AutomergeError;
fn try_from(byte_span: &AMbyteSpan) -> Result<Self, Self::Error> {
use am::AutomergeError::InvalidChangeHashBytes;
let slice: &[u8] = byte_span.into();
match slice.try_into() {
Ok(change_hash) => Ok(change_hash),
Err(e) => Err(InvalidChangeHashBytes(e)),
}
}
}
impl TryFrom<&AMbyteSpan> for &str { impl TryFrom<&AMbyteSpan> for &str {
type Error = am::AutomergeError; type Error = am::AutomergeError;
fn try_from(span: &AMbyteSpan) -> Result<Self, Self::Error> { fn try_from(byte_span: &AMbyteSpan) -> Result<Self, Self::Error> {
use am::AutomergeError::InvalidCharacter; use am::AutomergeError::InvalidCharacter;
let slice = unsafe { std::slice::from_raw_parts(span.src, span.count) }; let slice = byte_span.into();
match std::str::from_utf8(slice) { match std::str::from_utf8(slice) {
Ok(str_) => Ok(str_), Ok(str_) => Ok(str_),
Err(e) => Err(InvalidCharacter(e.valid_up_to())), Err(e) => Err(InvalidCharacter(e.valid_up_to())),
@ -125,17 +155,69 @@ impl TryFrom<&AMbyteSpan> for &str {
} }
} }
/// \brief Creates an AMbyteSpan from a pointer + length /// \memberof AMbyteSpan
/// \brief Creates a view onto an array of bytes.
/// ///
/// \param[in] src A pointer to a span of bytes /// \param[in] src A pointer to an array of bytes or `NULL`.
/// \param[in] count The number of bytes in the span /// \param[in] count The count of bytes to view from the array pointed to by
/// \return An `AMbyteSpan` struct /// \p src.
/// \return An `AMbyteSpan` struct.
/// \pre \p count `<= sizeof(`\p src `)`
/// \post `(`\p src `== NULL) -> (AMbyteSpan){NULL, 0}`
/// \internal /// \internal
/// ///
/// #Safety /// #Safety
/// AMbytes does not retain the underlying storage, so you must discard the /// src must be a byte array of length `>= count` or `std::ptr::null()`
/// return value before freeing the bytes.
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMbytes(src: *const u8, count: usize) -> AMbyteSpan { pub unsafe extern "C" fn AMbytes(src: *const u8, count: usize) -> AMbyteSpan {
AMbyteSpan { src, count } AMbyteSpan {
src,
count: if src.is_null() { 0 } else { count },
}
}
/// \memberof AMbyteSpan
/// \brief Creates a view onto a C string.
///
/// \param[in] c_str A null-terminated byte string or `NULL`.
/// \return An `AMbyteSpan` struct.
/// \pre Each byte in \p c_str encodes one UTF-8 character.
/// \internal
///
/// #Safety
/// c_str must be a null-terminated array of `std::os::raw::c_char` or `std::ptr::null()`.
#[no_mangle]
pub unsafe extern "C" fn AMstr(c_str: *const c_char) -> AMbyteSpan {
c_str.into()
}
/// \memberof AMbyteSpan
/// \brief Compares two UTF-8 string views lexicographically.
///
/// \param[in] lhs A UTF-8 string view as an `AMbyteSpan` struct.
/// \param[in] rhs A UTF-8 string view as an `AMbyteSpan` struct.
/// \return Negative value if \p lhs appears before \p rhs in lexicographical order.
/// Zero if \p lhs and \p rhs compare equal.
/// Positive value if \p lhs appears after \p rhs in lexicographical order.
/// \pre \p lhs.src `!= NULL`
/// \pre \p lhs.count `<= sizeof(`\p lhs.src `)`
/// \pre \p rhs.src `!= NULL`
/// \pre \p rhs.count `<= sizeof(`\p rhs.src `)`
/// \internal
///
/// #Safety
/// lhs.src must be a byte array of length >= lhs.count
/// rhs.src must be a a byte array of length >= rhs.count
#[no_mangle]
pub unsafe extern "C" fn AMstrCmp(lhs: AMbyteSpan, rhs: AMbyteSpan) -> c_int {
match (<&str>::try_from(&lhs), <&str>::try_from(&rhs)) {
(Ok(lhs), Ok(rhs)) => match lhs.cmp(rhs) {
Ordering::Less => -1,
Ordering::Equal => 0,
Ordering::Greater => 1,
},
(Err(_), Ok(_)) => -1,
(Err(_), Err(_)) => 0,
(Ok(_), Err(_)) => 1,
}
} }

View file

@ -2,7 +2,6 @@ use automerge as am;
use std::cell::RefCell; use std::cell::RefCell;
use crate::byte_span::AMbyteSpan; use crate::byte_span::AMbyteSpan;
use crate::change_hashes::AMchangeHashes;
use crate::result::{to_result, AMresult}; use crate::result::{to_result, AMresult};
macro_rules! to_change { macro_rules! to_change {
@ -10,7 +9,7 @@ macro_rules! to_change {
let handle = $handle.as_ref(); let handle = $handle.as_ref();
match handle { match handle {
Some(b) => b, Some(b) => b,
None => return AMresult::err("Invalid AMchange pointer").into(), None => return AMresult::error("Invalid `AMchange*`").into(),
} }
}}; }};
} }
@ -21,14 +20,14 @@ macro_rules! to_change {
#[derive(Eq, PartialEq)] #[derive(Eq, PartialEq)]
pub struct AMchange { pub struct AMchange {
body: *mut am::Change, body: *mut am::Change,
changehash: RefCell<Option<am::ChangeHash>>, change_hash: RefCell<Option<am::ChangeHash>>,
} }
impl AMchange { impl AMchange {
pub fn new(change: &mut am::Change) -> Self { pub fn new(change: &mut am::Change) -> Self {
Self { Self {
body: change, body: change,
changehash: Default::default(), change_hash: Default::default(),
} }
} }
@ -40,12 +39,12 @@ impl AMchange {
} }
pub fn hash(&self) -> AMbyteSpan { pub fn hash(&self) -> AMbyteSpan {
let mut changehash = self.changehash.borrow_mut(); let mut change_hash = self.change_hash.borrow_mut();
if let Some(changehash) = changehash.as_ref() { if let Some(change_hash) = change_hash.as_ref() {
changehash.into() change_hash.into()
} else { } else {
let hash = unsafe { (*self.body).hash() }; let hash = unsafe { (*self.body).hash() };
let ptr = changehash.insert(hash); let ptr = change_hash.insert(hash);
AMbyteSpan { AMbyteSpan {
src: ptr.0.as_ptr(), src: ptr.0.as_ptr(),
count: hash.as_ref().len(), count: hash.as_ref().len(),
@ -70,11 +69,10 @@ impl AsRef<am::Change> for AMchange {
/// \brief Gets the first referenced actor identifier in a change. /// \brief Gets the first referenced actor identifier in a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \pre \p change `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_ACTOR_ID` item.
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \pre \p change `!= NULL`
/// `AMactorId` struct. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -90,8 +88,8 @@ pub unsafe extern "C" fn AMchangeActorId(change: *const AMchange) -> *mut AMresu
/// \memberof AMchange /// \memberof AMchange
/// \brief Compresses the raw bytes of a change. /// \brief Compresses the raw bytes of a change.
/// ///
/// \param[in,out] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -107,18 +105,20 @@ pub unsafe extern "C" fn AMchangeCompress(change: *mut AMchange) {
/// \brief Gets the dependencies of a change. /// \brief Gets the dependencies of a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A pointer to an `AMchangeHashes` struct or `NULL`. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// change must be a valid pointer to an AMchange /// change must be a valid pointer to an AMchange
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMchangeDeps(change: *const AMchange) -> AMchangeHashes { pub unsafe extern "C" fn AMchangeDeps(change: *const AMchange) -> *mut AMresult {
match change.as_ref() { to_result(match change.as_ref() {
Some(change) => AMchangeHashes::new(change.as_ref().deps()), Some(change) => change.as_ref().deps(),
None => Default::default(), None => Default::default(),
} })
} }
/// \memberof AMchange /// \memberof AMchange
@ -126,7 +126,7 @@ pub unsafe extern "C" fn AMchangeDeps(change: *const AMchange) -> AMchangeHashes
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return An `AMbyteSpan` struct. /// \return An `AMbyteSpan` struct.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -141,32 +141,33 @@ pub unsafe extern "C" fn AMchangeExtraBytes(change: *const AMchange) -> AMbyteSp
} }
/// \memberof AMchange /// \memberof AMchange
/// \brief Loads a sequence of bytes into a change. /// \brief Allocates a new change and initializes it from an array of bytes value.
/// ///
/// \param[in] src A pointer to an array of bytes. /// \param[in] src A pointer to an array of bytes.
/// \param[in] count The number of bytes in \p src to load. /// \param[in] count The count of bytes to load from the array pointed to by
/// \return A pointer to an `AMresult` struct containing an `AMchange` struct. /// \p src.
/// \pre \p src `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_CHANGE` item.
/// \pre `0 <` \p count `<= sizeof(`\p src`)`. /// \pre \p src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `sizeof(`\p src `) > 0`
/// in order to prevent a memory leak. /// \pre \p count `<= sizeof(`\p src `)`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// src must be a byte array of size `>= count` /// src must be a byte array of length `>= count`
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMchangeFromBytes(src: *const u8, count: usize) -> *mut AMresult { pub unsafe extern "C" fn AMchangeFromBytes(src: *const u8, count: usize) -> *mut AMresult {
let mut data = Vec::new(); let data = std::slice::from_raw_parts(src, count);
data.extend_from_slice(std::slice::from_raw_parts(src, count)); to_result(am::Change::from_bytes(data.to_vec()))
to_result(am::Change::from_bytes(data))
} }
/// \memberof AMchange /// \memberof AMchange
/// \brief Gets the hash of a change. /// \brief Gets the hash of a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A change hash as an `AMbyteSpan` struct. /// \return An `AMbyteSpan` struct for a change hash.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -183,8 +184,8 @@ pub unsafe extern "C" fn AMchangeHash(change: *const AMchange) -> AMbyteSpan {
/// \brief Tests the emptiness of a change. /// \brief Tests the emptiness of a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A boolean. /// \return `true` if \p change is empty, `false` otherwise.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -198,12 +199,37 @@ pub unsafe extern "C" fn AMchangeIsEmpty(change: *const AMchange) -> bool {
} }
} }
/// \memberof AMchange
/// \brief Loads a document into a sequence of changes.
///
/// \param[in] src A pointer to an array of bytes.
/// \param[in] count The count of bytes to load from the array pointed to by
/// \p src.
/// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE` items.
/// \pre \p src `!= NULL`
/// \pre `sizeof(`\p src `) > 0`
/// \pre \p count `<= sizeof(`\p src `)`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal
///
/// # Safety
/// src must be a byte array of length `>= count`
#[no_mangle]
pub unsafe extern "C" fn AMchangeLoadDocument(src: *const u8, count: usize) -> *mut AMresult {
let data = std::slice::from_raw_parts(src, count);
to_result::<Result<Vec<am::Change>, _>>(
am::Automerge::load(data)
.and_then(|d| d.get_changes(&[]).map(|c| c.into_iter().cloned().collect())),
)
}
/// \memberof AMchange /// \memberof AMchange
/// \brief Gets the maximum operation index of a change. /// \brief Gets the maximum operation index of a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A 64-bit unsigned integer. /// \return A 64-bit unsigned integer.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -221,8 +247,8 @@ pub unsafe extern "C" fn AMchangeMaxOp(change: *const AMchange) -> u64 {
/// \brief Gets the message of a change. /// \brief Gets the message of a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A UTF-8 string view as an `AMbyteSpan` struct. /// \return An `AMbyteSpan` struct for a UTF-8 string.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -240,7 +266,7 @@ pub unsafe extern "C" fn AMchangeMessage(change: *const AMchange) -> AMbyteSpan
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A 64-bit unsigned integer. /// \return A 64-bit unsigned integer.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -259,7 +285,7 @@ pub unsafe extern "C" fn AMchangeSeq(change: *const AMchange) -> u64 {
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A 64-bit unsigned integer. /// \return A 64-bit unsigned integer.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -267,10 +293,9 @@ pub unsafe extern "C" fn AMchangeSeq(change: *const AMchange) -> u64 {
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMchangeSize(change: *const AMchange) -> usize { pub unsafe extern "C" fn AMchangeSize(change: *const AMchange) -> usize {
if let Some(change) = change.as_ref() { if let Some(change) = change.as_ref() {
change.as_ref().len() return change.as_ref().len();
} else {
0
} }
0
} }
/// \memberof AMchange /// \memberof AMchange
@ -278,7 +303,7 @@ pub unsafe extern "C" fn AMchangeSize(change: *const AMchange) -> usize {
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A 64-bit unsigned integer. /// \return A 64-bit unsigned integer.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -297,7 +322,7 @@ pub unsafe extern "C" fn AMchangeStartOp(change: *const AMchange) -> u64 {
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return A 64-bit signed integer. /// \return A 64-bit signed integer.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -315,8 +340,8 @@ pub unsafe extern "C" fn AMchangeTime(change: *const AMchange) -> i64 {
/// \brief Gets the raw bytes of a change. /// \brief Gets the raw bytes of a change.
/// ///
/// \param[in] change A pointer to an `AMchange` struct. /// \param[in] change A pointer to an `AMchange` struct.
/// \return An `AMbyteSpan` struct. /// \return An `AMbyteSpan` struct for an array of bytes.
/// \pre \p change `!= NULL`. /// \pre \p change `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -329,28 +354,3 @@ pub unsafe extern "C" fn AMchangeRawBytes(change: *const AMchange) -> AMbyteSpan
Default::default() Default::default()
} }
} }
/// \memberof AMchange
/// \brief Loads a document into a sequence of changes.
///
/// \param[in] src A pointer to an array of bytes.
/// \param[in] count The number of bytes in \p src to load.
/// \return A pointer to an `AMresult` struct containing a sequence of
/// `AMchange` structs.
/// \pre \p src `!= NULL`.
/// \pre `0 <` \p count `<= sizeof(`\p src`)`.
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()`
/// in order to prevent a memory leak.
/// \internal
///
/// # Safety
/// src must be a byte array of size `>= count`
#[no_mangle]
pub unsafe extern "C" fn AMchangeLoadDocument(src: *const u8, count: usize) -> *mut AMresult {
let mut data = Vec::new();
data.extend_from_slice(std::slice::from_raw_parts(src, count));
to_result::<Result<Vec<am::Change>, _>>(
am::Automerge::load(&data)
.and_then(|d| d.get_changes(&[]).map(|c| c.into_iter().cloned().collect())),
)
}

View file

@ -1,400 +0,0 @@
use automerge as am;
use std::cmp::Ordering;
use std::ffi::c_void;
use std::mem::size_of;
use crate::byte_span::AMbyteSpan;
use crate::result::{to_result, AMresult};
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(change_hashes: &[am::ChangeHash], offset: isize) -> Self {
Self {
len: change_hashes.len(),
offset,
ptr: change_hashes.as_ptr() as *const c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<&am::ChangeHash> {
if self.is_stopped() {
return None;
}
let slice: &[am::ChangeHash] =
unsafe { std::slice::from_raw_parts(self.ptr as *const am::ChangeHash, self.len) };
let value = &slice[self.get_index()];
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<&am::ChangeHash> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &[am::ChangeHash] =
unsafe { std::slice::from_raw_parts(self.ptr as *const am::ChangeHash, self.len) };
Some(&slice[self.get_index()])
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts((&detail as *const Detail) as *const u8, USIZE_USIZE_USIZE_)
.try_into()
.unwrap()
}
}
}
/// \struct AMchangeHashes
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of change hashes.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMchangeHashes {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_],
}
impl AMchangeHashes {
pub fn new(change_hashes: &[am::ChangeHash]) -> Self {
Self {
detail: Detail::new(change_hashes, 0).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<&am::ChangeHash> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<&am::ChangeHash> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[am::ChangeHash]> for AMchangeHashes {
fn as_ref(&self) -> &[am::ChangeHash] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const am::ChangeHash, detail.len) }
}
}
impl Default for AMchangeHashes {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMchangeHashes
/// \brief Advances an iterator over a sequence of change hashes by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] change_hashes A pointer to an `AMchangeHashes` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p change_hashes `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesAdvance(change_hashes: *mut AMchangeHashes, n: isize) {
if let Some(change_hashes) = change_hashes.as_mut() {
change_hashes.advance(n);
};
}
/// \memberof AMchangeHashes
/// \brief Compares the sequences of change hashes underlying a pair of
/// iterators.
///
/// \param[in] change_hashes1 A pointer to an `AMchangeHashes` struct.
/// \param[in] change_hashes2 A pointer to an `AMchangeHashes` struct.
/// \return `-1` if \p change_hashes1 `<` \p change_hashes2, `0` if
/// \p change_hashes1 `==` \p change_hashes2 and `1` if
/// \p change_hashes1 `>` \p change_hashes2.
/// \pre \p change_hashes1 `!= NULL`.
/// \pre \p change_hashes2 `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes1 must be a valid pointer to an AMchangeHashes
/// change_hashes2 must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesCmp(
change_hashes1: *const AMchangeHashes,
change_hashes2: *const AMchangeHashes,
) -> isize {
match (change_hashes1.as_ref(), change_hashes2.as_ref()) {
(Some(change_hashes1), Some(change_hashes2)) => {
match change_hashes1.as_ref().cmp(change_hashes2.as_ref()) {
Ordering::Less => -1,
Ordering::Equal => 0,
Ordering::Greater => 1,
}
}
(None, Some(_)) => -1,
(Some(_), None) => 1,
(None, None) => 0,
}
}
/// \memberof AMchangeHashes
/// \brief Allocates an iterator over a sequence of change hashes and
/// initializes it from a sequence of byte spans.
///
/// \param[in] src A pointer to an array of `AMbyteSpan` structs.
/// \param[in] count The number of `AMbyteSpan` structs to copy from \p src.
/// \return A pointer to an `AMresult` struct containing an `AMchangeHashes`
/// struct.
/// \pre \p src `!= NULL`.
/// \pre `0 <` \p count `<= sizeof(`\p src`) / sizeof(AMbyteSpan)`.
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()`
/// in order to prevent a memory leak.
/// \internal
///
/// # Safety
/// src must be an AMbyteSpan array of size `>= count`
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesInit(src: *const AMbyteSpan, count: usize) -> *mut AMresult {
let mut change_hashes = Vec::<am::ChangeHash>::new();
for n in 0..count {
let byte_span = &*src.add(n);
let slice = std::slice::from_raw_parts(byte_span.src, byte_span.count);
match slice.try_into() {
Ok(change_hash) => {
change_hashes.push(change_hash);
}
Err(e) => {
return to_result(Err(e));
}
}
}
to_result(Ok::<Vec<am::ChangeHash>, am::InvalidChangeHashSlice>(
change_hashes,
))
}
/// \memberof AMchangeHashes
/// \brief Gets the change hash at the current position of an iterator over a
/// sequence of change hashes and then advances it by at most \p |n|
/// positions where the sign of \p n is relative to the iterator's
/// direction.
///
/// \param[in,out] change_hashes A pointer to an `AMchangeHashes` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return An `AMbyteSpan` struct with `.src == NULL` when \p change_hashes
/// was previously advanced past its forward/reverse limit.
/// \pre \p change_hashes `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesNext(
change_hashes: *mut AMchangeHashes,
n: isize,
) -> AMbyteSpan {
if let Some(change_hashes) = change_hashes.as_mut() {
if let Some(change_hash) = change_hashes.next(n) {
return change_hash.into();
}
}
Default::default()
}
/// \memberof AMchangeHashes
/// \brief Advances an iterator over a sequence of change hashes by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the change hash at its new
/// position.
///
/// \param[in,out] change_hashes A pointer to an `AMchangeHashes` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return An `AMbyteSpan` struct with `.src == NULL` when \p change_hashes is
/// presently advanced past its forward/reverse limit.
/// \pre \p change_hashes `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesPrev(
change_hashes: *mut AMchangeHashes,
n: isize,
) -> AMbyteSpan {
if let Some(change_hashes) = change_hashes.as_mut() {
if let Some(change_hash) = change_hashes.prev(n) {
return change_hash.into();
}
}
Default::default()
}
/// \memberof AMchangeHashes
/// \brief Gets the size of the sequence of change hashes underlying an
/// iterator.
///
/// \param[in] change_hashes A pointer to an `AMchangeHashes` struct.
/// \return The count of values in \p change_hashes.
/// \pre \p change_hashes `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesSize(change_hashes: *const AMchangeHashes) -> usize {
if let Some(change_hashes) = change_hashes.as_ref() {
change_hashes.len()
} else {
0
}
}
/// \memberof AMchangeHashes
/// \brief Creates an iterator over the same sequence of change hashes as the
/// given one but with the opposite position and direction.
///
/// \param[in] change_hashes A pointer to an `AMchangeHashes` struct.
/// \return An `AMchangeHashes` struct
/// \pre \p change_hashes `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesReversed(
change_hashes: *const AMchangeHashes,
) -> AMchangeHashes {
if let Some(change_hashes) = change_hashes.as_ref() {
change_hashes.reversed()
} else {
Default::default()
}
}
/// \memberof AMchangeHashes
/// \brief Creates an iterator at the starting position over the same sequence
/// of change hashes as the given one.
///
/// \param[in] change_hashes A pointer to an `AMchangeHashes` struct.
/// \return An `AMchangeHashes` struct
/// \pre \p change_hashes `!= NULL`.
/// \internal
///
/// #Safety
/// change_hashes must be a valid pointer to an AMchangeHashes
#[no_mangle]
pub unsafe extern "C" fn AMchangeHashesRewound(
change_hashes: *const AMchangeHashes,
) -> AMchangeHashes {
if let Some(change_hashes) = change_hashes.as_ref() {
change_hashes.rewound()
} else {
Default::default()
}
}

View file

@ -1,399 +0,0 @@
use automerge as am;
use std::collections::BTreeMap;
use std::ffi::c_void;
use std::mem::size_of;
use crate::byte_span::AMbyteSpan;
use crate::change::AMchange;
use crate::result::{to_result, AMresult};
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
storage: *mut c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(changes: &[am::Change], offset: isize, storage: &mut BTreeMap<usize, AMchange>) -> Self {
let storage: *mut BTreeMap<usize, AMchange> = storage;
Self {
len: changes.len(),
offset,
ptr: changes.as_ptr() as *const c_void,
storage: storage as *mut c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<*const AMchange> {
if self.is_stopped() {
return None;
}
let slice: &mut [am::Change] =
unsafe { std::slice::from_raw_parts_mut(self.ptr as *mut am::Change, self.len) };
let storage = unsafe { &mut *(self.storage as *mut BTreeMap<usize, AMchange>) };
let index = self.get_index();
let value = match storage.get_mut(&index) {
Some(value) => value,
None => {
storage.insert(index, AMchange::new(&mut slice[index]));
storage.get_mut(&index).unwrap()
}
};
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<*const AMchange> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &mut [am::Change] =
unsafe { std::slice::from_raw_parts_mut(self.ptr as *mut am::Change, self.len) };
let storage = unsafe { &mut *(self.storage as *mut BTreeMap<usize, AMchange>) };
let index = self.get_index();
Some(match storage.get_mut(&index) {
Some(value) => value,
None => {
storage.insert(index, AMchange::new(&mut slice[index]));
storage.get_mut(&index).unwrap()
}
})
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
storage: self.storage,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
storage: self.storage,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts(
(&detail as *const Detail) as *const u8,
USIZE_USIZE_USIZE_USIZE_,
)
.try_into()
.unwrap()
}
}
}
/// \struct AMchanges
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of changes.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMchanges {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_USIZE_],
}
impl AMchanges {
pub fn new(changes: &[am::Change], storage: &mut BTreeMap<usize, AMchange>) -> Self {
Self {
detail: Detail::new(changes, 0, &mut *storage).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<*const AMchange> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<*const AMchange> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[am::Change]> for AMchanges {
fn as_ref(&self) -> &[am::Change] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const am::Change, detail.len) }
}
}
impl Default for AMchanges {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMchanges
/// \brief Advances an iterator over a sequence of changes by at most \p |n|
/// positions where the sign of \p n is relative to the iterator's
/// direction.
///
/// \param[in,out] changes A pointer to an `AMchanges` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p changes `!= NULL`.
/// \internal
///
/// #Safety
/// changes must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesAdvance(changes: *mut AMchanges, n: isize) {
if let Some(changes) = changes.as_mut() {
changes.advance(n);
};
}
/// \memberof AMchanges
/// \brief Tests the equality of two sequences of changes underlying a pair of
/// iterators.
///
/// \param[in] changes1 A pointer to an `AMchanges` struct.
/// \param[in] changes2 A pointer to an `AMchanges` struct.
/// \return `true` if \p changes1 `==` \p changes2 and `false` otherwise.
/// \pre \p changes1 `!= NULL`.
/// \pre \p changes2 `!= NULL`.
/// \internal
///
/// #Safety
/// changes1 must be a valid pointer to an AMchanges
/// changes2 must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesEqual(
changes1: *const AMchanges,
changes2: *const AMchanges,
) -> bool {
match (changes1.as_ref(), changes2.as_ref()) {
(Some(changes1), Some(changes2)) => changes1.as_ref() == changes2.as_ref(),
(None, Some(_)) | (Some(_), None) | (None, None) => false,
}
}
/// \memberof AMchanges
/// \brief Allocates an iterator over a sequence of changes and initializes it
/// from a sequence of byte spans.
///
/// \param[in] src A pointer to an array of `AMbyteSpan` structs.
/// \param[in] count The number of `AMbyteSpan` structs to copy from \p src.
/// \return A pointer to an `AMresult` struct containing an `AMchanges` struct.
/// \pre \p src `!= NULL`.
/// \pre `0 <` \p count `<= sizeof(`\p src`) / sizeof(AMbyteSpan)`.
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()`
/// in order to prevent a memory leak.
/// \internal
///
/// # Safety
/// src must be an AMbyteSpan array of size `>= count`
#[no_mangle]
pub unsafe extern "C" fn AMchangesInit(src: *const AMbyteSpan, count: usize) -> *mut AMresult {
let mut changes = Vec::<am::Change>::new();
for n in 0..count {
let byte_span = &*src.add(n);
let slice = std::slice::from_raw_parts(byte_span.src, byte_span.count);
match slice.try_into() {
Ok(change) => {
changes.push(change);
}
Err(e) => {
return to_result(Err::<Vec<am::Change>, am::LoadChangeError>(e));
}
}
}
to_result(Ok::<Vec<am::Change>, am::LoadChangeError>(changes))
}
/// \memberof AMchanges
/// \brief Gets the change at the current position of an iterator over a
/// sequence of changes and then advances it by at most \p |n| positions
/// where the sign of \p n is relative to the iterator's direction.
///
/// \param[in,out] changes A pointer to an `AMchanges` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMchange` struct that's `NULL` when \p changes was
/// previously advanced past its forward/reverse limit.
/// \pre \p changes `!= NULL`.
/// \internal
///
/// #Safety
/// changes must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesNext(changes: *mut AMchanges, n: isize) -> *const AMchange {
if let Some(changes) = changes.as_mut() {
if let Some(change) = changes.next(n) {
return change;
}
}
std::ptr::null()
}
/// \memberof AMchanges
/// \brief Advances an iterator over a sequence of changes by at most \p |n|
/// positions where the sign of \p n is relative to the iterator's
/// direction and then gets the change at its new position.
///
/// \param[in,out] changes A pointer to an `AMchanges` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMchange` struct that's `NULL` when \p changes is
/// presently advanced past its forward/reverse limit.
/// \pre \p changes `!= NULL`.
/// \internal
///
/// #Safety
/// changes must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesPrev(changes: *mut AMchanges, n: isize) -> *const AMchange {
if let Some(changes) = changes.as_mut() {
if let Some(change) = changes.prev(n) {
return change;
}
}
std::ptr::null()
}
/// \memberof AMchanges
/// \brief Gets the size of the sequence of changes underlying an iterator.
///
/// \param[in] changes A pointer to an `AMchanges` struct.
/// \return The count of values in \p changes.
/// \pre \p changes `!= NULL`.
/// \internal
///
/// #Safety
/// changes must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesSize(changes: *const AMchanges) -> usize {
if let Some(changes) = changes.as_ref() {
changes.len()
} else {
0
}
}
/// \memberof AMchanges
/// \brief Creates an iterator over the same sequence of changes as the given
/// one but with the opposite position and direction.
///
/// \param[in] changes A pointer to an `AMchanges` struct.
/// \return An `AMchanges` struct.
/// \pre \p changes `!= NULL`.
/// \internal
///
/// #Safety
/// changes must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesReversed(changes: *const AMchanges) -> AMchanges {
if let Some(changes) = changes.as_ref() {
changes.reversed()
} else {
Default::default()
}
}
/// \memberof AMchanges
/// \brief Creates an iterator at the starting position over the same sequence
/// of changes as the given one.
///
/// \param[in] changes A pointer to an `AMchanges` struct.
/// \return An `AMchanges` struct
/// \pre \p changes `!= NULL`.
/// \internal
///
/// #Safety
/// changes must be a valid pointer to an AMchanges
#[no_mangle]
pub unsafe extern "C" fn AMchangesRewound(changes: *const AMchanges) -> AMchanges {
if let Some(changes) = changes.as_ref() {
changes.rewound()
} else {
Default::default()
}
}

File diff suppressed because it is too large Load diff

View file

@ -1,48 +1,46 @@
use automerge as am; use automerge as am;
use automerge::transaction::Transactable; use automerge::transaction::Transactable;
use automerge::ReadDoc;
use crate::byte_span::{to_str, AMbyteSpan}; use crate::byte_span::{to_str, AMbyteSpan};
use crate::change_hashes::AMchangeHashes; use crate::doc::{to_doc, to_doc_mut, AMdoc};
use crate::doc::{to_doc, to_doc_mut, to_obj_id, AMdoc}; use crate::items::AMitems;
use crate::obj::{to_obj_type, AMobjId, AMobjType}; use crate::obj::{to_obj_id, to_obj_type, AMobjId, AMobjType};
use crate::result::{to_result, AMresult}; use crate::result::{to_result, AMresult};
pub mod item;
pub mod items;
macro_rules! adjust { macro_rules! adjust {
($index:expr, $insert:expr, $len:expr) => {{ ($pos:expr, $insert:expr, $len:expr) => {{
// An empty object can only be inserted into. // An empty object can only be inserted into.
let insert = $insert || $len == 0; let insert = $insert || $len == 0;
let end = if insert { $len } else { $len - 1 }; let end = if insert { $len } else { $len - 1 };
if $index > end && $index != usize::MAX { if $pos > end && $pos != usize::MAX {
return AMresult::err(&format!("Invalid index {}", $index)).into(); return AMresult::error(&format!("Invalid pos {}", $pos)).into();
} }
(std::cmp::min($index, end), insert) (std::cmp::min($pos, end), insert)
}}; }};
} }
macro_rules! to_range { macro_rules! to_range {
($begin:expr, $end:expr) => {{ ($begin:expr, $end:expr) => {{
if $begin > $end { if $begin > $end {
return AMresult::err(&format!("Invalid range [{}-{})", $begin, $end)).into(); return AMresult::error(&format!("Invalid range [{}-{})", $begin, $end)).into();
}; };
($begin..$end) ($begin..$end)
}}; }};
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Deletes an index in a list object. /// \brief Deletes an item from a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index. /// \p obj_id or `SIZE_MAX` to indicate its last item.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -52,101 +50,109 @@ macro_rules! to_range {
pub unsafe extern "C" fn AMlistDelete( pub unsafe extern "C" fn AMlistDelete(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, _) = adjust!(index, false, doc.length(obj_id)); let (pos, _) = adjust!(pos, false, doc.length(obj_id));
to_result(doc.delete(obj_id, index)) to_result(doc.delete(obj_id, pos))
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Gets the current or historical value at an index in a list object. /// \brief Gets a current or historical item within a list object.
/// ///
/// \param[in] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index. /// \p obj_id or `SIZE_MAX` to indicate its last item.
/// \param[in] heads A pointer to an `AMchangeHashes` struct for a historical /// \param[in] heads A pointer to an `AMitems` struct with `AM_VAL_TYPE_CHANGE_HASH`
/// value or `NULL` for the current value. /// items to select a historical item at \p pos or `NULL`
/// \return A pointer to an `AMresult` struct that doesn't contain a void. /// to select the current item at \p pos.
/// \pre \p doc `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AMitem` struct.
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre \p doc `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// heads must be a valid pointer to an AMchangeHashes or std::ptr::null() /// heads must be a valid pointer to an AMitems or std::ptr::null()
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMlistGet( pub unsafe extern "C" fn AMlistGet(
doc: *const AMdoc, doc: *const AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
heads: *const AMchangeHashes, heads: *const AMitems,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc!(doc); let doc = to_doc!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, _) = adjust!(index, false, doc.length(obj_id)); let (pos, _) = adjust!(pos, false, doc.length(obj_id));
to_result(match heads.as_ref() { match heads.as_ref() {
None => doc.get(obj_id, index), None => to_result(doc.get(obj_id, pos)),
Some(heads) => doc.get_at(obj_id, index, heads.as_ref()), Some(heads) => match <Vec<am::ChangeHash>>::try_from(heads) {
}) Ok(heads) => to_result(doc.get_at(obj_id, pos, &heads)),
Err(e) => AMresult::error(&e.to_string()).into(),
},
}
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Gets all of the historical values at an index in a list object until /// \brief Gets all of the historical items at a position within a list object
/// its current one or a specific one. /// until its current one or a specific one.
/// ///
/// \param[in] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index. /// \p obj_id or `SIZE_MAX` to indicate its last item.
/// \param[in] heads A pointer to an `AMchangeHashes` struct for a historical /// \param[in] heads A pointer to an `AMitems` struct with `AM_VAL_TYPE_CHANGE_HASH`
/// last value or `NULL` for the current last value. /// items to select a historical last item or `NULL` to select
/// \return A pointer to an `AMresult` struct containing an `AMobjItems` struct. /// the current last item.
/// \pre \p doc `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AMitems` struct.
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre \p doc `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// heads must be a valid pointer to an AMchangeHashes or std::ptr::null() /// heads must be a valid pointer to an AMitems or std::ptr::null()
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMlistGetAll( pub unsafe extern "C" fn AMlistGetAll(
doc: *const AMdoc, doc: *const AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
heads: *const AMchangeHashes, heads: *const AMitems,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc!(doc); let doc = to_doc!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, _) = adjust!(index, false, doc.length(obj_id)); let (pos, _) = adjust!(pos, false, doc.length(obj_id));
match heads.as_ref() { match heads.as_ref() {
None => to_result(doc.get_all(obj_id, index)), None => to_result(doc.get_all(obj_id, pos)),
Some(heads) => to_result(doc.get_all_at(obj_id, index, heads.as_ref())), Some(heads) => match <Vec<am::ChangeHash>>::try_from(heads) {
Ok(heads) => to_result(doc.get_all_at(obj_id, pos, &heads)),
Err(e) => AMresult::error(&e.to_string()).into(),
},
} }
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Increments a counter at an index in a list object by the given /// \brief Increments a counter value in an item within a list object by the
/// value. /// given value.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index. /// \p obj_id or `SIZE_MAX` to indicate its last item.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -156,32 +162,33 @@ pub unsafe extern "C" fn AMlistGetAll(
pub unsafe extern "C" fn AMlistIncrement( pub unsafe extern "C" fn AMlistIncrement(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
value: i64, value: i64,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, _) = adjust!(index, false, doc.length(obj_id)); let (pos, _) = adjust!(pos, false, doc.length(obj_id));
to_result(doc.increment(obj_id, index, value)) to_result(doc.increment(obj_id, pos, value))
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a boolean as the value at an index in a list object. /// \brief Puts a boolean value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A boolean. /// \param[in] value A boolean.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -191,84 +198,85 @@ pub unsafe extern "C" fn AMlistIncrement(
pub unsafe extern "C" fn AMlistPutBool( pub unsafe extern "C" fn AMlistPutBool(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: bool, value: bool,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
let value = am::ScalarValue::Boolean(value); let value = am::ScalarValue::Boolean(value);
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a sequence of bytes as the value at an index in a list object. /// \brief Puts an array of bytes value at a position within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p src before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p src over \p index. /// \p pos instead of putting \p value into the item at
/// \param[in] src A pointer to an array of bytes. /// \p pos.
/// \param[in] count The number of bytes to copy from \p src. /// \param[in] value A view onto the array of bytes to copy from as an
/// \return A pointer to an `AMresult` struct containing a void. /// `AMbyteSpan` struct.
/// \pre \p doc `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre \p doc `!= NULL`
/// \pre \p src `!= NULL`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \pre `0 <` \p count `<= sizeof(`\p src`)`. /// \pre \p value.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `0 <` \p value.count `<= sizeof(`\p value.src `)`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// src must be a byte array of size `>= count` /// value.src must be a byte array of length >= value.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMlistPutBytes( pub unsafe extern "C" fn AMlistPutBytes(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
val: AMbyteSpan, value: AMbyteSpan,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
let mut value = Vec::new(); let value: Vec<u8> = (&value).into();
value.extend_from_slice(std::slice::from_raw_parts(val.src, val.count));
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a CRDT counter as the value at an index in a list object. /// \brief Puts a CRDT counter value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -278,38 +286,39 @@ pub unsafe extern "C" fn AMlistPutBytes(
pub unsafe extern "C" fn AMlistPutCounter( pub unsafe extern "C" fn AMlistPutCounter(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: i64, value: i64,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
let value = am::ScalarValue::Counter(value.into()); let value = am::ScalarValue::Counter(value.into());
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a float as the value at an index in a list object. /// \brief Puts a float value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A 64-bit float. /// \param[in] value A 64-bit float.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -319,37 +328,38 @@ pub unsafe extern "C" fn AMlistPutCounter(
pub unsafe extern "C" fn AMlistPutF64( pub unsafe extern "C" fn AMlistPutF64(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: f64, value: f64,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a signed integer as the value at an index in a list object. /// \brief Puts a signed integer value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -359,36 +369,37 @@ pub unsafe extern "C" fn AMlistPutF64(
pub unsafe extern "C" fn AMlistPutInt( pub unsafe extern "C" fn AMlistPutInt(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: i64, value: i64,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts null as the value at an index in a list object. /// \brief Puts a null value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \return A pointer to an `AMresult` struct containing a void. /// \p pos.
/// \pre \p doc `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre \p doc `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -398,38 +409,37 @@ pub unsafe extern "C" fn AMlistPutInt(
pub unsafe extern "C" fn AMlistPutNull( pub unsafe extern "C" fn AMlistPutNull(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, ()) doc.insert(obj_id, pos, ())
} else { } else {
doc.put(obj_id, index, ()) doc.put(obj_id, pos, ())
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts an empty object as the value at an index in a list object. /// \brief Puts an empty object value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] obj_type An `AMobjIdType` enum tag. /// \param[in] obj_type An `AMobjIdType` enum tag.
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_OBJ_TYPE` item.
/// `AMobjId` struct. /// \pre \p doc `!= NULL`
/// \pre \p doc `!= NULL`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \warning The returned `AMresult` struct pointer must be passed to
/// \pre \p obj_type != `AM_OBJ_TYPE_VOID`. /// `AMresultFree()` in order to avoid a memory leak.
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()`
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -439,82 +449,85 @@ pub unsafe extern "C" fn AMlistPutNull(
pub unsafe extern "C" fn AMlistPutObject( pub unsafe extern "C" fn AMlistPutObject(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
obj_type: AMobjType, obj_type: AMobjType,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
let object = to_obj_type!(obj_type); let obj_type = to_obj_type!(obj_type);
to_result(if insert { to_result(if insert {
doc.insert_object(obj_id, index, object) (doc.insert_object(obj_id, pos, obj_type), obj_type)
} else { } else {
doc.put_object(obj_id, index, object) (doc.put_object(obj_id, pos, obj_type), obj_type)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a UTF-8 string as the value at an index in a list object. /// \brief Puts a UTF-8 string value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A UTF-8 string view as an `AMbyteSpan` struct. /// \param[in] value A UTF-8 string view as an `AMbyteSpan` struct.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \pre \p value `!= NULL`. /// \pre \p value.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre `0 <` \p value.count `<= sizeof(`\p value.src `)`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// value must be a null-terminated array of `c_char` /// value.src must be a byte array of length >= value.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMlistPutStr( pub unsafe extern "C" fn AMlistPutStr(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: AMbyteSpan, value: AMbyteSpan,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
let value = to_str!(value); let value = to_str!(value);
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a *nix timestamp (milliseconds) as the value at an index in a /// \brief Puts a *nix timestamp (milliseconds) value into an item within a
/// list object. /// list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -524,38 +537,39 @@ pub unsafe extern "C" fn AMlistPutStr(
pub unsafe extern "C" fn AMlistPutTimestamp( pub unsafe extern "C" fn AMlistPutTimestamp(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: i64, value: i64,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
let value = am::ScalarValue::Timestamp(value); let value = am::ScalarValue::Timestamp(value);
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts an unsigned integer as the value at an index in a list object. /// \brief Puts an unsigned integer value into an item within a list object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] index An index in the list object identified by \p obj_id or /// \param[in] pos The position of an item within the list object identified by
/// `SIZE_MAX` to indicate its last index if \p insert /// \p obj_id or `SIZE_MAX` to indicate its last item if
/// `== false` or one past its last index if \p insert /// \p insert `== false` or one past its last item if
/// `== true`. /// \p insert `== true`.
/// \param[in] insert A flag to insert \p value before \p index instead of /// \param[in] insert A flag for inserting a new item for \p value before
/// writing \p value over \p index. /// \p pos instead of putting \p value into the item at
/// \p pos.
/// \param[in] value A 64-bit unsigned integer. /// \param[in] value A 64-bit unsigned integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre `0 <=` \p index `<= AMobjSize(`\p obj_id`)` or \p index `== SIZE_MAX`. /// \pre `0 <=` \p pos `<= AMobjSize(`\p obj_id `)` or \p pos `== SIZE_MAX`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -565,56 +579,58 @@ pub unsafe extern "C" fn AMlistPutTimestamp(
pub unsafe extern "C" fn AMlistPutUint( pub unsafe extern "C" fn AMlistPutUint(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
index: usize, pos: usize,
insert: bool, insert: bool,
value: u64, value: u64,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let (index, insert) = adjust!(index, insert, doc.length(obj_id)); let (pos, insert) = adjust!(pos, insert, doc.length(obj_id));
to_result(if insert { to_result(if insert {
doc.insert(obj_id, index, value) doc.insert(obj_id, pos, value)
} else { } else {
doc.put(obj_id, index, value) doc.put(obj_id, pos, value)
}) })
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Gets the current or historical indices and values of the list object /// \brief Gets the current or historical items in the list object within the
/// within the given range. /// given range.
/// ///
/// \param[in] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] begin The first index in a range of indices. /// \param[in] begin The first pos in a range of indices.
/// \param[in] end At least one past the last index in a range of indices. /// \param[in] end At least one past the last pos in a range of indices.
/// \param[in] heads A pointer to an `AMchangeHashes` struct for historical /// \param[in] heads A pointer to an `AMitems` struct with `AM_VAL_TYPE_CHANGE_HASH`
/// indices and values or `NULL` for current indices and /// items to select historical items or `NULL` to select
/// values. /// current items.
/// \return A pointer to an `AMresult` struct containing an `AMlistItems` /// \return A pointer to an `AMresult` struct with an `AMitems` struct.
/// struct. /// \pre \p doc `!= NULL`
/// \pre \p doc `!= NULL`. /// \pre \p begin `<=` \p end `<= SIZE_MAX`
/// \pre \p begin `<=` \p end `<= SIZE_MAX`. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// heads must be a valid pointer to an AMchangeHashes or std::ptr::null() /// heads must be a valid pointer to an AMitems or std::ptr::null()
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMlistRange( pub unsafe extern "C" fn AMlistRange(
doc: *const AMdoc, doc: *const AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
begin: usize, begin: usize,
end: usize, end: usize,
heads: *const AMchangeHashes, heads: *const AMitems,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc!(doc); let doc = to_doc!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let range = to_range!(begin, end); let range = to_range!(begin, end);
match heads.as_ref() { match heads.as_ref() {
None => to_result(doc.list_range(obj_id, range)), None => to_result(doc.list_range(obj_id, range)),
Some(heads) => to_result(doc.list_range_at(obj_id, range, heads.as_ref())), Some(heads) => match <Vec<am::ChangeHash>>::try_from(heads) {
Ok(heads) => to_result(doc.list_range_at(obj_id, range, &heads)),
Err(e) => AMresult::error(&e.to_string()).into(),
},
} }
} }

View file

@ -1,97 +0,0 @@
use automerge as am;
use crate::obj::AMobjId;
use crate::result::AMvalue;
/// \struct AMlistItem
/// \installed_headerfile
/// \brief An item in a list object.
pub struct AMlistItem {
/// The index of an item in a list object.
index: usize,
/// The object identifier of an item in a list object.
obj_id: AMobjId,
/// The value of an item in a list object.
value: am::Value<'static>,
}
impl AMlistItem {
pub fn new(index: usize, value: am::Value<'static>, obj_id: am::ObjId) -> Self {
Self {
index,
obj_id: AMobjId::new(obj_id),
value,
}
}
}
impl PartialEq for AMlistItem {
fn eq(&self, other: &Self) -> bool {
self.index == other.index && self.obj_id == other.obj_id && self.value == other.value
}
}
/*
impl From<&AMlistItem> for (usize, am::Value<'static>, am::ObjId) {
fn from(list_item: &AMlistItem) -> Self {
(list_item.index, list_item.value.0.clone(), list_item.obj_id.as_ref().clone())
}
}
*/
/// \memberof AMlistItem
/// \brief Gets the index of an item in a list object.
///
/// \param[in] list_item A pointer to an `AMlistItem` struct.
/// \return A 64-bit unsigned integer.
/// \pre \p list_item `!= NULL`.
/// \internal
///
/// # Safety
/// list_item must be a valid pointer to an AMlistItem
#[no_mangle]
pub unsafe extern "C" fn AMlistItemIndex(list_item: *const AMlistItem) -> usize {
if let Some(list_item) = list_item.as_ref() {
list_item.index
} else {
usize::MAX
}
}
/// \memberof AMlistItem
/// \brief Gets the object identifier of an item in a list object.
///
/// \param[in] list_item A pointer to an `AMlistItem` struct.
/// \return A pointer to an `AMobjId` struct.
/// \pre \p list_item `!= NULL`.
/// \internal
///
/// # Safety
/// list_item must be a valid pointer to an AMlistItem
#[no_mangle]
pub unsafe extern "C" fn AMlistItemObjId(list_item: *const AMlistItem) -> *const AMobjId {
if let Some(list_item) = list_item.as_ref() {
&list_item.obj_id
} else {
std::ptr::null()
}
}
/// \memberof AMlistItem
/// \brief Gets the value of an item in a list object.
///
/// \param[in] list_item A pointer to an `AMlistItem` struct.
/// \return An `AMvalue` struct.
/// \pre \p list_item `!= NULL`.
/// \internal
///
/// # Safety
/// list_item must be a valid pointer to an AMlistItem
#[no_mangle]
pub unsafe extern "C" fn AMlistItemValue<'a>(list_item: *const AMlistItem) -> AMvalue<'a> {
if let Some(list_item) = list_item.as_ref() {
(&list_item.value).into()
} else {
AMvalue::Void
}
}

View file

@ -1,348 +0,0 @@
use std::ffi::c_void;
use std::mem::size_of;
use crate::doc::list::item::AMlistItem;
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(list_items: &[AMlistItem], offset: isize) -> Self {
Self {
len: list_items.len(),
offset,
ptr: list_items.as_ptr() as *const c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<&AMlistItem> {
if self.is_stopped() {
return None;
}
let slice: &[AMlistItem] =
unsafe { std::slice::from_raw_parts(self.ptr as *const AMlistItem, self.len) };
let value = &slice[self.get_index()];
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<&AMlistItem> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &[AMlistItem] =
unsafe { std::slice::from_raw_parts(self.ptr as *const AMlistItem, self.len) };
Some(&slice[self.get_index()])
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts((&detail as *const Detail) as *const u8, USIZE_USIZE_USIZE_)
.try_into()
.unwrap()
}
}
}
/// \struct AMlistItems
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of list object items.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMlistItems {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_],
}
impl AMlistItems {
pub fn new(list_items: &[AMlistItem]) -> Self {
Self {
detail: Detail::new(list_items, 0).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<&AMlistItem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<&AMlistItem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[AMlistItem]> for AMlistItems {
fn as_ref(&self) -> &[AMlistItem] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const AMlistItem, detail.len) }
}
}
impl Default for AMlistItems {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMlistItems
/// \brief Advances an iterator over a sequence of list object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] list_items A pointer to an `AMlistItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p list_items `!= NULL`.
/// \internal
///
/// #Safety
/// list_items must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsAdvance(list_items: *mut AMlistItems, n: isize) {
if let Some(list_items) = list_items.as_mut() {
list_items.advance(n);
};
}
/// \memberof AMlistItems
/// \brief Tests the equality of two sequences of list object items underlying
/// a pair of iterators.
///
/// \param[in] list_items1 A pointer to an `AMlistItems` struct.
/// \param[in] list_items2 A pointer to an `AMlistItems` struct.
/// \return `true` if \p list_items1 `==` \p list_items2 and `false` otherwise.
/// \pre \p list_items1 `!= NULL`.
/// \pre \p list_items2 `!= NULL`.
/// \internal
///
/// #Safety
/// list_items1 must be a valid pointer to an AMlistItems
/// list_items2 must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsEqual(
list_items1: *const AMlistItems,
list_items2: *const AMlistItems,
) -> bool {
match (list_items1.as_ref(), list_items2.as_ref()) {
(Some(list_items1), Some(list_items2)) => list_items1.as_ref() == list_items2.as_ref(),
(None, Some(_)) | (Some(_), None) | (None, None) => false,
}
}
/// \memberof AMlistItems
/// \brief Gets the list object item at the current position of an iterator
/// over a sequence of list object items and then advances it by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] list_items A pointer to an `AMlistItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMlistItem` struct that's `NULL` when
/// \p list_items was previously advanced past its forward/reverse
/// limit.
/// \pre \p list_items `!= NULL`.
/// \internal
///
/// #Safety
/// list_items must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsNext(
list_items: *mut AMlistItems,
n: isize,
) -> *const AMlistItem {
if let Some(list_items) = list_items.as_mut() {
if let Some(list_item) = list_items.next(n) {
return list_item;
}
}
std::ptr::null()
}
/// \memberof AMlistItems
/// \brief Advances an iterator over a sequence of list object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the list object item at its new
/// position.
///
/// \param[in,out] list_items A pointer to an `AMlistItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMlistItem` struct that's `NULL` when
/// \p list_items is presently advanced past its forward/reverse limit.
/// \pre \p list_items `!= NULL`.
/// \internal
///
/// #Safety
/// list_items must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsPrev(
list_items: *mut AMlistItems,
n: isize,
) -> *const AMlistItem {
if let Some(list_items) = list_items.as_mut() {
if let Some(list_item) = list_items.prev(n) {
return list_item;
}
}
std::ptr::null()
}
/// \memberof AMlistItems
/// \brief Gets the size of the sequence of list object items underlying an
/// iterator.
///
/// \param[in] list_items A pointer to an `AMlistItems` struct.
/// \return The count of values in \p list_items.
/// \pre \p list_items `!= NULL`.
/// \internal
///
/// #Safety
/// list_items must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsSize(list_items: *const AMlistItems) -> usize {
if let Some(list_items) = list_items.as_ref() {
list_items.len()
} else {
0
}
}
/// \memberof AMlistItems
/// \brief Creates an iterator over the same sequence of list object items as
/// the given one but with the opposite position and direction.
///
/// \param[in] list_items A pointer to an `AMlistItems` struct.
/// \return An `AMlistItems` struct
/// \pre \p list_items `!= NULL`.
/// \internal
///
/// #Safety
/// list_items must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsReversed(list_items: *const AMlistItems) -> AMlistItems {
if let Some(list_items) = list_items.as_ref() {
list_items.reversed()
} else {
Default::default()
}
}
/// \memberof AMlistItems
/// \brief Creates an iterator at the starting position over the same sequence
/// of list object items as the given one.
///
/// \param[in] list_items A pointer to an `AMlistItems` struct.
/// \return An `AMlistItems` struct
/// \pre \p list_items `!= NULL`.
/// \internal
///
/// #Safety
/// list_items must be a valid pointer to an AMlistItems
#[no_mangle]
pub unsafe extern "C" fn AMlistItemsRewound(list_items: *const AMlistItems) -> AMlistItems {
if let Some(list_items) = list_items.as_ref() {
list_items.rewound()
} else {
Default::default()
}
}

View file

@ -1,32 +1,31 @@
use automerge as am; use automerge as am;
use automerge::transaction::Transactable; use automerge::transaction::Transactable;
use automerge::ReadDoc;
use crate::byte_span::{to_str, AMbyteSpan}; use crate::byte_span::{to_str, AMbyteSpan};
use crate::change_hashes::AMchangeHashes; use crate::doc::{to_doc, to_doc_mut, AMdoc};
use crate::doc::{to_doc, to_doc_mut, to_obj_id, AMdoc}; use crate::items::AMitems;
use crate::obj::{to_obj_type, AMobjId, AMobjType}; use crate::obj::{to_obj_id, to_obj_type, AMobjId, AMobjType};
use crate::result::{to_result, AMresult}; use crate::result::{to_result, AMresult};
pub mod item;
pub mod items;
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Deletes a key in a map object. /// \brief Deletes an item from a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key The UTF-8 string view key of an item within the map object
/// \p obj_id as an `AMbyteSpan` struct. /// identified by \p obj_id as an `AMbyteSpan` struct.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapDelete( pub unsafe extern "C" fn AMmapDelete(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -39,96 +38,107 @@ pub unsafe extern "C" fn AMmapDelete(
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Gets the current or historical value for a key in a map object. /// \brief Gets a current or historical item within a map object.
/// ///
/// \param[in] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key The UTF-8 string view key of an item within the map object
/// \p obj_id as an `AMbyteSpan` struct. /// identified by \p obj_id as an `AMbyteSpan` struct.
/// \param[in] heads A pointer to an `AMchangeHashes` struct for a historical /// \param[in] heads A pointer to an `AMitems` struct with `AM_VAL_TYPE_CHANGE_HASH`
/// value or `NULL` for the current value. /// items to select a historical item at \p key or `NULL`
/// \return A pointer to an `AMresult` struct that doesn't contain a void. /// to select the current item at \p key.
/// \pre \p doc `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AMitem` struct.
/// \pre \p key `!= NULL`. /// \pre \p doc `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre \p key.src `!= NULL`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// heads must be a valid pointer to an AMchangeHashes or std::ptr::null() /// key.src must be a byte array of length >= key.count
/// heads must be a valid pointer to an AMitems or std::ptr::null()
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapGet( pub unsafe extern "C" fn AMmapGet(
doc: *const AMdoc, doc: *const AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
key: AMbyteSpan, key: AMbyteSpan,
heads: *const AMchangeHashes, heads: *const AMitems,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc!(doc); let doc = to_doc!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let key = to_str!(key); let key = to_str!(key);
match heads.as_ref() { match heads.as_ref() {
None => to_result(doc.get(obj_id, key)), None => to_result(doc.get(obj_id, key)),
Some(heads) => to_result(doc.get_at(obj_id, key, heads.as_ref())), Some(heads) => match <Vec<am::ChangeHash>>::try_from(heads) {
Ok(heads) => to_result(doc.get_at(obj_id, key, &heads)),
Err(e) => AMresult::error(&e.to_string()).into(),
},
} }
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Gets all of the historical values for a key in a map object until /// \brief Gets all of the historical items at a key within a map object until
/// its current one or a specific one. /// its current one or a specific one.
/// ///
/// \param[in] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key The UTF-8 string view key of an item within the map object
/// \p obj_id as an `AMbyteSpan` struct. /// identified by \p obj_id as an `AMbyteSpan` struct.
/// \param[in] heads A pointer to an `AMchangeHashes` struct for a historical /// \param[in] heads A pointer to an `AMitems` struct with `AM_VAL_TYPE_CHANGE_HASH`
/// last value or `NULL` for the current last value. /// items to select a historical last item or `NULL` to
/// \return A pointer to an `AMresult` struct containing an `AMobjItems` struct. /// select the current last item.
/// \pre \p doc `!= NULL`. /// \return A pointer to an `AMresult` struct with an `AMItems` struct.
/// \pre \p key `!= NULL`. /// \pre \p doc `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre \p key.src `!= NULL`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// heads must be a valid pointer to an AMchangeHashes or std::ptr::null() /// key.src must be a byte array of length >= key.count
/// heads must be a valid pointer to an AMitems or std::ptr::null()
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapGetAll( pub unsafe extern "C" fn AMmapGetAll(
doc: *const AMdoc, doc: *const AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
key: AMbyteSpan, key: AMbyteSpan,
heads: *const AMchangeHashes, heads: *const AMitems,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc!(doc); let doc = to_doc!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let key = to_str!(key); let key = to_str!(key);
match heads.as_ref() { match heads.as_ref() {
None => to_result(doc.get_all(obj_id, key)), None => to_result(doc.get_all(obj_id, key)),
Some(heads) => to_result(doc.get_all_at(obj_id, key, heads.as_ref())), Some(heads) => match <Vec<am::ChangeHash>>::try_from(heads) {
Ok(heads) => to_result(doc.get_all_at(obj_id, key, &heads)),
Err(e) => AMresult::error(&e.to_string()).into(),
},
} }
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Increments a counter for a key in a map object by the given value. /// \brief Increments a counter at a key in a map object by the given value.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key The UTF-8 string view key of an item within the map object
/// \p obj_id as an `AMbyteSpan` struct. /// identified by \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapIncrement( pub unsafe extern "C" fn AMmapIncrement(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -144,21 +154,22 @@ pub unsafe extern "C" fn AMmapIncrement(
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a boolean as the value of a key in a map object. /// \brief Puts a boolean as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key The UTF-8 string view key of an item within the map object
/// \p obj_id as an `AMbyteSpan` struct. /// identified by \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A boolean. /// \param[in] value A boolean.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutBool( pub unsafe extern "C" fn AMmapPutBool(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -172,59 +183,58 @@ pub unsafe extern "C" fn AMmapPutBool(
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a sequence of bytes as the value of a key in a map object. /// \brief Puts an array of bytes value at a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key The UTF-8 string view key of an item within the map object
/// \p obj_id as an `AMbyteSpan` struct. /// identified by \p obj_id as an `AMbyteSpan` struct.
/// \param[in] src A pointer to an array of bytes. /// \param[in] value A view onto an array of bytes as an `AMbyteSpan` struct.
/// \param[in] count The number of bytes to copy from \p src. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \return A pointer to an `AMresult` struct containing a void. /// \pre \p doc `!= NULL`
/// \pre \p doc `!= NULL`. /// \pre \p key.src `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p value.src `!= NULL`
/// \pre \p src `!= NULL`. /// \pre `0 <` \p value.count `<= sizeof(`\p value.src `)`
/// \pre `0 <` \p count `<= sizeof(`\p src`)`. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// src must be a byte array of size `>= count` /// key.src must be a byte array of length >= key.count
/// value.src must be a byte array of length >= value.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutBytes( pub unsafe extern "C" fn AMmapPutBytes(
doc: *mut AMdoc, doc: *mut AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
key: AMbyteSpan, key: AMbyteSpan,
val: AMbyteSpan, value: AMbyteSpan,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let key = to_str!(key); let key = to_str!(key);
let mut vec = Vec::new(); to_result(doc.put(to_obj_id!(obj_id), key, Vec::<u8>::from(&value)))
vec.extend_from_slice(std::slice::from_raw_parts(val.src, val.count));
to_result(doc.put(to_obj_id!(obj_id), key, vec))
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a CRDT counter as the value of a key in a map object. /// \brief Puts a CRDT counter as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutCounter( pub unsafe extern "C" fn AMmapPutCounter(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -244,20 +254,21 @@ pub unsafe extern "C" fn AMmapPutCounter(
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts null as the value of a key in a map object. /// \brief Puts null as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutNull( pub unsafe extern "C" fn AMmapPutNull(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -272,23 +283,22 @@ pub unsafe extern "C" fn AMmapPutNull(
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts an empty object as the value of a key in a map object. /// \brief Puts an empty object as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] obj_type An `AMobjIdType` enum tag. /// \param[in] obj_type An `AMobjIdType` enum tag.
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_OBJ_TYPE` item.
/// `AMobjId` struct. /// \pre \p doc `!= NULL`
/// \pre \p doc `!= NULL`. /// \pre \p key.src `!= NULL`
/// \pre \p key `!= NULL`. /// \warning The returned `AMresult` struct pointer must be passed to
/// \pre \p obj_type != `AM_OBJ_TYPE_VOID`. /// `AMresultFree()` in order to avoid a memory leak.
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()`
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutObject( pub unsafe extern "C" fn AMmapPutObject(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -298,27 +308,29 @@ pub unsafe extern "C" fn AMmapPutObject(
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc_mut!(doc); let doc = to_doc_mut!(doc);
let key = to_str!(key); let key = to_str!(key);
to_result(doc.put_object(to_obj_id!(obj_id), key, to_obj_type!(obj_type))) let obj_type = to_obj_type!(obj_type);
to_result((doc.put_object(to_obj_id!(obj_id), key, obj_type), obj_type))
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a float as the value of a key in a map object. /// \brief Puts a float as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A 64-bit float. /// \param[in] value A 64-bit float.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutF64( pub unsafe extern "C" fn AMmapPutF64(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -334,21 +346,22 @@ pub unsafe extern "C" fn AMmapPutF64(
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a signed integer as the value of a key in a map object. /// \brief Puts a signed integer as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutInt( pub unsafe extern "C" fn AMmapPutInt(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -364,21 +377,22 @@ pub unsafe extern "C" fn AMmapPutInt(
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts a UTF-8 string as the value of a key in a map object. /// \brief Puts a UTF-8 string as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A UTF-8 string view as an `AMbyteSpan` struct. /// \param[in] value A UTF-8 string view as an `AMbyteSpan` struct.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutStr( pub unsafe extern "C" fn AMmapPutStr(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -394,21 +408,22 @@ pub unsafe extern "C" fn AMmapPutStr(
/// \brief Puts a *nix timestamp (milliseconds) as the value of a key in a map /// \brief Puts a *nix timestamp (milliseconds) as the value of a key in a map
/// object. /// object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A 64-bit signed integer. /// \param[in] value A 64-bit signed integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutTimestamp( pub unsafe extern "C" fn AMmapPutTimestamp(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -424,21 +439,22 @@ pub unsafe extern "C" fn AMmapPutTimestamp(
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Puts an unsigned integer as the value of a key in a map object. /// \brief Puts an unsigned integer as the value of a key in a map object.
/// ///
/// \param[in,out] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] key A UTF-8 string view key for the map object identified by /// \param[in] key A UTF-8 string view key for the map object identified by
/// \p obj_id as an `AMbyteSpan` struct. /// \p obj_id as an `AMbyteSpan` struct.
/// \param[in] value A 64-bit unsigned integer. /// \param[in] value A 64-bit unsigned integer.
/// \return A pointer to an `AMresult` struct containing a void. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_VOID` item.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \pre \p key `!= NULL`. /// \pre \p key.src `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// key.src must be a byte array of length >= key.count
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapPutUint( pub unsafe extern "C" fn AMmapPutUint(
doc: *mut AMdoc, doc: *mut AMdoc,
@ -452,71 +468,82 @@ pub unsafe extern "C" fn AMmapPutUint(
} }
/// \memberof AMdoc /// \memberof AMdoc
/// \brief Gets the current or historical keys and values of the map object /// \brief Gets the current or historical items of the map object within the
/// within the given range. /// given range.
/// ///
/// \param[in] doc A pointer to an `AMdoc` struct. /// \param[in] doc A pointer to an `AMdoc` struct.
/// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`. /// \param[in] obj_id A pointer to an `AMobjId` struct or `AM_ROOT`.
/// \param[in] begin The first key in a subrange or `AMstr(NULL)` to indicate the /// \param[in] begin The first key in a subrange or `AMstr(NULL)` to indicate the
/// absolute first key. /// absolute first key.
/// \param[in] end The key one past the last key in a subrange or `AMstr(NULL)` to /// \param[in] end The key one past the last key in a subrange or `AMstr(NULL)`
/// indicate one past the absolute last key. /// to indicate one past the absolute last key.
/// \param[in] heads A pointer to an `AMchangeHashes` struct for historical /// \param[in] heads A pointer to an `AMitems` struct with `AM_VAL_TYPE_CHANGE_HASH`
/// keys and values or `NULL` for current keys and values. /// items to select historical items or `NULL` to select
/// \return A pointer to an `AMresult` struct containing an `AMmapItems` /// current items.
/// struct. /// \return A pointer to an `AMresult` struct with an `AMitems` struct.
/// \pre \p doc `!= NULL`. /// \pre \p doc `!= NULL`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \warning The returned `AMresult` struct pointer must be passed to
/// in order to prevent a memory leak. /// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// doc must be a valid pointer to an AMdoc /// doc must be a valid pointer to an AMdoc
/// obj_id must be a valid pointer to an AMobjId or std::ptr::null() /// obj_id must be a valid pointer to an AMobjId or std::ptr::null()
/// heads must be a valid pointer to an AMchangeHashes or std::ptr::null() /// begin.src must be a byte array of length >= begin.count or std::ptr::null()
/// end.src must be a byte array of length >= end.count or std::ptr::null()
/// heads must be a valid pointer to an AMitems or std::ptr::null()
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMmapRange( pub unsafe extern "C" fn AMmapRange(
doc: *const AMdoc, doc: *const AMdoc,
obj_id: *const AMobjId, obj_id: *const AMobjId,
begin: AMbyteSpan, begin: AMbyteSpan,
end: AMbyteSpan, end: AMbyteSpan,
heads: *const AMchangeHashes, heads: *const AMitems,
) -> *mut AMresult { ) -> *mut AMresult {
let doc = to_doc!(doc); let doc = to_doc!(doc);
let obj_id = to_obj_id!(obj_id); let obj_id = to_obj_id!(obj_id);
let heads = match heads.as_ref() {
None => None,
Some(heads) => match <Vec<am::ChangeHash>>::try_from(heads) {
Ok(heads) => Some(heads),
Err(e) => {
return AMresult::error(&e.to_string()).into();
}
},
};
match (begin.is_null(), end.is_null()) { match (begin.is_null(), end.is_null()) {
(false, false) => { (false, false) => {
let (begin, end) = (to_str!(begin).to_string(), to_str!(end).to_string()); let (begin, end) = (to_str!(begin).to_string(), to_str!(end).to_string());
if begin > end { if begin > end {
return AMresult::err(&format!("Invalid range [{}-{})", begin, end)).into(); return AMresult::error(&format!("Invalid range [{}-{})", begin, end)).into();
}; };
let bounds = begin..end; let bounds = begin..end;
if let Some(heads) = heads.as_ref() { if let Some(heads) = heads {
to_result(doc.map_range_at(obj_id, bounds, heads.as_ref())) to_result(doc.map_range_at(obj_id, bounds, &heads))
} else { } else {
to_result(doc.map_range(obj_id, bounds)) to_result(doc.map_range(obj_id, bounds))
} }
} }
(false, true) => { (false, true) => {
let bounds = to_str!(begin).to_string()..; let bounds = to_str!(begin).to_string()..;
if let Some(heads) = heads.as_ref() { if let Some(heads) = heads {
to_result(doc.map_range_at(obj_id, bounds, heads.as_ref())) to_result(doc.map_range_at(obj_id, bounds, &heads))
} else { } else {
to_result(doc.map_range(obj_id, bounds)) to_result(doc.map_range(obj_id, bounds))
} }
} }
(true, false) => { (true, false) => {
let bounds = ..to_str!(end).to_string(); let bounds = ..to_str!(end).to_string();
if let Some(heads) = heads.as_ref() { if let Some(heads) = heads {
to_result(doc.map_range_at(obj_id, bounds, heads.as_ref())) to_result(doc.map_range_at(obj_id, bounds, &heads))
} else { } else {
to_result(doc.map_range(obj_id, bounds)) to_result(doc.map_range(obj_id, bounds))
} }
} }
(true, true) => { (true, true) => {
let bounds = ..; let bounds = ..;
if let Some(heads) = heads.as_ref() { if let Some(heads) = heads {
to_result(doc.map_range_at(obj_id, bounds, heads.as_ref())) to_result(doc.map_range_at(obj_id, bounds, &heads))
} else { } else {
to_result(doc.map_range(obj_id, bounds)) to_result(doc.map_range(obj_id, bounds))
} }

View file

@ -1,98 +0,0 @@
use automerge as am;
use crate::byte_span::AMbyteSpan;
use crate::obj::AMobjId;
use crate::result::AMvalue;
/// \struct AMmapItem
/// \installed_headerfile
/// \brief An item in a map object.
pub struct AMmapItem {
/// The key of an item in a map object.
key: String,
/// The object identifier of an item in a map object.
obj_id: AMobjId,
/// The value of an item in a map object.
value: am::Value<'static>,
}
impl AMmapItem {
pub fn new(key: &'static str, value: am::Value<'static>, obj_id: am::ObjId) -> Self {
Self {
key: key.to_string(),
obj_id: AMobjId::new(obj_id),
value,
}
}
}
impl PartialEq for AMmapItem {
fn eq(&self, other: &Self) -> bool {
self.key == other.key && self.obj_id == other.obj_id && self.value == other.value
}
}
/*
impl From<&AMmapItem> for (String, am::Value<'static>, am::ObjId) {
fn from(map_item: &AMmapItem) -> Self {
(map_item.key.into_string().unwrap(), map_item.value.0.clone(), map_item.obj_id.as_ref().clone())
}
}
*/
/// \memberof AMmapItem
/// \brief Gets the key of an item in a map object.
///
/// \param[in] map_item A pointer to an `AMmapItem` struct.
/// \return An `AMbyteSpan` view of a UTF-8 string.
/// \pre \p map_item `!= NULL`.
/// \internal
///
/// # Safety
/// map_item must be a valid pointer to an AMmapItem
#[no_mangle]
pub unsafe extern "C" fn AMmapItemKey(map_item: *const AMmapItem) -> AMbyteSpan {
if let Some(map_item) = map_item.as_ref() {
map_item.key.as_bytes().into()
} else {
Default::default()
}
}
/// \memberof AMmapItem
/// \brief Gets the object identifier of an item in a map object.
///
/// \param[in] map_item A pointer to an `AMmapItem` struct.
/// \return A pointer to an `AMobjId` struct.
/// \pre \p map_item `!= NULL`.
/// \internal
///
/// # Safety
/// map_item must be a valid pointer to an AMmapItem
#[no_mangle]
pub unsafe extern "C" fn AMmapItemObjId(map_item: *const AMmapItem) -> *const AMobjId {
if let Some(map_item) = map_item.as_ref() {
&map_item.obj_id
} else {
std::ptr::null()
}
}
/// \memberof AMmapItem
/// \brief Gets the value of an item in a map object.
///
/// \param[in] map_item A pointer to an `AMmapItem` struct.
/// \return An `AMvalue` struct.
/// \pre \p map_item `!= NULL`.
/// \internal
///
/// # Safety
/// map_item must be a valid pointer to an AMmapItem
#[no_mangle]
pub unsafe extern "C" fn AMmapItemValue<'a>(map_item: *const AMmapItem) -> AMvalue<'a> {
if let Some(map_item) = map_item.as_ref() {
(&map_item.value).into()
} else {
AMvalue::Void
}
}

View file

@ -1,340 +0,0 @@
use std::ffi::c_void;
use std::mem::size_of;
use crate::doc::map::item::AMmapItem;
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(map_items: &[AMmapItem], offset: isize) -> Self {
Self {
len: map_items.len(),
offset,
ptr: map_items.as_ptr() as *const c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<&AMmapItem> {
if self.is_stopped() {
return None;
}
let slice: &[AMmapItem] =
unsafe { std::slice::from_raw_parts(self.ptr as *const AMmapItem, self.len) };
let value = &slice[self.get_index()];
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<&AMmapItem> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &[AMmapItem] =
unsafe { std::slice::from_raw_parts(self.ptr as *const AMmapItem, self.len) };
Some(&slice[self.get_index()])
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts((&detail as *const Detail) as *const u8, USIZE_USIZE_USIZE_)
.try_into()
.unwrap()
}
}
}
/// \struct AMmapItems
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of map object items.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMmapItems {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_],
}
impl AMmapItems {
pub fn new(map_items: &[AMmapItem]) -> Self {
Self {
detail: Detail::new(map_items, 0).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<&AMmapItem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<&AMmapItem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[AMmapItem]> for AMmapItems {
fn as_ref(&self) -> &[AMmapItem] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const AMmapItem, detail.len) }
}
}
impl Default for AMmapItems {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMmapItems
/// \brief Advances an iterator over a sequence of map object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] map_items A pointer to an `AMmapItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p map_items `!= NULL`.
/// \internal
///
/// #Safety
/// map_items must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsAdvance(map_items: *mut AMmapItems, n: isize) {
if let Some(map_items) = map_items.as_mut() {
map_items.advance(n);
};
}
/// \memberof AMmapItems
/// \brief Tests the equality of two sequences of map object items underlying
/// a pair of iterators.
///
/// \param[in] map_items1 A pointer to an `AMmapItems` struct.
/// \param[in] map_items2 A pointer to an `AMmapItems` struct.
/// \return `true` if \p map_items1 `==` \p map_items2 and `false` otherwise.
/// \pre \p map_items1 `!= NULL`.
/// \pre \p map_items2 `!= NULL`.
/// \internal
///
/// #Safety
/// map_items1 must be a valid pointer to an AMmapItems
/// map_items2 must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsEqual(
map_items1: *const AMmapItems,
map_items2: *const AMmapItems,
) -> bool {
match (map_items1.as_ref(), map_items2.as_ref()) {
(Some(map_items1), Some(map_items2)) => map_items1.as_ref() == map_items2.as_ref(),
(None, Some(_)) | (Some(_), None) | (None, None) => false,
}
}
/// \memberof AMmapItems
/// \brief Gets the map object item at the current position of an iterator
/// over a sequence of map object items and then advances it by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] map_items A pointer to an `AMmapItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMmapItem` struct that's `NULL` when \p map_items
/// was previously advanced past its forward/reverse limit.
/// \pre \p map_items `!= NULL`.
/// \internal
///
/// #Safety
/// map_items must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsNext(map_items: *mut AMmapItems, n: isize) -> *const AMmapItem {
if let Some(map_items) = map_items.as_mut() {
if let Some(map_item) = map_items.next(n) {
return map_item;
}
}
std::ptr::null()
}
/// \memberof AMmapItems
/// \brief Advances an iterator over a sequence of map object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the map object item at its new
/// position.
///
/// \param[in,out] map_items A pointer to an `AMmapItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMmapItem` struct that's `NULL` when \p map_items
/// is presently advanced past its forward/reverse limit.
/// \pre \p map_items `!= NULL`.
/// \internal
///
/// #Safety
/// map_items must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsPrev(map_items: *mut AMmapItems, n: isize) -> *const AMmapItem {
if let Some(map_items) = map_items.as_mut() {
if let Some(map_item) = map_items.prev(n) {
return map_item;
}
}
std::ptr::null()
}
/// \memberof AMmapItems
/// \brief Gets the size of the sequence of map object items underlying an
/// iterator.
///
/// \param[in] map_items A pointer to an `AMmapItems` struct.
/// \return The count of values in \p map_items.
/// \pre \p map_items `!= NULL`.
/// \internal
///
/// #Safety
/// map_items must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsSize(map_items: *const AMmapItems) -> usize {
if let Some(map_items) = map_items.as_ref() {
map_items.len()
} else {
0
}
}
/// \memberof AMmapItems
/// \brief Creates an iterator over the same sequence of map object items as
/// the given one but with the opposite position and direction.
///
/// \param[in] map_items A pointer to an `AMmapItems` struct.
/// \return An `AMmapItems` struct
/// \pre \p map_items `!= NULL`.
/// \internal
///
/// #Safety
/// map_items must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsReversed(map_items: *const AMmapItems) -> AMmapItems {
if let Some(map_items) = map_items.as_ref() {
map_items.reversed()
} else {
Default::default()
}
}
/// \memberof AMmapItems
/// \brief Creates an iterator at the starting position over the same sequence of map object items as the given one.
///
/// \param[in] map_items A pointer to an `AMmapItems` struct.
/// \return An `AMmapItems` struct
/// \pre \p map_items `!= NULL`.
/// \internal
///
/// #Safety
/// map_items must be a valid pointer to an AMmapItems
#[no_mangle]
pub unsafe extern "C" fn AMmapItemsRewound(map_items: *const AMmapItems) -> AMmapItems {
if let Some(map_items) = map_items.as_ref() {
map_items.rewound()
} else {
Default::default()
}
}

View file

@ -1,9 +1,20 @@
macro_rules! clamp {
($index:expr, $len:expr, $param_name:expr) => {{
if $index > $len && $index != usize::MAX {
return AMresult::error(&format!("Invalid {} {}", $param_name, $index)).into();
}
std::cmp::min($index, $len)
}};
}
pub(crate) use clamp;
macro_rules! to_doc { macro_rules! to_doc {
($handle:expr) => {{ ($handle:expr) => {{
let handle = $handle.as_ref(); let handle = $handle.as_ref();
match handle { match handle {
Some(b) => b, Some(b) => b,
None => return AMresult::err("Invalid AMdoc pointer").into(), None => return AMresult::error("Invalid `AMdoc*`").into(),
} }
}}; }};
} }
@ -15,9 +26,21 @@ macro_rules! to_doc_mut {
let handle = $handle.as_mut(); let handle = $handle.as_mut();
match handle { match handle {
Some(b) => b, Some(b) => b,
None => return AMresult::err("Invalid AMdoc pointer").into(), None => return AMresult::error("Invalid `AMdoc*`").into(),
} }
}}; }};
} }
pub(crate) use to_doc_mut; pub(crate) use to_doc_mut;
macro_rules! to_items {
($handle:expr) => {{
let handle = $handle.as_ref();
match handle {
Some(b) => b,
None => return AMresult::error("Invalid `AMitems*`").into(),
}
}};
}
pub(crate) use to_items;

View file

@ -0,0 +1,84 @@
use automerge as am;
use std::any::type_name;
use smol_str::SmolStr;
use crate::byte_span::AMbyteSpan;
/// \struct AMindex
/// \installed_headerfile
/// \brief An item index.
#[derive(PartialEq)]
pub enum AMindex {
/// A UTF-8 string key variant.
Key(SmolStr),
/// A 64-bit unsigned integer position variant.
Pos(usize),
}
impl TryFrom<&AMindex> for AMbyteSpan {
type Error = am::AutomergeError;
fn try_from(item: &AMindex) -> Result<Self, Self::Error> {
use am::AutomergeError::InvalidValueType;
use AMindex::*;
if let Key(key) = item {
return Ok(key.into());
}
Err(InvalidValueType {
expected: type_name::<SmolStr>().to_string(),
unexpected: type_name::<usize>().to_string(),
})
}
}
impl TryFrom<&AMindex> for usize {
type Error = am::AutomergeError;
fn try_from(item: &AMindex) -> Result<Self, Self::Error> {
use am::AutomergeError::InvalidValueType;
use AMindex::*;
if let Pos(pos) = item {
return Ok(*pos);
}
Err(InvalidValueType {
expected: type_name::<usize>().to_string(),
unexpected: type_name::<SmolStr>().to_string(),
})
}
}
/// \ingroup enumerations
/// \enum AMidxType
/// \installed_headerfile
/// \brief The type of an item's index.
#[derive(PartialEq, Eq)]
#[repr(u8)]
pub enum AMidxType {
/// The default tag, not a type signifier.
Default = 0,
/// A UTF-8 string view key.
Key,
/// A 64-bit unsigned integer position.
Pos,
}
impl Default for AMidxType {
fn default() -> Self {
Self::Default
}
}
impl From<&AMindex> for AMidxType {
fn from(index: &AMindex) -> Self {
use AMindex::*;
match index {
Key(_) => Self::Key,
Pos(_) => Self::Pos,
}
}
}

1963
rust/automerge-c/src/item.rs Normal file

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,401 @@
use automerge as am;
use std::ffi::c_void;
use std::marker::PhantomData;
use std::mem::size_of;
use crate::item::AMitem;
use crate::result::AMresult;
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(items: &[AMitem], offset: isize) -> Self {
Self {
len: items.len(),
offset,
ptr: items.as_ptr() as *mut c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<&mut AMitem> {
if self.is_stopped() {
return None;
}
let slice: &mut [AMitem] =
unsafe { std::slice::from_raw_parts_mut(self.ptr as *mut AMitem, self.len) };
let value = &mut slice[self.get_index()];
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<&mut AMitem> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &mut [AMitem] =
unsafe { std::slice::from_raw_parts_mut(self.ptr as *mut AMitem, self.len) };
Some(&mut slice[self.get_index()])
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts((&detail as *const Detail) as *const u8, USIZE_USIZE_USIZE_)
.try_into()
.unwrap()
}
}
}
/// \struct AMitems
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of `AMitem` structs.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMitems<'a> {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_],
phantom: PhantomData<&'a mut AMresult>,
}
impl<'a> AMitems<'a> {
pub fn new(items: &[AMitem]) -> Self {
Self {
detail: Detail::new(items, 0).into(),
phantom: PhantomData,
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<&mut AMitem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<&mut AMitem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
phantom: PhantomData,
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
phantom: PhantomData,
}
}
}
impl<'a> AsRef<[AMitem]> for AMitems<'a> {
fn as_ref(&self) -> &[AMitem] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const AMitem, detail.len) }
}
}
impl<'a> Default for AMitems<'a> {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_],
phantom: PhantomData,
}
}
}
impl TryFrom<&AMitems<'_>> for Vec<am::Change> {
type Error = am::AutomergeError;
fn try_from(items: &AMitems<'_>) -> Result<Self, Self::Error> {
let mut changes = Vec::<am::Change>::with_capacity(items.len());
for item in items.as_ref().iter() {
match <&am::Change>::try_from(item.as_ref()) {
Ok(change) => {
changes.push(change.clone());
}
Err(e) => {
return Err(e);
}
}
}
Ok(changes)
}
}
impl TryFrom<&AMitems<'_>> for Vec<am::ChangeHash> {
type Error = am::AutomergeError;
fn try_from(items: &AMitems<'_>) -> Result<Self, Self::Error> {
let mut change_hashes = Vec::<am::ChangeHash>::with_capacity(items.len());
for item in items.as_ref().iter() {
match <&am::ChangeHash>::try_from(item.as_ref()) {
Ok(change_hash) => {
change_hashes.push(*change_hash);
}
Err(e) => {
return Err(e);
}
}
}
Ok(change_hashes)
}
}
impl TryFrom<&AMitems<'_>> for Vec<am::ScalarValue> {
type Error = am::AutomergeError;
fn try_from(items: &AMitems<'_>) -> Result<Self, Self::Error> {
let mut scalars = Vec::<am::ScalarValue>::with_capacity(items.len());
for item in items.as_ref().iter() {
match <&am::ScalarValue>::try_from(item.as_ref()) {
Ok(scalar) => {
scalars.push(scalar.clone());
}
Err(e) => {
return Err(e);
}
}
}
Ok(scalars)
}
}
/// \memberof AMitems
/// \brief Advances an iterator over a sequence of object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in] items A pointer to an `AMitems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p items `!= NULL`
/// \internal
///
/// #Safety
/// items must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsAdvance(items: *mut AMitems, n: isize) {
if let Some(items) = items.as_mut() {
items.advance(n);
};
}
/// \memberof AMitems
/// \brief Tests the equality of two sequences of object items underlying a
/// pair of iterators.
///
/// \param[in] items1 A pointer to an `AMitems` struct.
/// \param[in] items2 A pointer to an `AMitems` struct.
/// \return `true` if \p items1 `==` \p items2 and `false` otherwise.
/// \pre \p items1 `!= NULL`
/// \pre \p items1 `!= NULL`
/// \post `!(`\p items1 `&&` \p items2 `) -> false`
/// \internal
///
/// #Safety
/// items1 must be a valid pointer to an AMitems
/// items2 must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsEqual(items1: *const AMitems, items2: *const AMitems) -> bool {
match (items1.as_ref(), items2.as_ref()) {
(Some(items1), Some(items2)) => items1.as_ref() == items2.as_ref(),
(None, None) | (None, Some(_)) | (Some(_), None) => false,
}
}
/// \memberof AMitems
/// \brief Gets the object item at the current position of an iterator over a
/// sequence of object items and then advances it by at most \p |n|
/// positions where the sign of \p n is relative to the iterator's
/// direction.
///
/// \param[in] items A pointer to an `AMitems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMitem` struct that's `NULL` when \p items
/// was previously advanced past its forward/reverse limit.
/// \pre \p items `!= NULL`
/// \internal
///
/// #Safety
/// items must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsNext(items: *mut AMitems, n: isize) -> *mut AMitem {
if let Some(items) = items.as_mut() {
if let Some(item) = items.next(n) {
return item;
}
}
std::ptr::null_mut()
}
/// \memberof AMitems
/// \brief Advances an iterator over a sequence of object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the object item at its new
/// position.
///
/// \param[in] items A pointer to an `AMitems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMitem` struct that's `NULL` when \p items
/// is presently advanced past its forward/reverse limit.
/// \pre \p items `!= NULL`
/// \internal
///
/// #Safety
/// items must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsPrev(items: *mut AMitems, n: isize) -> *mut AMitem {
if let Some(items) = items.as_mut() {
if let Some(obj_item) = items.prev(n) {
return obj_item;
}
}
std::ptr::null_mut()
}
/// \memberof AMitems
/// \brief Gets the size of the sequence underlying an iterator.
///
/// \param[in] items A pointer to an `AMitems` struct.
/// \return The count of items in \p items.
/// \pre \p items `!= NULL`
/// \internal
///
/// #Safety
/// items must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsSize(items: *const AMitems) -> usize {
if let Some(items) = items.as_ref() {
return items.len();
}
0
}
/// \memberof AMitems
/// \brief Creates an iterator over the same sequence of items as the
/// given one but with the opposite position and direction.
///
/// \param[in] items A pointer to an `AMitems` struct.
/// \return An `AMitems` struct
/// \pre \p items `!= NULL`
/// \internal
///
/// #Safety
/// items must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsReversed(items: *const AMitems) -> AMitems {
if let Some(items) = items.as_ref() {
return items.reversed();
}
Default::default()
}
/// \memberof AMitems
/// \brief Creates an iterator at the starting position over the same sequence
/// of items as the given one.
///
/// \param[in] items A pointer to an `AMitems` struct.
/// \return An `AMitems` struct
/// \pre \p items `!= NULL`
/// \internal
///
/// #Safety
/// items must be a valid pointer to an AMitems
#[no_mangle]
pub unsafe extern "C" fn AMitemsRewound(items: *const AMitems) -> AMitems {
if let Some(items) = items.as_ref() {
return items.rewound();
}
Default::default()
}

View file

@ -1,11 +1,12 @@
mod actor_id; mod actor_id;
mod byte_span; mod byte_span;
mod change; mod change;
mod change_hashes;
mod changes;
mod doc; mod doc;
mod index;
mod item;
mod items;
mod obj; mod obj;
mod result; mod result;
mod result_stack;
mod strs;
mod sync; mod sync;
// include!(concat!(env!("OUT_DIR"), "/enum_string_functions.rs"));

View file

@ -1,12 +1,10 @@
use automerge as am; use automerge as am;
use std::any::type_name;
use std::cell::RefCell; use std::cell::RefCell;
use std::ops::Deref; use std::ops::Deref;
use crate::actor_id::AMactorId; use crate::actor_id::AMactorId;
pub mod item;
pub mod items;
macro_rules! to_obj_id { macro_rules! to_obj_id {
($handle:expr) => {{ ($handle:expr) => {{
match $handle.as_ref() { match $handle.as_ref() {
@ -19,12 +17,11 @@ macro_rules! to_obj_id {
pub(crate) use to_obj_id; pub(crate) use to_obj_id;
macro_rules! to_obj_type { macro_rules! to_obj_type {
($am_obj_type:expr) => {{ ($c_obj_type:expr) => {{
match $am_obj_type { let result: Result<am::ObjType, am::AutomergeError> = (&$c_obj_type).try_into();
AMobjType::Map => am::ObjType::Map, match result {
AMobjType::List => am::ObjType::List, Ok(obj_type) => obj_type,
AMobjType::Text => am::ObjType::Text, Err(e) => return AMresult::error(&e.to_string()).into(),
AMobjType::Void => return AMresult::err("Invalid AMobjType value").into(),
} }
}}; }};
} }
@ -79,11 +76,11 @@ impl Deref for AMobjId {
} }
/// \memberof AMobjId /// \memberof AMobjId
/// \brief Gets the actor identifier of an object identifier. /// \brief Gets the actor identifier component of an object identifier.
/// ///
/// \param[in] obj_id A pointer to an `AMobjId` struct. /// \param[in] obj_id A pointer to an `AMobjId` struct.
/// \return A pointer to an `AMactorId` struct or `NULL`. /// \return A pointer to an `AMactorId` struct or `NULL`.
/// \pre \p obj_id `!= NULL`. /// \pre \p obj_id `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -97,11 +94,11 @@ pub unsafe extern "C" fn AMobjIdActorId(obj_id: *const AMobjId) -> *const AMacto
} }
/// \memberof AMobjId /// \memberof AMobjId
/// \brief Gets the counter of an object identifier. /// \brief Gets the counter component of an object identifier.
/// ///
/// \param[in] obj_id A pointer to an `AMobjId` struct. /// \param[in] obj_id A pointer to an `AMobjId` struct.
/// \return A 64-bit unsigned integer. /// \return A 64-bit unsigned integer.
/// \pre \p obj_id `!= NULL`. /// \pre \p obj_id `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -124,8 +121,9 @@ pub unsafe extern "C" fn AMobjIdCounter(obj_id: *const AMobjId) -> u64 {
/// \param[in] obj_id1 A pointer to an `AMobjId` struct. /// \param[in] obj_id1 A pointer to an `AMobjId` struct.
/// \param[in] obj_id2 A pointer to an `AMobjId` struct. /// \param[in] obj_id2 A pointer to an `AMobjId` struct.
/// \return `true` if \p obj_id1 `==` \p obj_id2 and `false` otherwise. /// \return `true` if \p obj_id1 `==` \p obj_id2 and `false` otherwise.
/// \pre \p obj_id1 `!= NULL`. /// \pre \p obj_id1 `!= NULL`
/// \pre \p obj_id2 `!= NULL`. /// \pre \p obj_id1 `!= NULL`
/// \post `!(`\p obj_id1 `&&` \p obj_id2 `) -> false`
/// \internal /// \internal
/// ///
/// #Safety /// #Safety
@ -135,26 +133,28 @@ pub unsafe extern "C" fn AMobjIdCounter(obj_id: *const AMobjId) -> u64 {
pub unsafe extern "C" fn AMobjIdEqual(obj_id1: *const AMobjId, obj_id2: *const AMobjId) -> bool { pub unsafe extern "C" fn AMobjIdEqual(obj_id1: *const AMobjId, obj_id2: *const AMobjId) -> bool {
match (obj_id1.as_ref(), obj_id2.as_ref()) { match (obj_id1.as_ref(), obj_id2.as_ref()) {
(Some(obj_id1), Some(obj_id2)) => obj_id1 == obj_id2, (Some(obj_id1), Some(obj_id2)) => obj_id1 == obj_id2,
(None, Some(_)) | (Some(_), None) | (None, None) => false, (None, None) | (None, Some(_)) | (Some(_), None) => false,
} }
} }
/// \memberof AMobjId /// \memberof AMobjId
/// \brief Gets the index of an object identifier. /// \brief Gets the index component of an object identifier.
/// ///
/// \param[in] obj_id A pointer to an `AMobjId` struct. /// \param[in] obj_id A pointer to an `AMobjId` struct.
/// \return A 64-bit unsigned integer. /// \return A 64-bit unsigned integer.
/// \pre \p obj_id `!= NULL`. /// \pre \p obj_id `!= NULL`
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// obj_id must be a valid pointer to an AMobjId /// obj_id must be a valid pointer to an AMobjId
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMobjIdIndex(obj_id: *const AMobjId) -> usize { pub unsafe extern "C" fn AMobjIdIndex(obj_id: *const AMobjId) -> usize {
use am::ObjId::*;
if let Some(obj_id) = obj_id.as_ref() { if let Some(obj_id) = obj_id.as_ref() {
match obj_id.as_ref() { match obj_id.as_ref() {
am::ObjId::Id(_, _, index) => *index, Id(_, _, index) => *index,
am::ObjId::Root => 0, Root => 0,
} }
} else { } else {
usize::MAX usize::MAX
@ -163,26 +163,54 @@ pub unsafe extern "C" fn AMobjIdIndex(obj_id: *const AMobjId) -> usize {
/// \ingroup enumerations /// \ingroup enumerations
/// \enum AMobjType /// \enum AMobjType
/// \installed_headerfile
/// \brief The type of an object value. /// \brief The type of an object value.
#[derive(PartialEq, Eq)]
#[repr(u8)] #[repr(u8)]
pub enum AMobjType { pub enum AMobjType {
/// A void. /// The default tag, not a type signifier.
/// \note This tag is unalphabetized to evaluate as false. Default = 0,
Void = 0,
/// A list. /// A list.
List, List = 1,
/// A key-value map. /// A key-value map.
Map, Map,
/// A list of Unicode graphemes. /// A list of Unicode graphemes.
Text, Text,
} }
impl From<am::ObjType> for AMobjType { impl Default for AMobjType {
fn from(o: am::ObjType) -> Self { fn default() -> Self {
Self::Default
}
}
impl From<&am::ObjType> for AMobjType {
fn from(o: &am::ObjType) -> Self {
use am::ObjType::*;
match o { match o {
am::ObjType::Map | am::ObjType::Table => AMobjType::Map, List => Self::List,
am::ObjType::List => AMobjType::List, Map | Table => Self::Map,
am::ObjType::Text => AMobjType::Text, Text => Self::Text,
}
}
}
impl TryFrom<&AMobjType> for am::ObjType {
type Error = am::AutomergeError;
fn try_from(c_obj_type: &AMobjType) -> Result<Self, Self::Error> {
use am::AutomergeError::InvalidValueType;
use AMobjType::*;
match c_obj_type {
List => Ok(Self::List),
Map => Ok(Self::Map),
Text => Ok(Self::Text),
_ => Err(InvalidValueType {
expected: type_name::<Self>().to_string(),
unexpected: type_name::<AMobjType>().to_string(),
}),
} }
} }
} }

View file

@ -1,73 +0,0 @@
use automerge as am;
use crate::obj::AMobjId;
use crate::result::AMvalue;
/// \struct AMobjItem
/// \installed_headerfile
/// \brief An item in an object.
pub struct AMobjItem {
/// The object identifier of an item in an object.
obj_id: AMobjId,
/// The value of an item in an object.
value: am::Value<'static>,
}
impl AMobjItem {
pub fn new(value: am::Value<'static>, obj_id: am::ObjId) -> Self {
Self {
obj_id: AMobjId::new(obj_id),
value,
}
}
}
impl PartialEq for AMobjItem {
fn eq(&self, other: &Self) -> bool {
self.obj_id == other.obj_id && self.value == other.value
}
}
impl From<&AMobjItem> for (am::Value<'static>, am::ObjId) {
fn from(obj_item: &AMobjItem) -> Self {
(obj_item.value.clone(), obj_item.obj_id.as_ref().clone())
}
}
/// \memberof AMobjItem
/// \brief Gets the object identifier of an item in an object.
///
/// \param[in] obj_item A pointer to an `AMobjItem` struct.
/// \return A pointer to an `AMobjId` struct.
/// \pre \p obj_item `!= NULL`.
/// \internal
///
/// # Safety
/// obj_item must be a valid pointer to an AMobjItem
#[no_mangle]
pub unsafe extern "C" fn AMobjItemObjId(obj_item: *const AMobjItem) -> *const AMobjId {
if let Some(obj_item) = obj_item.as_ref() {
&obj_item.obj_id
} else {
std::ptr::null()
}
}
/// \memberof AMobjItem
/// \brief Gets the value of an item in an object.
///
/// \param[in] obj_item A pointer to an `AMobjItem` struct.
/// \return An `AMvalue` struct.
/// \pre \p obj_item `!= NULL`.
/// \internal
///
/// # Safety
/// obj_item must be a valid pointer to an AMobjItem
#[no_mangle]
pub unsafe extern "C" fn AMobjItemValue<'a>(obj_item: *const AMobjItem) -> AMvalue<'a> {
if let Some(obj_item) = obj_item.as_ref() {
(&obj_item.value).into()
} else {
AMvalue::Void
}
}

View file

@ -1,341 +0,0 @@
use std::ffi::c_void;
use std::mem::size_of;
use crate::obj::item::AMobjItem;
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(obj_items: &[AMobjItem], offset: isize) -> Self {
Self {
len: obj_items.len(),
offset,
ptr: obj_items.as_ptr() as *const c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<&AMobjItem> {
if self.is_stopped() {
return None;
}
let slice: &[AMobjItem] =
unsafe { std::slice::from_raw_parts(self.ptr as *const AMobjItem, self.len) };
let value = &slice[self.get_index()];
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<&AMobjItem> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &[AMobjItem] =
unsafe { std::slice::from_raw_parts(self.ptr as *const AMobjItem, self.len) };
Some(&slice[self.get_index()])
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts((&detail as *const Detail) as *const u8, USIZE_USIZE_USIZE_)
.try_into()
.unwrap()
}
}
}
/// \struct AMobjItems
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of object items.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMobjItems {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_],
}
impl AMobjItems {
pub fn new(obj_items: &[AMobjItem]) -> Self {
Self {
detail: Detail::new(obj_items, 0).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<&AMobjItem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<&AMobjItem> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[AMobjItem]> for AMobjItems {
fn as_ref(&self) -> &[AMobjItem] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const AMobjItem, detail.len) }
}
}
impl Default for AMobjItems {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMobjItems
/// \brief Advances an iterator over a sequence of object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] obj_items A pointer to an `AMobjItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p obj_items `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsAdvance(obj_items: *mut AMobjItems, n: isize) {
if let Some(obj_items) = obj_items.as_mut() {
obj_items.advance(n);
};
}
/// \memberof AMobjItems
/// \brief Tests the equality of two sequences of object items underlying a
/// pair of iterators.
///
/// \param[in] obj_items1 A pointer to an `AMobjItems` struct.
/// \param[in] obj_items2 A pointer to an `AMobjItems` struct.
/// \return `true` if \p obj_items1 `==` \p obj_items2 and `false` otherwise.
/// \pre \p obj_items1 `!= NULL`.
/// \pre \p obj_items2 `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items1 must be a valid pointer to an AMobjItems
/// obj_items2 must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsEqual(
obj_items1: *const AMobjItems,
obj_items2: *const AMobjItems,
) -> bool {
match (obj_items1.as_ref(), obj_items2.as_ref()) {
(Some(obj_items1), Some(obj_items2)) => obj_items1.as_ref() == obj_items2.as_ref(),
(None, Some(_)) | (Some(_), None) | (None, None) => false,
}
}
/// \memberof AMobjItems
/// \brief Gets the object item at the current position of an iterator over a
/// sequence of object items and then advances it by at most \p |n|
/// positions where the sign of \p n is relative to the iterator's
/// direction.
///
/// \param[in,out] obj_items A pointer to an `AMobjItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMobjItem` struct that's `NULL` when \p obj_items
/// was previously advanced past its forward/reverse limit.
/// \pre \p obj_items `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsNext(obj_items: *mut AMobjItems, n: isize) -> *const AMobjItem {
if let Some(obj_items) = obj_items.as_mut() {
if let Some(obj_item) = obj_items.next(n) {
return obj_item;
}
}
std::ptr::null()
}
/// \memberof AMobjItems
/// \brief Advances an iterator over a sequence of object items by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the object item at its new
/// position.
///
/// \param[in,out] obj_items A pointer to an `AMobjItems` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMobjItem` struct that's `NULL` when \p obj_items
/// is presently advanced past its forward/reverse limit.
/// \pre \p obj_items `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsPrev(obj_items: *mut AMobjItems, n: isize) -> *const AMobjItem {
if let Some(obj_items) = obj_items.as_mut() {
if let Some(obj_item) = obj_items.prev(n) {
return obj_item;
}
}
std::ptr::null()
}
/// \memberof AMobjItems
/// \brief Gets the size of the sequence of object items underlying an
/// iterator.
///
/// \param[in] obj_items A pointer to an `AMobjItems` struct.
/// \return The count of values in \p obj_items.
/// \pre \p obj_items `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsSize(obj_items: *const AMobjItems) -> usize {
if let Some(obj_items) = obj_items.as_ref() {
obj_items.len()
} else {
0
}
}
/// \memberof AMobjItems
/// \brief Creates an iterator over the same sequence of object items as the
/// given one but with the opposite position and direction.
///
/// \param[in] obj_items A pointer to an `AMobjItems` struct.
/// \return An `AMobjItems` struct
/// \pre \p obj_items `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsReversed(obj_items: *const AMobjItems) -> AMobjItems {
if let Some(obj_items) = obj_items.as_ref() {
obj_items.reversed()
} else {
Default::default()
}
}
/// \memberof AMobjItems
/// \brief Creates an iterator at the starting position over the same sequence
/// of object items as the given one.
///
/// \param[in] obj_items A pointer to an `AMobjItems` struct.
/// \return An `AMobjItems` struct
/// \pre \p obj_items `!= NULL`.
/// \internal
///
/// #Safety
/// obj_items must be a valid pointer to an AMobjItems
#[no_mangle]
pub unsafe extern "C" fn AMobjItemsRewound(obj_items: *const AMobjItems) -> AMobjItems {
if let Some(obj_items) = obj_items.as_ref() {
obj_items.rewound()
} else {
Default::default()
}
}

File diff suppressed because it is too large Load diff

View file

@ -1,156 +0,0 @@
use crate::result::{AMfree, AMresult, AMresultStatus, AMresultValue, AMstatus, AMvalue};
/// \struct AMresultStack
/// \installed_headerfile
/// \brief A node in a singly-linked list of result pointers.
///
/// \note Using this data structure is purely optional because its only purpose
/// is to make memory management tolerable for direct usage of this API
/// in C, C++ and Objective-C.
#[repr(C)]
pub struct AMresultStack {
/// A result to be deallocated.
pub result: *mut AMresult,
/// The next node in the singly-linked list or `NULL`.
pub next: *mut AMresultStack,
}
impl AMresultStack {
pub fn new(result: *mut AMresult, next: *mut AMresultStack) -> Self {
Self { result, next }
}
}
/// \memberof AMresultStack
/// \brief Deallocates the storage for a stack of results.
///
/// \param[in,out] stack A pointer to a pointer to an `AMresultStack` struct.
/// \return The number of `AMresult` structs freed.
/// \pre \p stack `!= NULL`.
/// \post `*stack == NULL`.
/// \note Calling this function is purely optional because its only purpose is
/// to make memory management tolerable for direct usage of this API in
/// C, C++ and Objective-C.
/// \internal
///
/// # Safety
/// stack must be a valid AMresultStack pointer pointer
#[no_mangle]
pub unsafe extern "C" fn AMfreeStack(stack: *mut *mut AMresultStack) -> usize {
if stack.is_null() {
return 0;
}
let mut count: usize = 0;
while !(*stack).is_null() {
AMfree(AMpop(stack));
count += 1;
}
count
}
/// \memberof AMresultStack
/// \brief Gets the topmost result from the stack after removing it.
///
/// \param[in,out] stack A pointer to a pointer to an `AMresultStack` struct.
/// \return A pointer to an `AMresult` struct or `NULL`.
/// \pre \p stack `!= NULL`.
/// \post `*stack == NULL`.
/// \note Calling this function is purely optional because its only purpose is
/// to make memory management tolerable for direct usage of this API in
/// C, C++ and Objective-C.
/// \internal
///
/// # Safety
/// stack must be a valid AMresultStack pointer pointer
#[no_mangle]
pub unsafe extern "C" fn AMpop(stack: *mut *mut AMresultStack) -> *mut AMresult {
if stack.is_null() || (*stack).is_null() {
return std::ptr::null_mut();
}
let top = Box::from_raw(*stack);
*stack = top.next;
let result = top.result;
drop(top);
result
}
/// \memberof AMresultStack
/// \brief The prototype of a function to be called when a value matching the
/// given discriminant cannot be extracted from the result at the top of
/// the given stack.
///
/// \note Implementing this function is purely optional because its only purpose
/// is to make memory management tolerable for direct usage of this API
/// in C, C++ and Objective-C.
pub type AMpushCallback =
Option<extern "C" fn(stack: *mut *mut AMresultStack, discriminant: u8) -> ()>;
/// \memberof AMresultStack
/// \brief Pushes the given result onto the given stack and then either extracts
/// a value matching the given discriminant from that result or,
/// failing that, calls the given function and gets a void value instead.
///
/// \param[in,out] stack A pointer to a pointer to an `AMresultStack` struct.
/// \param[in] result A pointer to an `AMresult` struct.
/// \param[in] discriminant An `AMvalue` variant's corresponding enum tag.
/// \param[in] callback A pointer to a function with the same signature as
/// `AMpushCallback()` or `NULL`.
/// \return An `AMvalue` struct.
/// \pre \p stack `!= NULL`.
/// \pre \p result `!= NULL`.
/// \warning If \p stack `== NULL` then \p result is deallocated in order to
/// prevent a memory leak.
/// \note Calling this function is purely optional because its only purpose is
/// to make memory management tolerable for direct usage of this API in
/// C, C++ and Objective-C.
/// \internal
///
/// # Safety
/// stack must be a valid AMresultStack pointer pointer
/// result must be a valid AMresult pointer
#[no_mangle]
pub unsafe extern "C" fn AMpush<'a>(
stack: *mut *mut AMresultStack,
result: *mut AMresult,
discriminant: u8,
callback: AMpushCallback,
) -> AMvalue<'a> {
if stack.is_null() {
// There's no stack to push the result onto so it has to be freed in
// order to prevent a memory leak.
AMfree(result);
if let Some(callback) = callback {
callback(stack, discriminant);
}
return AMvalue::Void;
} else if result.is_null() {
if let Some(callback) = callback {
callback(stack, discriminant);
}
return AMvalue::Void;
}
// Always push the result onto the stack, even if it's wrong, so that the
// given callback can retrieve it.
let node = Box::new(AMresultStack::new(result, *stack));
let top = Box::into_raw(node);
*stack = top;
// Test that the result contains a value.
match AMresultStatus(result) {
AMstatus::Ok => {}
_ => {
if let Some(callback) = callback {
callback(stack, discriminant);
}
return AMvalue::Void;
}
}
// Test that the result's value matches the given discriminant.
let value = AMresultValue(result);
if discriminant != u8::from(&value) {
if let Some(callback) = callback {
callback(stack, discriminant);
}
return AMvalue::Void;
}
value
}

View file

@ -1,359 +0,0 @@
use std::cmp::Ordering;
use std::ffi::c_void;
use std::mem::size_of;
use std::os::raw::c_char;
use crate::byte_span::AMbyteSpan;
/// \brief Creates a string view from a C string.
///
/// \param[in] c_str A UTF-8 C string.
/// \return A UTF-8 string view as an `AMbyteSpan` struct.
/// \internal
///
/// #Safety
/// c_str must be a null-terminated array of `c_char`
#[no_mangle]
pub unsafe extern "C" fn AMstr(c_str: *const c_char) -> AMbyteSpan {
c_str.into()
}
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(strings: &[String], offset: isize) -> Self {
Self {
len: strings.len(),
offset,
ptr: strings.as_ptr() as *const c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<AMbyteSpan> {
if self.is_stopped() {
return None;
}
let slice: &[String] =
unsafe { std::slice::from_raw_parts(self.ptr as *const String, self.len) };
let value = slice[self.get_index()].as_bytes().into();
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<AMbyteSpan> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &[String] =
unsafe { std::slice::from_raw_parts(self.ptr as *const String, self.len) };
Some(slice[self.get_index()].as_bytes().into())
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts((&detail as *const Detail) as *const u8, USIZE_USIZE_USIZE_)
.try_into()
.unwrap()
}
}
}
/// \struct AMstrs
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of UTF-8 strings.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMstrs {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_],
}
impl AMstrs {
pub fn new(strings: &[String]) -> Self {
Self {
detail: Detail::new(strings, 0).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<AMbyteSpan> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<AMbyteSpan> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[String]> for AMstrs {
fn as_ref(&self) -> &[String] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const String, detail.len) }
}
}
impl Default for AMstrs {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMstrs
/// \brief Advances an iterator over a sequence of UTF-8 strings by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] strs A pointer to an `AMstrs` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p strs `!= NULL`.
/// \internal
///
/// #Safety
/// strs must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsAdvance(strs: *mut AMstrs, n: isize) {
if let Some(strs) = strs.as_mut() {
strs.advance(n);
};
}
/// \memberof AMstrs
/// \brief Compares the sequences of UTF-8 strings underlying a pair of
/// iterators.
///
/// \param[in] strs1 A pointer to an `AMstrs` struct.
/// \param[in] strs2 A pointer to an `AMstrs` struct.
/// \return `-1` if \p strs1 `<` \p strs2, `0` if
/// \p strs1 `==` \p strs2 and `1` if
/// \p strs1 `>` \p strs2.
/// \pre \p strs1 `!= NULL`.
/// \pre \p strs2 `!= NULL`.
/// \internal
///
/// #Safety
/// strs1 must be a valid pointer to an AMstrs
/// strs2 must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsCmp(strs1: *const AMstrs, strs2: *const AMstrs) -> isize {
match (strs1.as_ref(), strs2.as_ref()) {
(Some(strs1), Some(strs2)) => match strs1.as_ref().cmp(strs2.as_ref()) {
Ordering::Less => -1,
Ordering::Equal => 0,
Ordering::Greater => 1,
},
(None, Some(_)) => -1,
(Some(_), None) => 1,
(None, None) => 0,
}
}
/// \memberof AMstrs
/// \brief Gets the key at the current position of an iterator over a sequence
/// of UTF-8 strings and then advances it by at most \p |n| positions
/// where the sign of \p n is relative to the iterator's direction.
///
/// \param[in,out] strs A pointer to an `AMstrs` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A UTF-8 string view as an `AMbyteSpan` struct that's `AMstr(NULL)`
/// when \p strs was previously advanced past its forward/reverse limit.
/// \pre \p strs `!= NULL`.
/// \internal
///
/// #Safety
/// strs must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsNext(strs: *mut AMstrs, n: isize) -> AMbyteSpan {
if let Some(strs) = strs.as_mut() {
if let Some(key) = strs.next(n) {
return key;
}
}
Default::default()
}
/// \memberof AMstrs
/// \brief Advances an iterator over a sequence of UTF-8 strings by at most
/// \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the key at its new position.
///
/// \param[in,out] strs A pointer to an `AMstrs` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A UTF-8 string view as an `AMbyteSpan` struct that's `AMstr(NULL)`
/// when \p strs is presently advanced past its forward/reverse limit.
/// \pre \p strs `!= NULL`.
/// \internal
///
/// #Safety
/// strs must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsPrev(strs: *mut AMstrs, n: isize) -> AMbyteSpan {
if let Some(strs) = strs.as_mut() {
if let Some(key) = strs.prev(n) {
return key;
}
}
Default::default()
}
/// \memberof AMstrs
/// \brief Gets the size of the sequence of UTF-8 strings underlying an
/// iterator.
///
/// \param[in] strs A pointer to an `AMstrs` struct.
/// \return The count of values in \p strs.
/// \pre \p strs `!= NULL`.
/// \internal
///
/// #Safety
/// strs must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsSize(strs: *const AMstrs) -> usize {
if let Some(strs) = strs.as_ref() {
strs.len()
} else {
0
}
}
/// \memberof AMstrs
/// \brief Creates an iterator over the same sequence of UTF-8 strings as the
/// given one but with the opposite position and direction.
///
/// \param[in] strs A pointer to an `AMstrs` struct.
/// \return An `AMstrs` struct.
/// \pre \p strs `!= NULL`.
/// \internal
///
/// #Safety
/// strs must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsReversed(strs: *const AMstrs) -> AMstrs {
if let Some(strs) = strs.as_ref() {
strs.reversed()
} else {
AMstrs::default()
}
}
/// \memberof AMstrs
/// \brief Creates an iterator at the starting position over the same sequence
/// of UTF-8 strings as the given one.
///
/// \param[in] strs A pointer to an `AMstrs` struct.
/// \return An `AMstrs` struct
/// \pre \p strs `!= NULL`.
/// \internal
///
/// #Safety
/// strs must be a valid pointer to an AMstrs
#[no_mangle]
pub unsafe extern "C" fn AMstrsRewound(strs: *const AMstrs) -> AMstrs {
if let Some(strs) = strs.as_ref() {
strs.rewound()
} else {
Default::default()
}
}

View file

@ -1,7 +1,7 @@
mod have; mod have;
mod haves;
mod message; mod message;
mod state; mod state;
pub(crate) use have::AMsyncHave;
pub(crate) use message::{to_sync_message, AMsyncMessage}; pub(crate) use message::{to_sync_message, AMsyncMessage};
pub(crate) use state::AMsyncState; pub(crate) use state::AMsyncState;

View file

@ -1,23 +1,23 @@
use automerge as am; use automerge as am;
use crate::change_hashes::AMchangeHashes; use crate::result::{to_result, AMresult};
/// \struct AMsyncHave /// \struct AMsyncHave
/// \installed_headerfile /// \installed_headerfile
/// \brief A summary of the changes that the sender of a synchronization /// \brief A summary of the changes that the sender of a synchronization
/// message already has. /// message already has.
#[derive(Clone, Eq, PartialEq)] #[derive(Clone, Eq, PartialEq)]
pub struct AMsyncHave(*const am::sync::Have); pub struct AMsyncHave(am::sync::Have);
impl AMsyncHave { impl AMsyncHave {
pub fn new(have: &am::sync::Have) -> Self { pub fn new(have: am::sync::Have) -> Self {
Self(have) Self(have)
} }
} }
impl AsRef<am::sync::Have> for AMsyncHave { impl AsRef<am::sync::Have> for AMsyncHave {
fn as_ref(&self) -> &am::sync::Have { fn as_ref(&self) -> &am::sync::Have {
unsafe { &*self.0 } &self.0
} }
} }
@ -25,17 +25,18 @@ impl AsRef<am::sync::Have> for AMsyncHave {
/// \brief Gets the heads of the sender. /// \brief Gets the heads of the sender.
/// ///
/// \param[in] sync_have A pointer to an `AMsyncHave` struct. /// \param[in] sync_have A pointer to an `AMsyncHave` struct.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_have `!= NULL`. /// \pre \p sync_have `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_have must be a valid pointer to an AMsyncHave /// sync_have must be a valid pointer to an AMsyncHave
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncHaveLastSync(sync_have: *const AMsyncHave) -> AMchangeHashes { pub unsafe extern "C" fn AMsyncHaveLastSync(sync_have: *const AMsyncHave) -> *mut AMresult {
if let Some(sync_have) = sync_have.as_ref() { to_result(match sync_have.as_ref() {
AMchangeHashes::new(&sync_have.as_ref().last_sync) Some(sync_have) => sync_have.as_ref().last_sync.as_slice(),
} else { None => Default::default(),
Default::default() })
}
} }

View file

@ -1,378 +0,0 @@
use automerge as am;
use std::collections::BTreeMap;
use std::ffi::c_void;
use std::mem::size_of;
use crate::sync::have::AMsyncHave;
#[repr(C)]
struct Detail {
len: usize,
offset: isize,
ptr: *const c_void,
storage: *mut c_void,
}
/// \note cbindgen won't propagate the value of a `std::mem::size_of<T>()` call
/// (https://github.com/eqrion/cbindgen/issues/252) but it will
/// propagate the name of a constant initialized from it so if the
/// constant's name is a symbolic representation of the value it can be
/// converted into a number by post-processing the header it generated.
pub const USIZE_USIZE_USIZE_USIZE_: usize = size_of::<Detail>();
impl Detail {
fn new(
haves: &[am::sync::Have],
offset: isize,
storage: &mut BTreeMap<usize, AMsyncHave>,
) -> Self {
let storage: *mut BTreeMap<usize, AMsyncHave> = storage;
Self {
len: haves.len(),
offset,
ptr: haves.as_ptr() as *const c_void,
storage: storage as *mut c_void,
}
}
pub fn advance(&mut self, n: isize) {
if n == 0 {
return;
}
let len = self.len as isize;
self.offset = if self.offset < 0 {
// It's reversed.
let unclipped = self.offset.checked_sub(n).unwrap_or(isize::MIN);
if unclipped >= 0 {
// Clip it to the forward stop.
len
} else {
std::cmp::min(std::cmp::max(-(len + 1), unclipped), -1)
}
} else {
let unclipped = self.offset.checked_add(n).unwrap_or(isize::MAX);
if unclipped < 0 {
// Clip it to the reverse stop.
-(len + 1)
} else {
std::cmp::max(0, std::cmp::min(unclipped, len))
}
}
}
pub fn get_index(&self) -> usize {
(self.offset
+ if self.offset < 0 {
self.len as isize
} else {
0
}) as usize
}
pub fn next(&mut self, n: isize) -> Option<*const AMsyncHave> {
if self.is_stopped() {
return None;
}
let slice: &[am::sync::Have] =
unsafe { std::slice::from_raw_parts(self.ptr as *const am::sync::Have, self.len) };
let storage = unsafe { &mut *(self.storage as *mut BTreeMap<usize, AMsyncHave>) };
let index = self.get_index();
let value = match storage.get_mut(&index) {
Some(value) => value,
None => {
storage.insert(index, AMsyncHave::new(&slice[index]));
storage.get_mut(&index).unwrap()
}
};
self.advance(n);
Some(value)
}
pub fn is_stopped(&self) -> bool {
let len = self.len as isize;
self.offset < -len || self.offset == len
}
pub fn prev(&mut self, n: isize) -> Option<*const AMsyncHave> {
self.advance(-n);
if self.is_stopped() {
return None;
}
let slice: &[am::sync::Have] =
unsafe { std::slice::from_raw_parts(self.ptr as *const am::sync::Have, self.len) };
let storage = unsafe { &mut *(self.storage as *mut BTreeMap<usize, AMsyncHave>) };
let index = self.get_index();
Some(match storage.get_mut(&index) {
Some(value) => value,
None => {
storage.insert(index, AMsyncHave::new(&slice[index]));
storage.get_mut(&index).unwrap()
}
})
}
pub fn reversed(&self) -> Self {
Self {
len: self.len,
offset: -(self.offset + 1),
ptr: self.ptr,
storage: self.storage,
}
}
pub fn rewound(&self) -> Self {
Self {
len: self.len,
offset: if self.offset < 0 { -1 } else { 0 },
ptr: self.ptr,
storage: self.storage,
}
}
}
impl From<Detail> for [u8; USIZE_USIZE_USIZE_USIZE_] {
fn from(detail: Detail) -> Self {
unsafe {
std::slice::from_raw_parts(
(&detail as *const Detail) as *const u8,
USIZE_USIZE_USIZE_USIZE_,
)
.try_into()
.unwrap()
}
}
}
/// \struct AMsyncHaves
/// \installed_headerfile
/// \brief A random-access iterator over a sequence of synchronization haves.
#[repr(C)]
#[derive(Eq, PartialEq)]
pub struct AMsyncHaves {
/// An implementation detail that is intentionally opaque.
/// \warning Modifying \p detail will cause undefined behavior.
/// \note The actual size of \p detail will vary by platform, this is just
/// the one for the platform this documentation was built on.
detail: [u8; USIZE_USIZE_USIZE_USIZE_],
}
impl AMsyncHaves {
pub fn new(haves: &[am::sync::Have], storage: &mut BTreeMap<usize, AMsyncHave>) -> Self {
Self {
detail: Detail::new(haves, 0, storage).into(),
}
}
pub fn advance(&mut self, n: isize) {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.advance(n);
}
pub fn len(&self) -> usize {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
detail.len
}
pub fn next(&mut self, n: isize) -> Option<*const AMsyncHave> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.next(n)
}
pub fn prev(&mut self, n: isize) -> Option<*const AMsyncHave> {
let detail = unsafe { &mut *(self.detail.as_mut_ptr() as *mut Detail) };
detail.prev(n)
}
pub fn reversed(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.reversed().into(),
}
}
pub fn rewound(&self) -> Self {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
Self {
detail: detail.rewound().into(),
}
}
}
impl AsRef<[am::sync::Have]> for AMsyncHaves {
fn as_ref(&self) -> &[am::sync::Have] {
let detail = unsafe { &*(self.detail.as_ptr() as *const Detail) };
unsafe { std::slice::from_raw_parts(detail.ptr as *const am::sync::Have, detail.len) }
}
}
impl Default for AMsyncHaves {
fn default() -> Self {
Self {
detail: [0; USIZE_USIZE_USIZE_USIZE_],
}
}
}
/// \memberof AMsyncHaves
/// \brief Advances an iterator over a sequence of synchronization haves by at
/// most \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] sync_haves A pointer to an `AMsyncHaves` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \pre \p sync_haves `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesAdvance(sync_haves: *mut AMsyncHaves, n: isize) {
if let Some(sync_haves) = sync_haves.as_mut() {
sync_haves.advance(n);
};
}
/// \memberof AMsyncHaves
/// \brief Tests the equality of two sequences of synchronization haves
/// underlying a pair of iterators.
///
/// \param[in] sync_haves1 A pointer to an `AMsyncHaves` struct.
/// \param[in] sync_haves2 A pointer to an `AMsyncHaves` struct.
/// \return `true` if \p sync_haves1 `==` \p sync_haves2 and `false` otherwise.
/// \pre \p sync_haves1 `!= NULL`.
/// \pre \p sync_haves2 `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves1 must be a valid pointer to an AMsyncHaves
/// sync_haves2 must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesEqual(
sync_haves1: *const AMsyncHaves,
sync_haves2: *const AMsyncHaves,
) -> bool {
match (sync_haves1.as_ref(), sync_haves2.as_ref()) {
(Some(sync_haves1), Some(sync_haves2)) => sync_haves1.as_ref() == sync_haves2.as_ref(),
(None, Some(_)) | (Some(_), None) | (None, None) => false,
}
}
/// \memberof AMsyncHaves
/// \brief Gets the synchronization have at the current position of an iterator
/// over a sequence of synchronization haves and then advances it by at
/// most \p |n| positions where the sign of \p n is relative to the
/// iterator's direction.
///
/// \param[in,out] sync_haves A pointer to an `AMsyncHaves` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMsyncHave` struct that's `NULL` when
/// \p sync_haves was previously advanced past its forward/reverse
/// limit.
/// \pre \p sync_haves `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesNext(
sync_haves: *mut AMsyncHaves,
n: isize,
) -> *const AMsyncHave {
if let Some(sync_haves) = sync_haves.as_mut() {
if let Some(sync_have) = sync_haves.next(n) {
return sync_have;
}
}
std::ptr::null()
}
/// \memberof AMsyncHaves
/// \brief Advances an iterator over a sequence of synchronization haves by at
/// most \p |n| positions where the sign of \p n is relative to the
/// iterator's direction and then gets the synchronization have at its
/// new position.
///
/// \param[in,out] sync_haves A pointer to an `AMsyncHaves` struct.
/// \param[in] n The direction (\p -n -> opposite, \p n -> same) and maximum
/// number of positions to advance.
/// \return A pointer to an `AMsyncHave` struct that's `NULL` when
/// \p sync_haves is presently advanced past its forward/reverse limit.
/// \pre \p sync_haves `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesPrev(
sync_haves: *mut AMsyncHaves,
n: isize,
) -> *const AMsyncHave {
if let Some(sync_haves) = sync_haves.as_mut() {
if let Some(sync_have) = sync_haves.prev(n) {
return sync_have;
}
}
std::ptr::null()
}
/// \memberof AMsyncHaves
/// \brief Gets the size of the sequence of synchronization haves underlying an
/// iterator.
///
/// \param[in] sync_haves A pointer to an `AMsyncHaves` struct.
/// \return The count of values in \p sync_haves.
/// \pre \p sync_haves `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesSize(sync_haves: *const AMsyncHaves) -> usize {
if let Some(sync_haves) = sync_haves.as_ref() {
sync_haves.len()
} else {
0
}
}
/// \memberof AMsyncHaves
/// \brief Creates an iterator over the same sequence of synchronization haves
/// as the given one but with the opposite position and direction.
///
/// \param[in] sync_haves A pointer to an `AMsyncHaves` struct.
/// \return An `AMsyncHaves` struct
/// \pre \p sync_haves `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesReversed(sync_haves: *const AMsyncHaves) -> AMsyncHaves {
if let Some(sync_haves) = sync_haves.as_ref() {
sync_haves.reversed()
} else {
Default::default()
}
}
/// \memberof AMsyncHaves
/// \brief Creates an iterator at the starting position over the same sequence
/// of synchronization haves as the given one.
///
/// \param[in] sync_haves A pointer to an `AMsyncHaves` struct.
/// \return An `AMsyncHaves` struct
/// \pre \p sync_haves `!= NULL`.
/// \internal
///
/// #Safety
/// sync_haves must be a valid pointer to an AMsyncHaves
#[no_mangle]
pub unsafe extern "C" fn AMsyncHavesRewound(sync_haves: *const AMsyncHaves) -> AMsyncHaves {
if let Some(sync_haves) = sync_haves.as_ref() {
sync_haves.rewound()
} else {
Default::default()
}
}

View file

@ -3,18 +3,15 @@ use std::cell::RefCell;
use std::collections::BTreeMap; use std::collections::BTreeMap;
use crate::change::AMchange; use crate::change::AMchange;
use crate::change_hashes::AMchangeHashes;
use crate::changes::AMchanges;
use crate::result::{to_result, AMresult}; use crate::result::{to_result, AMresult};
use crate::sync::have::AMsyncHave; use crate::sync::have::AMsyncHave;
use crate::sync::haves::AMsyncHaves;
macro_rules! to_sync_message { macro_rules! to_sync_message {
($handle:expr) => {{ ($handle:expr) => {{
let handle = $handle.as_ref(); let handle = $handle.as_ref();
match handle { match handle {
Some(b) => b, Some(b) => b,
None => return AMresult::err("Invalid AMsyncMessage pointer").into(), None => return AMresult::error("Invalid `AMsyncMessage*`").into(),
} }
}}; }};
} }
@ -51,55 +48,52 @@ impl AsRef<am::sync::Message> for AMsyncMessage {
/// \brief Gets the changes for the recipient to apply. /// \brief Gets the changes for the recipient to apply.
/// ///
/// \param[in] sync_message A pointer to an `AMsyncMessage` struct. /// \param[in] sync_message A pointer to an `AMsyncMessage` struct.
/// \return An `AMchanges` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE` items.
/// \pre \p sync_message `!= NULL`. /// \pre \p sync_message `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_message must be a valid pointer to an AMsyncMessage /// sync_message must be a valid pointer to an AMsyncMessage
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncMessageChanges(sync_message: *const AMsyncMessage) -> AMchanges { pub unsafe extern "C" fn AMsyncMessageChanges(sync_message: *const AMsyncMessage) -> *mut AMresult {
if let Some(sync_message) = sync_message.as_ref() { to_result(match sync_message.as_ref() {
AMchanges::new( Some(sync_message) => sync_message.body.changes.as_slice(),
&sync_message.body.changes, None => Default::default(),
&mut sync_message.changes_storage.borrow_mut(), })
)
} else {
Default::default()
}
} }
/// \memberof AMsyncMessage /// \memberof AMsyncMessage
/// \brief Decodes a sequence of bytes into a synchronization message. /// \brief Decodes an array of bytes into a synchronization message.
/// ///
/// \param[in] src A pointer to an array of bytes. /// \param[in] src A pointer to an array of bytes.
/// \param[in] count The number of bytes in \p src to decode. /// \param[in] count The count of bytes to decode from the array pointed to by
/// \return A pointer to an `AMresult` struct containing an `AMsyncMessage` /// \p src.
/// struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_SYNC_MESSAGE` item.
/// \pre \p src `!= NULL`. /// \pre \p src `!= NULL`
/// \pre `0 <` \p count `<= sizeof(`\p src`)`. /// \pre `sizeof(`\p src `) > 0`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre \p count `<= sizeof(`\p src `)`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// src must be a byte array of size `>= count` /// src must be a byte array of length `>= count`
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncMessageDecode(src: *const u8, count: usize) -> *mut AMresult { pub unsafe extern "C" fn AMsyncMessageDecode(src: *const u8, count: usize) -> *mut AMresult {
let mut data = Vec::new(); let data = std::slice::from_raw_parts(src, count);
data.extend_from_slice(std::slice::from_raw_parts(src, count)); to_result(am::sync::Message::decode(data))
to_result(am::sync::Message::decode(&data))
} }
/// \memberof AMsyncMessage /// \memberof AMsyncMessage
/// \brief Encodes a synchronization message as a sequence of bytes. /// \brief Encodes a synchronization message as an array of bytes.
/// ///
/// \param[in] sync_message A pointer to an `AMsyncMessage` struct. /// \param[in] sync_message A pointer to an `AMsyncMessage` struct.
/// \return A pointer to an `AMresult` struct containing an array of bytes as /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_BYTES` item.
/// an `AMbyteSpan` struct. /// \pre \p sync_message `!= NULL`
/// \pre \p sync_message `!= NULL`. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -114,41 +108,40 @@ pub unsafe extern "C" fn AMsyncMessageEncode(sync_message: *const AMsyncMessage)
/// \brief Gets a summary of the changes that the sender already has. /// \brief Gets a summary of the changes that the sender already has.
/// ///
/// \param[in] sync_message A pointer to an `AMsyncMessage` struct. /// \param[in] sync_message A pointer to an `AMsyncMessage` struct.
/// \return An `AMhaves` struct. /// \return A pointer to an `AMresult` struct with `AM_SYNC_HAVE` items.
/// \pre \p sync_message `!= NULL`. /// \pre \p sync_message `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_message must be a valid pointer to an AMsyncMessage /// sync_message must be a valid pointer to an AMsyncMessage
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncMessageHaves(sync_message: *const AMsyncMessage) -> AMsyncHaves { pub unsafe extern "C" fn AMsyncMessageHaves(sync_message: *const AMsyncMessage) -> *mut AMresult {
if let Some(sync_message) = sync_message.as_ref() { to_result(match sync_message.as_ref() {
AMsyncHaves::new( Some(sync_message) => sync_message.as_ref().have.as_slice(),
&sync_message.as_ref().have, None => Default::default(),
&mut sync_message.haves_storage.borrow_mut(), })
)
} else {
Default::default()
}
} }
/// \memberof AMsyncMessage /// \memberof AMsyncMessage
/// \brief Gets the heads of the sender. /// \brief Gets the heads of the sender.
/// ///
/// \param[in] sync_message A pointer to an `AMsyncMessage` struct. /// \param[in] sync_message A pointer to an `AMsyncMessage` struct.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_message `!= NULL`. /// \pre \p sync_message `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_message must be a valid pointer to an AMsyncMessage /// sync_message must be a valid pointer to an AMsyncMessage
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncMessageHeads(sync_message: *const AMsyncMessage) -> AMchangeHashes { pub unsafe extern "C" fn AMsyncMessageHeads(sync_message: *const AMsyncMessage) -> *mut AMresult {
if let Some(sync_message) = sync_message.as_ref() { to_result(match sync_message.as_ref() {
AMchangeHashes::new(&sync_message.as_ref().heads) Some(sync_message) => sync_message.as_ref().heads.as_slice(),
} else { None => Default::default(),
Default::default() })
}
} }
/// \memberof AMsyncMessage /// \memberof AMsyncMessage
@ -156,17 +149,18 @@ pub unsafe extern "C" fn AMsyncMessageHeads(sync_message: *const AMsyncMessage)
/// by the recipient. /// by the recipient.
/// ///
/// \param[in] sync_message A pointer to an `AMsyncMessage` struct. /// \param[in] sync_message A pointer to an `AMsyncMessage` struct.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_message `!= NULL`. /// \pre \p sync_message `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_message must be a valid pointer to an AMsyncMessage /// sync_message must be a valid pointer to an AMsyncMessage
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncMessageNeeds(sync_message: *const AMsyncMessage) -> AMchangeHashes { pub unsafe extern "C" fn AMsyncMessageNeeds(sync_message: *const AMsyncMessage) -> *mut AMresult {
if let Some(sync_message) = sync_message.as_ref() { to_result(match sync_message.as_ref() {
AMchangeHashes::new(&sync_message.as_ref().need) Some(sync_message) => sync_message.as_ref().need.as_slice(),
} else { None => Default::default(),
Default::default() })
}
} }

View file

@ -2,17 +2,15 @@ use automerge as am;
use std::cell::RefCell; use std::cell::RefCell;
use std::collections::BTreeMap; use std::collections::BTreeMap;
use crate::change_hashes::AMchangeHashes;
use crate::result::{to_result, AMresult}; use crate::result::{to_result, AMresult};
use crate::sync::have::AMsyncHave; use crate::sync::have::AMsyncHave;
use crate::sync::haves::AMsyncHaves;
macro_rules! to_sync_state { macro_rules! to_sync_state {
($handle:expr) => {{ ($handle:expr) => {{
let handle = $handle.as_ref(); let handle = $handle.as_ref();
match handle { match handle {
Some(b) => b, Some(b) => b,
None => return AMresult::err("Invalid AMsyncState pointer").into(), None => return AMresult::error("Invalid `AMsyncState*`").into(),
} }
}}; }};
} }
@ -56,36 +54,35 @@ impl From<AMsyncState> for *mut AMsyncState {
} }
/// \memberof AMsyncState /// \memberof AMsyncState
/// \brief Decodes a sequence of bytes into a synchronization state. /// \brief Decodes an array of bytes into a synchronization state.
/// ///
/// \param[in] src A pointer to an array of bytes. /// \param[in] src A pointer to an array of bytes.
/// \param[in] count The number of bytes in \p src to decode. /// \param[in] count The count of bytes to decode from the array pointed to by
/// \return A pointer to an `AMresult` struct containing an `AMsyncState` /// \p src.
/// struct. /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_SYNC_STATE` item.
/// \pre \p src `!= NULL`. /// \pre \p src `!= NULL`
/// \pre `0 <` \p count `<= sizeof(`\p src`)`. /// \pre `sizeof(`\p src `) > 0`
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// \pre \p count `<= sizeof(`\p src `)`
/// in order to prevent a memory leak. /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// src must be a byte array of size `>= count` /// src must be a byte array of length `>= count`
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncStateDecode(src: *const u8, count: usize) -> *mut AMresult { pub unsafe extern "C" fn AMsyncStateDecode(src: *const u8, count: usize) -> *mut AMresult {
let mut data = Vec::new(); let data = std::slice::from_raw_parts(src, count);
data.extend_from_slice(std::slice::from_raw_parts(src, count)); to_result(am::sync::State::decode(data))
to_result(am::sync::State::decode(&data))
} }
/// \memberof AMsyncState /// \memberof AMsyncState
/// \brief Encodes a synchronizaton state as a sequence of bytes. /// \brief Encodes a synchronization state as an array of bytes.
/// ///
/// \param[in] sync_state A pointer to an `AMsyncState` struct. /// \param[in] sync_state A pointer to an `AMsyncState` struct.
/// \return A pointer to an `AMresult` struct containing an array of bytes as /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_BYTE_SPAN` item.
/// an `AMbyteSpan` struct. /// \pre \p sync_state `!= NULL`
/// \pre \p sync_state `!= NULL`. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
@ -102,8 +99,9 @@ pub unsafe extern "C" fn AMsyncStateEncode(sync_state: *const AMsyncState) -> *m
/// \param[in] sync_state1 A pointer to an `AMsyncState` struct. /// \param[in] sync_state1 A pointer to an `AMsyncState` struct.
/// \param[in] sync_state2 A pointer to an `AMsyncState` struct. /// \param[in] sync_state2 A pointer to an `AMsyncState` struct.
/// \return `true` if \p sync_state1 `==` \p sync_state2 and `false` otherwise. /// \return `true` if \p sync_state1 `==` \p sync_state2 and `false` otherwise.
/// \pre \p sync_state1 `!= NULL`. /// \pre \p sync_state1 `!= NULL`
/// \pre \p sync_state2 `!= NULL`. /// \pre \p sync_state2 `!= NULL`
/// \post `!(`\p sync_state1 `&&` \p sync_state2 `) -> false`
/// \internal /// \internal
/// ///
/// #Safety /// #Safety
@ -116,18 +114,17 @@ pub unsafe extern "C" fn AMsyncStateEqual(
) -> bool { ) -> bool {
match (sync_state1.as_ref(), sync_state2.as_ref()) { match (sync_state1.as_ref(), sync_state2.as_ref()) {
(Some(sync_state1), Some(sync_state2)) => sync_state1.as_ref() == sync_state2.as_ref(), (Some(sync_state1), Some(sync_state2)) => sync_state1.as_ref() == sync_state2.as_ref(),
(None, Some(_)) | (Some(_), None) | (None, None) => false, (None, None) | (None, Some(_)) | (Some(_), None) => false,
} }
} }
/// \memberof AMsyncState /// \memberof AMsyncState
/// \brief Allocates a new synchronization state and initializes it with /// \brief Allocates a new synchronization state and initializes it from
/// defaults. /// default values.
/// ///
/// \return A pointer to an `AMresult` struct containing a pointer to an /// \return A pointer to an `AMresult` struct with an `AM_VAL_TYPE_SYNC_STATE` item.
/// `AMsyncState` struct. /// \warning The returned `AMresult` struct pointer must be passed to
/// \warning The returned `AMresult` struct must be deallocated with `AMfree()` /// `AMresultFree()` in order to avoid a memory leak.
/// in order to prevent a memory leak.
#[no_mangle] #[no_mangle]
pub extern "C" fn AMsyncStateInit() -> *mut AMresult { pub extern "C" fn AMsyncStateInit() -> *mut AMresult {
to_result(am::sync::State::new()) to_result(am::sync::State::new())
@ -137,40 +134,36 @@ pub extern "C" fn AMsyncStateInit() -> *mut AMresult {
/// \brief Gets the heads that are shared by both peers. /// \brief Gets the heads that are shared by both peers.
/// ///
/// \param[in] sync_state A pointer to an `AMsyncState` struct. /// \param[in] sync_state A pointer to an `AMsyncState` struct.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_state `!= NULL`. /// \pre \p sync_state `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_state must be a valid pointer to an AMsyncState /// sync_state must be a valid pointer to an AMsyncState
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncStateSharedHeads(sync_state: *const AMsyncState) -> AMchangeHashes { pub unsafe extern "C" fn AMsyncStateSharedHeads(sync_state: *const AMsyncState) -> *mut AMresult {
if let Some(sync_state) = sync_state.as_ref() { let sync_state = to_sync_state!(sync_state);
AMchangeHashes::new(&sync_state.as_ref().shared_heads) to_result(sync_state.as_ref().shared_heads.as_slice())
} else {
Default::default()
}
} }
/// \memberof AMsyncState /// \memberof AMsyncState
/// \brief Gets the heads that were last sent by this peer. /// \brief Gets the heads that were last sent by this peer.
/// ///
/// \param[in] sync_state A pointer to an `AMsyncState` struct. /// \param[in] sync_state A pointer to an `AMsyncState` struct.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_state `!= NULL`. /// \pre \p sync_state `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_state must be a valid pointer to an AMsyncState /// sync_state must be a valid pointer to an AMsyncState
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncStateLastSentHeads( pub unsafe extern "C" fn AMsyncStateLastSentHeads(sync_state: *const AMsyncState) -> *mut AMresult {
sync_state: *const AMsyncState, let sync_state = to_sync_state!(sync_state);
) -> AMchangeHashes { to_result(sync_state.as_ref().last_sent_heads.as_slice())
if let Some(sync_state) = sync_state.as_ref() {
AMchangeHashes::new(&sync_state.as_ref().last_sent_heads)
} else {
Default::default()
}
} }
/// \memberof AMsyncState /// \memberof AMsyncState
@ -178,11 +171,13 @@ pub unsafe extern "C" fn AMsyncStateLastSentHeads(
/// ///
/// \param[in] sync_state A pointer to an `AMsyncState` struct. /// \param[in] sync_state A pointer to an `AMsyncState` struct.
/// \param[out] has_value A pointer to a boolean flag that is set to `true` if /// \param[out] has_value A pointer to a boolean flag that is set to `true` if
/// the returned `AMhaves` struct is relevant, `false` otherwise. /// the returned `AMitems` struct is relevant, `false` otherwise.
/// \return An `AMhaves` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_SYNC_HAVE` items.
/// \pre \p sync_state `!= NULL`. /// \pre \p sync_state `!= NULL`
/// \pre \p has_value `!= NULL`. /// \pre \p has_value `!= NULL`
/// \internal /// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
//// \internal
/// ///
/// # Safety /// # Safety
/// sync_state must be a valid pointer to an AMsyncState /// sync_state must be a valid pointer to an AMsyncState
@ -191,15 +186,15 @@ pub unsafe extern "C" fn AMsyncStateLastSentHeads(
pub unsafe extern "C" fn AMsyncStateTheirHaves( pub unsafe extern "C" fn AMsyncStateTheirHaves(
sync_state: *const AMsyncState, sync_state: *const AMsyncState,
has_value: *mut bool, has_value: *mut bool,
) -> AMsyncHaves { ) -> *mut AMresult {
if let Some(sync_state) = sync_state.as_ref() { if let Some(sync_state) = sync_state.as_ref() {
if let Some(haves) = &sync_state.as_ref().their_have { if let Some(haves) = &sync_state.as_ref().their_have {
*has_value = true; *has_value = true;
return AMsyncHaves::new(haves, &mut sync_state.their_haves_storage.borrow_mut()); return to_result(haves.as_slice());
}; }
}; };
*has_value = false; *has_value = false;
Default::default() to_result(Vec::<am::sync::Have>::new())
} }
/// \memberof AMsyncState /// \memberof AMsyncState
@ -207,29 +202,31 @@ pub unsafe extern "C" fn AMsyncStateTheirHaves(
/// ///
/// \param[in] sync_state A pointer to an `AMsyncState` struct. /// \param[in] sync_state A pointer to an `AMsyncState` struct.
/// \param[out] has_value A pointer to a boolean flag that is set to `true` if /// \param[out] has_value A pointer to a boolean flag that is set to `true` if
/// the returned `AMchangeHashes` struct is relevant, `false` /// the returned `AMitems` struct is relevant, `false`
/// otherwise. /// otherwise.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_state `!= NULL`. /// \pre \p sync_state `!= NULL`
/// \pre \p has_value `!= NULL`. /// \pre \p has_value `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_state must be a valid pointer to an AMsyncState /// sync_state must be a valid pointer to an AMsyncState
/// has_value must be a valid pointer to a bool. /// has_value must be a valid pointer to a bool
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncStateTheirHeads( pub unsafe extern "C" fn AMsyncStateTheirHeads(
sync_state: *const AMsyncState, sync_state: *const AMsyncState,
has_value: *mut bool, has_value: *mut bool,
) -> AMchangeHashes { ) -> *mut AMresult {
if let Some(sync_state) = sync_state.as_ref() { if let Some(sync_state) = sync_state.as_ref() {
if let Some(change_hashes) = &sync_state.as_ref().their_heads { if let Some(change_hashes) = &sync_state.as_ref().their_heads {
*has_value = true; *has_value = true;
return AMchangeHashes::new(change_hashes); return to_result(change_hashes.as_slice());
} }
}; };
*has_value = false; *has_value = false;
Default::default() to_result(Vec::<am::ChangeHash>::new())
} }
/// \memberof AMsyncState /// \memberof AMsyncState
@ -237,27 +234,29 @@ pub unsafe extern "C" fn AMsyncStateTheirHeads(
/// ///
/// \param[in] sync_state A pointer to an `AMsyncState` struct. /// \param[in] sync_state A pointer to an `AMsyncState` struct.
/// \param[out] has_value A pointer to a boolean flag that is set to `true` if /// \param[out] has_value A pointer to a boolean flag that is set to `true` if
/// the returned `AMchangeHashes` struct is relevant, `false` /// the returned `AMitems` struct is relevant, `false`
/// otherwise. /// otherwise.
/// \return An `AMchangeHashes` struct. /// \return A pointer to an `AMresult` struct with `AM_VAL_TYPE_CHANGE_HASH` items.
/// \pre \p sync_state `!= NULL`. /// \pre \p sync_state `!= NULL`
/// \pre \p has_value `!= NULL`. /// \pre \p has_value `!= NULL`
/// \warning The returned `AMresult` struct pointer must be passed to
/// `AMresultFree()` in order to avoid a memory leak.
/// \internal /// \internal
/// ///
/// # Safety /// # Safety
/// sync_state must be a valid pointer to an AMsyncState /// sync_state must be a valid pointer to an AMsyncState
/// has_value must be a valid pointer to a bool. /// has_value must be a valid pointer to a bool
#[no_mangle] #[no_mangle]
pub unsafe extern "C" fn AMsyncStateTheirNeeds( pub unsafe extern "C" fn AMsyncStateTheirNeeds(
sync_state: *const AMsyncState, sync_state: *const AMsyncState,
has_value: *mut bool, has_value: *mut bool,
) -> AMchangeHashes { ) -> *mut AMresult {
if let Some(sync_state) = sync_state.as_ref() { if let Some(sync_state) = sync_state.as_ref() {
if let Some(change_hashes) = &sync_state.as_ref().their_need { if let Some(change_hashes) = &sync_state.as_ref().their_need {
*has_value = true; *has_value = true;
return AMchangeHashes::new(change_hashes); return to_result(change_hashes.as_slice());
} }
}; };
*has_value = false; *has_value = false;
Default::default() to_result(Vec::<am::ChangeHash>::new())
} }

View file

@ -0,0 +1,33 @@
#include <stdarg.h>
#include <automerge-c/utils/result.h>
AMresult* AMresultFrom(int count, ...) {
AMresult* result = NULL;
bool is_ok = true;
va_list args;
va_start(args, count);
for (int i = 0; i != count; ++i) {
AMresult* src = va_arg(args, AMresult*);
AMresult* dest = result;
is_ok = (AMresultStatus(src) == AM_STATUS_OK);
if (is_ok) {
if (dest) {
result = AMresultCat(dest, src);
is_ok = (AMresultStatus(result) == AM_STATUS_OK);
AMresultFree(dest);
AMresultFree(src);
} else {
result = src;
}
} else {
AMresultFree(src);
}
}
va_end(args);
if (!is_ok) {
AMresultFree(result);
result = NULL;
}
return result;
}

View file

@ -0,0 +1,106 @@
#include <stdio.h>
#include <stdlib.h>
#include <automerge-c/utils/stack.h>
#include <automerge-c/utils/string.h>
void AMstackFree(AMstack** stack) {
if (stack) {
while (*stack) {
AMresultFree(AMstackPop(stack, NULL));
}
}
}
AMresult* AMstackPop(AMstack** stack, const AMresult* result) {
if (!stack) {
return NULL;
}
AMstack** prev = stack;
if (result) {
while (*prev && ((*prev)->result != result)) {
*prev = (*prev)->prev;
}
}
if (!*prev) {
return NULL;
}
AMstack* target = *prev;
*prev = target->prev;
AMresult* popped = target->result;
free(target);
return popped;
}
AMresult* AMstackResult(AMstack** stack, AMresult* result, AMstackCallback callback, void* data) {
if (!stack) {
if (callback) {
/* Create a local stack so that the callback can still examine the
* result. */
AMstack node = {.result = result, .prev = NULL};
AMstack* stack = &node;
callback(&stack, data);
} else {
/* \note There is no reason to call this function when both the
* stack and the callback are null. */
fprintf(stderr, "ERROR: NULL AMstackCallback!\n");
}
/* \note Nothing can be returned without a stack regardless of
* whether or not the callback validated the result. */
AMresultFree(result);
return NULL;
}
/* Always push the result onto the stack, even if it's null, so that the
* callback can examine it. */
AMstack* next = calloc(1, sizeof(AMstack));
*next = (AMstack){.result = result, .prev = *stack};
AMstack* top = next;
*stack = top;
if (callback) {
if (!callback(stack, data)) {
/* The result didn't pass the callback's examination. */
return NULL;
}
} else {
/* Report an obvious error. */
if (result) {
AMbyteSpan const err_msg = AMresultError(result);
if (err_msg.src && err_msg.count) {
/* \note The callback may be null because the result is supposed
* to be examined externally so return it despite an
* error. */
char* const cstr = AMstrdup(err_msg, NULL);
fprintf(stderr, "WARNING: %s.\n", cstr);
free(cstr);
}
} else {
/* \note There's no reason to call this function when both the
* result and the callback are null. */
fprintf(stderr, "ERROR: NULL AMresult*!\n");
return NULL;
}
}
return result;
}
AMitem* AMstackItem(AMstack** stack, AMresult* result, AMstackCallback callback, void* data) {
AMitems items = AMstackItems(stack, result, callback, data);
return AMitemsNext(&items, 1);
}
AMitems AMstackItems(AMstack** stack, AMresult* result, AMstackCallback callback, void* data) {
return (AMstackResult(stack, result, callback, data)) ? AMresultItems(result) : (AMitems){0};
}
size_t AMstackSize(AMstack const* const stack) {
if (!stack) {
return 0;
}
size_t count = 0;
AMstack const* prev = stack;
while (prev) {
++count;
prev = prev->prev;
}
return count;
}

View file

@ -0,0 +1,9 @@
#include <stdlib.h>
#include <automerge-c/utils/stack_callback_data.h>
AMstackCallbackData* AMstackCallbackDataInit(AMvalType const bitmask, char const* const file, int const line) {
AMstackCallbackData* data = malloc(sizeof(AMstackCallbackData));
*data = (AMstackCallbackData){.bitmask = bitmask, .file = file, .line = line};
return data;
}

View file

@ -0,0 +1,46 @@
#include <stdlib.h>
#include <string.h>
#include <automerge-c/utils/string.h>
char* AMstrdup(AMbyteSpan const str, char const* nul) {
if (!str.src) {
return NULL;
} else if (!str.count) {
return strdup("");
}
nul = (nul) ? nul : "\\0";
size_t const nul_len = strlen(nul);
char* dup = NULL;
size_t dup_len = 0;
char const* begin = str.src;
char const* end = begin;
for (size_t i = 0; i != str.count; ++i, ++end) {
if (!*end) {
size_t const len = end - begin;
size_t const alloc_len = dup_len + len + nul_len;
if (dup) {
dup = realloc(dup, alloc_len + 1);
} else {
dup = malloc(alloc_len + 1);
}
memcpy(dup + dup_len, begin, len);
memcpy(dup + dup_len + len, nul, nul_len);
dup[alloc_len] = '\0';
begin = end + 1;
dup_len = alloc_len;
}
}
if (begin != end) {
size_t const len = end - begin;
size_t const alloc_len = dup_len + len;
if (dup) {
dup = realloc(dup, alloc_len + 1);
} else {
dup = malloc(alloc_len + 1);
}
memcpy(dup + dup_len, begin, len);
dup[alloc_len] = '\0';
}
return dup;
}

View file

@ -1,53 +1,51 @@
cmake_minimum_required(VERSION 3.18 FATAL_ERROR) find_package(cmocka CONFIG REQUIRED)
find_package(cmocka REQUIRED)
add_executable( add_executable(
test_${LIBRARY_NAME} ${LIBRARY_NAME}_test
actor_id_tests.c actor_id_tests.c
base_state.c
byte_span_tests.c
cmocka_utils.c
enum_string_tests.c
doc_state.c
doc_tests.c doc_tests.c
group_state.c item_tests.c
list_tests.c list_tests.c
macro_utils.c macro_utils.c
main.c main.c
map_tests.c map_tests.c
stack_utils.c
str_utils.c str_utils.c
ported_wasm/basic_tests.c ported_wasm/basic_tests.c
ported_wasm/suite.c ported_wasm/suite.c
ported_wasm/sync_tests.c ported_wasm/sync_tests.c
) )
set_target_properties(test_${LIBRARY_NAME} PROPERTIES LINKER_LANGUAGE C) set_target_properties(${LIBRARY_NAME}_test PROPERTIES LINKER_LANGUAGE C)
# \note An imported library's INTERFACE_INCLUDE_DIRECTORIES property can't if(WIN32)
# contain a non-existent path so its build-time include directory set(CMOCKA "cmocka::cmocka")
# must be specified for all of its dependent targets instead. else()
target_include_directories( set(CMOCKA "cmocka")
test_${LIBRARY_NAME} endif()
PRIVATE "$<BUILD_INTERFACE:${CBINDGEN_INCLUDEDIR}>"
)
target_link_libraries(test_${LIBRARY_NAME} PRIVATE cmocka ${LIBRARY_NAME}) target_link_libraries(${LIBRARY_NAME}_test PRIVATE ${CMOCKA} ${LIBRARY_NAME})
add_dependencies(test_${LIBRARY_NAME} ${LIBRARY_NAME}_artifacts) add_dependencies(${LIBRARY_NAME}_test ${BINDINGS_NAME}_artifacts)
if(BUILD_SHARED_LIBS AND WIN32) if(BUILD_SHARED_LIBS AND WIN32)
add_custom_command( add_custom_command(
TARGET test_${LIBRARY_NAME} TARGET ${LIBRARY_NAME}_test
POST_BUILD POST_BUILD
COMMAND ${CMAKE_COMMAND} -E copy_if_different COMMAND ${CMAKE_COMMAND} -E copy_if_different $<TARGET_FILE:${LIBRARY_NAME}> $<TARGET_FILE_DIR:${LIBRARY_NAME}_test>
${CARGO_CURRENT_BINARY_DIR}/${CMAKE_SHARED_LIBRARY_PREFIX}${LIBRARY_NAME}${CMAKE_${CMAKE_BUILD_TYPE}_POSTFIX}${CMAKE_SHARED_LIBRARY_SUFFIX} COMMENT "Copying the DLL into the tests directory..."
${CMAKE_CURRENT_BINARY_DIR}
COMMENT "Copying the DLL built by Cargo into the test directory..."
VERBATIM VERBATIM
) )
endif() endif()
add_test(NAME test_${LIBRARY_NAME} COMMAND test_${LIBRARY_NAME}) add_test(NAME ${LIBRARY_NAME}_test COMMAND ${LIBRARY_NAME}_test)
add_custom_command( add_custom_command(
TARGET test_${LIBRARY_NAME} TARGET ${LIBRARY_NAME}_test
POST_BUILD POST_BUILD
COMMAND COMMAND
${CMAKE_CTEST_COMMAND} --config $<CONFIG> --output-on-failure ${CMAKE_CTEST_COMMAND} --config $<CONFIG> --output-on-failure

View file

@ -14,99 +14,126 @@
#include "cmocka_utils.h" #include "cmocka_utils.h"
#include "str_utils.h" #include "str_utils.h"
/**
* \brief State for a group of cmocka test cases.
*/
typedef struct { typedef struct {
/** An actor ID as an array of bytes. */
uint8_t* src; uint8_t* src;
AMbyteSpan str; /** The count of bytes in \p src. */
size_t count; size_t count;
} GroupState; /** A stack of results. */
AMstack* stack;
/** An actor ID as a hexadecimal string. */
AMbyteSpan str;
} DocState;
static int group_setup(void** state) { static int group_setup(void** state) {
GroupState* group_state = test_calloc(1, sizeof(GroupState)); DocState* doc_state = test_calloc(1, sizeof(DocState));
group_state->str.src = "000102030405060708090a0b0c0d0e0f"; doc_state->str = AMstr("000102030405060708090a0b0c0d0e0f");
group_state->str.count = strlen(group_state->str.src); doc_state->count = doc_state->str.count / 2;
group_state->count = group_state->str.count / 2; doc_state->src = test_calloc(doc_state->count, sizeof(uint8_t));
group_state->src = test_malloc(group_state->count); hex_to_bytes(doc_state->str.src, doc_state->src, doc_state->count);
hex_to_bytes(group_state->str.src, group_state->src, group_state->count); *state = doc_state;
*state = group_state;
return 0; return 0;
} }
static int group_teardown(void** state) { static int group_teardown(void** state) {
GroupState* group_state = *state; DocState* doc_state = *state;
test_free(group_state->src); test_free(doc_state->src);
test_free(group_state); AMstackFree(&doc_state->stack);
test_free(doc_state);
return 0; return 0;
} }
static void test_AMactorIdInit() { static void test_AMactorIdFromBytes(void** state) {
DocState* doc_state = *state;
AMstack** stack_ptr = &doc_state->stack;
/* Non-empty string. */
AMresult* result = AMstackResult(stack_ptr, AMactorIdFromBytes(doc_state->src, doc_state->count), NULL, NULL);
if (AMresultStatus(result) != AM_STATUS_OK) {
fail_msg_view("%s", AMresultError(result));
}
assert_int_equal(AMresultSize(result), 1);
AMitem* const item = AMresultItem(result);
assert_int_equal(AMitemValType(item), AM_VAL_TYPE_ACTOR_ID);
AMactorId const* actor_id;
assert_true(AMitemToActorId(item, &actor_id));
AMbyteSpan const bytes = AMactorIdBytes(actor_id);
assert_int_equal(bytes.count, doc_state->count);
assert_memory_equal(bytes.src, doc_state->src, bytes.count);
/* Empty array. */
/** \todo Find out if this is intentionally allowed. */
result = AMstackResult(stack_ptr, AMactorIdFromBytes(doc_state->src, 0), NULL, NULL);
if (AMresultStatus(result) != AM_STATUS_OK) {
fail_msg_view("%s", AMresultError(result));
}
/* NULL array. */
result = AMstackResult(stack_ptr, AMactorIdFromBytes(NULL, doc_state->count), NULL, NULL);
if (AMresultStatus(result) == AM_STATUS_OK) {
fail_msg("AMactorId from NULL.");
}
}
static void test_AMactorIdFromStr(void** state) {
DocState* doc_state = *state;
AMstack** stack_ptr = &doc_state->stack;
AMresult* result = AMstackResult(stack_ptr, AMactorIdFromStr(doc_state->str), NULL, NULL);
if (AMresultStatus(result) != AM_STATUS_OK) {
fail_msg_view("%s", AMresultError(result));
}
assert_int_equal(AMresultSize(result), 1);
AMitem* const item = AMresultItem(result);
assert_int_equal(AMitemValType(item), AM_VAL_TYPE_ACTOR_ID);
/* The hexadecimal string should've been decoded as identical bytes. */
AMactorId const* actor_id;
assert_true(AMitemToActorId(item, &actor_id));
AMbyteSpan const bytes = AMactorIdBytes(actor_id);
assert_int_equal(bytes.count, doc_state->count);
assert_memory_equal(bytes.src, doc_state->src, bytes.count);
/* The bytes should've been encoded as an identical hexadecimal string. */
assert_true(AMitemToActorId(item, &actor_id));
AMbyteSpan const str = AMactorIdStr(actor_id);
assert_int_equal(str.count, doc_state->str.count);
assert_memory_equal(str.src, doc_state->str.src, str.count);
}
static void test_AMactorIdInit(void** state) {
DocState* doc_state = *state;
AMstack** stack_ptr = &doc_state->stack;
AMresult* prior_result = NULL; AMresult* prior_result = NULL;
AMbyteSpan prior_bytes = {NULL, 0}; AMbyteSpan prior_bytes = {NULL, 0};
AMbyteSpan prior_str = {NULL, 0}; AMbyteSpan prior_str = {NULL, 0};
AMresult* result = NULL;
for (size_t i = 0; i != 11; ++i) { for (size_t i = 0; i != 11; ++i) {
result = AMactorIdInit(); AMresult* result = AMstackResult(stack_ptr, AMactorIdInit(), NULL, NULL);
if (AMresultStatus(result) != AM_STATUS_OK) { if (AMresultStatus(result) != AM_STATUS_OK) {
fail_msg_view("%s", AMerrorMessage(result)); fail_msg_view("%s", AMresultError(result));
} }
assert_int_equal(AMresultSize(result), 1); assert_int_equal(AMresultSize(result), 1);
AMvalue const value = AMresultValue(result); AMitem* const item = AMresultItem(result);
assert_int_equal(value.tag, AM_VALUE_ACTOR_ID); assert_int_equal(AMitemValType(item), AM_VAL_TYPE_ACTOR_ID);
AMbyteSpan const bytes = AMactorIdBytes(value.actor_id); AMactorId const* actor_id;
AMbyteSpan const str = AMactorIdStr(value.actor_id); assert_true(AMitemToActorId(item, &actor_id));
AMbyteSpan const bytes = AMactorIdBytes(actor_id);
assert_true(AMitemToActorId(item, &actor_id));
AMbyteSpan const str = AMactorIdStr(actor_id);
if (prior_result) { if (prior_result) {
size_t const max_byte_count = fmax(bytes.count, prior_bytes.count); size_t const max_byte_count = fmax(bytes.count, prior_bytes.count);
assert_memory_not_equal(bytes.src, prior_bytes.src, max_byte_count); assert_memory_not_equal(bytes.src, prior_bytes.src, max_byte_count);
size_t const max_char_count = fmax(str.count, prior_str.count); size_t const max_char_count = fmax(str.count, prior_str.count);
assert_memory_not_equal(str.src, prior_str.src, max_char_count); assert_memory_not_equal(str.src, prior_str.src, max_char_count);
AMfree(prior_result);
} }
prior_result = result; prior_result = result;
prior_bytes = bytes; prior_bytes = bytes;
prior_str = str; prior_str = str;
} }
AMfree(result);
}
static void test_AMactorIdInitBytes(void **state) {
GroupState* group_state = *state;
AMresult* const result = AMactorIdInitBytes(group_state->src, group_state->count);
if (AMresultStatus(result) != AM_STATUS_OK) {
fail_msg_view("%s", AMerrorMessage(result));
}
assert_int_equal(AMresultSize(result), 1);
AMvalue const value = AMresultValue(result);
assert_int_equal(value.tag, AM_VALUE_ACTOR_ID);
AMbyteSpan const bytes = AMactorIdBytes(value.actor_id);
assert_int_equal(bytes.count, group_state->count);
assert_memory_equal(bytes.src, group_state->src, bytes.count);
AMfree(result);
}
static void test_AMactorIdInitStr(void **state) {
GroupState* group_state = *state;
AMresult* const result = AMactorIdInitStr(group_state->str);
if (AMresultStatus(result) != AM_STATUS_OK) {
fail_msg_view("%s", AMerrorMessage(result));
}
assert_int_equal(AMresultSize(result), 1);
AMvalue const value = AMresultValue(result);
assert_int_equal(value.tag, AM_VALUE_ACTOR_ID);
/* The hexadecimal string should've been decoded as identical bytes. */
AMbyteSpan const bytes = AMactorIdBytes(value.actor_id);
assert_int_equal(bytes.count, group_state->count);
assert_memory_equal(bytes.src, group_state->src, bytes.count);
/* The bytes should've been encoded as an identical hexadecimal string. */
AMbyteSpan const str = AMactorIdStr(value.actor_id);
assert_int_equal(str.count, group_state->str.count);
assert_memory_equal(str.src, group_state->str.src, str.count);
AMfree(result);
} }
int run_actor_id_tests(void) { int run_actor_id_tests(void) {
const struct CMUnitTest tests[] = { const struct CMUnitTest tests[] = {
cmocka_unit_test(test_AMactorIdFromBytes),
cmocka_unit_test(test_AMactorIdFromStr),
cmocka_unit_test(test_AMactorIdInit), cmocka_unit_test(test_AMactorIdInit),
cmocka_unit_test(test_AMactorIdInitBytes),
cmocka_unit_test(test_AMactorIdInitStr),
}; };
return cmocka_run_group_tests(tests, group_setup, group_teardown); return cmocka_run_group_tests(tests, group_setup, group_teardown);

View file

@ -0,0 +1,17 @@
#include <stdlib.h>
/* local */
#include "base_state.h"
int setup_base(void** state) {
BaseState* base_state = calloc(1, sizeof(BaseState));
*state = base_state;
return 0;
}
int teardown_base(void** state) {
BaseState* base_state = *state;
AMstackFree(&base_state->stack);
free(base_state);
return 0;
}

View file

@ -0,0 +1,39 @@
#ifndef TESTS_BASE_STATE_H
#define TESTS_BASE_STATE_H
#include <stdint.h>
/* local */
#include <automerge-c/automerge.h>
#include <automerge-c/utils/stack.h>
/**
* \struct BaseState
* \brief The shared state for one or more cmocka test cases.
*/
typedef struct {
/** A stack of results. */
AMstack* stack;
} BaseState;
/**
* \memberof BaseState
* \brief Sets up the shared state for one or more cmocka test cases.
*
* \param[in,out] state A pointer to a pointer to a `BaseState` struct.
* \pre \p state `!= NULL`.
* \warning The `BaseState` struct returned through \p state must be
* passed to `teardown_base()` in order to avoid a memory leak.
*/
int setup_base(void** state);
/**
* \memberof BaseState
* \brief Tears down the shared state for one or more cmocka test cases.
*
* \param[in] state A pointer to a pointer to a `BaseState` struct.
* \pre \p state `!= NULL`.
*/
int teardown_base(void** state);
#endif /* TESTS_BASE_STATE_H */

View file

@ -0,0 +1,119 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* third-party */
#include <cmocka.h>
/* local */
#include <automerge-c/automerge.h>
#include <automerge-c/utils/string.h>
static void test_AMbytes(void** state) {
static char const DATA[] = {0x0, 0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8, 0x9, 0xa, 0xb, 0xc, 0xd, 0xe, 0xf};
AMbyteSpan bytes = AMbytes(DATA, sizeof(DATA));
assert_int_equal(bytes.count, sizeof(DATA));
assert_memory_equal(bytes.src, DATA, bytes.count);
assert_ptr_equal(bytes.src, DATA);
/* Empty view */
bytes = AMbytes(DATA, 0);
assert_int_equal(bytes.count, 0);
assert_ptr_equal(bytes.src, DATA);
/* Invalid array */
bytes = AMbytes(NULL, SIZE_MAX);
assert_int_not_equal(bytes.count, SIZE_MAX);
assert_int_equal(bytes.count, 0);
assert_ptr_equal(bytes.src, NULL);
}
static void test_AMstr(void** state) {
AMbyteSpan str = AMstr("abcdefghijkl");
assert_int_equal(str.count, strlen("abcdefghijkl"));
assert_memory_equal(str.src, "abcdefghijkl", str.count);
/* Empty string */
static char const* const EMPTY = "";
str = AMstr(EMPTY);
assert_int_equal(str.count, 0);
assert_ptr_equal(str.src, EMPTY);
/* Invalid string */
str = AMstr(NULL);
assert_int_equal(str.count, 0);
assert_ptr_equal(str.src, NULL);
}
static void test_AMstrCmp(void** state) {
/* Length ordering */
assert_int_equal(AMstrCmp(AMstr("abcdef"), AMstr("abcdefghijkl")), -1);
assert_int_equal(AMstrCmp(AMstr("abcdefghijkl"), AMstr("abcdefghijkl")), 0);
assert_int_equal(AMstrCmp(AMstr("abcdefghijkl"), AMstr("abcdef")), 1);
/* Lexicographical ordering */
assert_int_equal(AMstrCmp(AMstr("abcdef"), AMstr("ghijkl")), -1);
assert_int_equal(AMstrCmp(AMstr("ghijkl"), AMstr("abcdef")), 1);
/* Case ordering */
assert_int_equal(AMstrCmp(AMstr("ABCDEFGHIJKL"), AMstr("abcdefghijkl")), -1);
assert_int_equal(AMstrCmp(AMstr("ABCDEFGHIJKL"), AMstr("ABCDEFGHIJKL")), 0);
assert_int_equal(AMstrCmp(AMstr("abcdefghijkl"), AMstr("ABCDEFGHIJKL")), 1);
assert_int_equal(AMstrCmp(AMstr("ABCDEFGHIJKL"), AMstr("abcdef")), -1);
assert_int_equal(AMstrCmp(AMstr("abcdef"), AMstr("ABCDEFGHIJKL")), 1);
assert_int_equal(AMstrCmp(AMstr("GHIJKL"), AMstr("abcdef")), -1);
assert_int_equal(AMstrCmp(AMstr("abcdef"), AMstr("GHIJKL")), 1);
/* NUL character inclusion */
static char const SRC[] = {'a', 'b', 'c', 'd', 'e', 'f', '\0', 'g', 'h', 'i', 'j', 'k', 'l'};
static AMbyteSpan const NUL_STR = {.src = SRC, .count = 13};
assert_int_equal(AMstrCmp(AMstr("abcdef"), NUL_STR), -1);
assert_int_equal(AMstrCmp(NUL_STR, NUL_STR), 0);
assert_int_equal(AMstrCmp(NUL_STR, AMstr("abcdef")), 1);
/* Empty string */
assert_int_equal(AMstrCmp(AMstr(""), AMstr("abcdefghijkl")), -1);
assert_int_equal(AMstrCmp(AMstr(""), AMstr("")), 0);
assert_int_equal(AMstrCmp(AMstr("abcdefghijkl"), AMstr("")), 1);
/* Invalid string */
assert_int_equal(AMstrCmp(AMstr(NULL), AMstr("abcdefghijkl")), -1);
assert_int_equal(AMstrCmp(AMstr(NULL), AMstr(NULL)), 0);
assert_int_equal(AMstrCmp(AMstr("abcdefghijkl"), AMstr(NULL)), 1);
}
static void test_AMstrdup(void** state) {
static char const SRC[] = {'a', 'b', 'c', '\0', 'd', 'e', 'f', '\0', 'g', 'h', 'i', '\0', 'j', 'k', 'l'};
static AMbyteSpan const NUL_STR = {.src = SRC, .count = 15};
/* Default substitution ("\\0") for NUL */
char* dup = AMstrdup(NUL_STR, NULL);
assert_int_equal(strlen(dup), 18);
assert_string_equal(dup, "abc\\0def\\0ghi\\0jkl");
free(dup);
/* Arbitrary substitution for NUL */
dup = AMstrdup(NUL_STR, ":-O");
assert_int_equal(strlen(dup), 21);
assert_string_equal(dup, "abc:-Odef:-Oghi:-Ojkl");
free(dup);
/* Empty substitution for NUL */
dup = AMstrdup(NUL_STR, "");
assert_int_equal(strlen(dup), 12);
assert_string_equal(dup, "abcdefghijkl");
free(dup);
/* Empty string */
dup = AMstrdup(AMstr(""), NULL);
assert_int_equal(strlen(dup), 0);
assert_string_equal(dup, "");
free(dup);
/* Invalid string */
assert_null(AMstrdup(AMstr(NULL), NULL));
}
int run_byte_span_tests(void) {
const struct CMUnitTest tests[] = {
cmocka_unit_test(test_AMbytes),
cmocka_unit_test(test_AMstr),
cmocka_unit_test(test_AMstrCmp),
cmocka_unit_test(test_AMstrdup),
};
return cmocka_run_group_tests(tests, NULL, NULL);
}

View file

@ -0,0 +1,88 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <stdlib.h>
/* third-party */
#include <automerge-c/utils/enum_string.h>
#include <automerge-c/utils/stack_callback_data.h>
#include <automerge-c/utils/string.h>
#include <cmocka.h>
/* local */
#include "cmocka_utils.h"
/**
* \brief Assert that the given expression is true and report failure in terms
* of a line number within a file.
*
* \param[in] c An expression.
* \param[in] file A file's full path string.
* \param[in] line A line number.
*/
#define assert_true_where(c, file, line) _assert_true(cast_ptr_to_largest_integral_type(c), #c, file, line)
/**
* \brief Assert that the given pointer is non-NULL and report failure in terms
* of a line number within a file.
*
* \param[in] c An expression.
* \param[in] file A file's full path string.
* \param[in] line A line number.
*/
#define assert_non_null_where(c, file, line) assert_true_where(c, file, line)
/**
* \brief Forces the test to fail immediately and quit, printing the reason in
* terms of a line number within a file.
*
* \param[in] msg A message string into which \p str is interpolated.
* \param[in] str An owned string.
* \param[in] file A file's full path string.
* \param[in] line A line number.
*/
#define fail_msg_where(msg, str, file, line) \
do { \
print_error("ERROR: " msg "\n", str); \
_fail(file, line); \
} while (0)
/**
* \brief Forces the test to fail immediately and quit, printing the reason in
* terms of a line number within a file.
*
* \param[in] msg A message string into which \p view.src is interpolated.
* \param[in] view A UTF-8 string view as an `AMbyteSpan` struct.
* \param[in] file A file's full path string.
* \param[in] line A line number.
*/
#define fail_msg_view_where(msg, view, file, line) \
do { \
char* const str = AMstrdup(view, NULL); \
print_error("ERROR: " msg "\n", str); \
free(str); \
_fail(file, line); \
} while (0)
bool cmocka_cb(AMstack** stack, void* data) {
assert_non_null(data);
AMstackCallbackData* const sc_data = (AMstackCallbackData*)data;
assert_non_null_where(stack, sc_data->file, sc_data->line);
assert_non_null_where(*stack, sc_data->file, sc_data->line);
assert_non_null_where((*stack)->result, sc_data->file, sc_data->line);
if (AMresultStatus((*stack)->result) != AM_STATUS_OK) {
fail_msg_view_where("%s", AMresultError((*stack)->result), sc_data->file, sc_data->line);
return false;
}
/* Test that the types of all item values are members of the mask. */
AMitems items = AMresultItems((*stack)->result);
AMitem* item = NULL;
while ((item = AMitemsNext(&items, 1)) != NULL) {
AMvalType const tag = AMitemValType(item);
if (!(tag & sc_data->bitmask)) {
fail_msg_where("Unexpected value type `%s`.", AMvalTypeToString(tag), sc_data->file, sc_data->line);
return false;
}
}
return true;
}

View file

@ -1,22 +1,42 @@
#ifndef CMOCKA_UTILS_H #ifndef TESTS_CMOCKA_UTILS_H
#define CMOCKA_UTILS_H #define TESTS_CMOCKA_UTILS_H
#include <stdlib.h>
#include <string.h> #include <string.h>
/* third-party */ /* third-party */
#include <automerge-c/utils/string.h>
#include <cmocka.h> #include <cmocka.h>
/* local */
#include "base_state.h"
/** /**
* \brief Forces the test to fail immediately and quit, printing the reason. * \brief Forces the test to fail immediately and quit, printing the reason.
* *
* \param[in] view A string view as an `AMbyteSpan` struct. * \param[in] msg A message string into which \p view.src is interpolated.
* \param[in] view A UTF-8 string view as an `AMbyteSpan` struct.
*/ */
#define fail_msg_view(msg, view) do { \ #define fail_msg_view(msg, view) \
char* const c_str = test_calloc(1, view.count + 1); \ do { \
strncpy(c_str, view.src, view.count); \ char* const c_str = AMstrdup(view, NULL); \
print_error(msg, c_str); \ print_error("ERROR: " msg "\n", c_str); \
test_free(c_str); \ free(c_str); \
fail(); \ fail(); \
} while (0) } while (0)
#endif /* CMOCKA_UTILS_H */ /**
* \brief Validates the top result in a stack based upon the parameters
* specified within the given data structure and reports violations
* using cmocka assertions.
*
* \param[in,out] stack A pointer to a pointer to an `AMstack` struct.
* \param[in] data A pointer to an owned `AMpushData` struct.
* \return `true` if the top `AMresult` struct in \p stack is valid, `false`
* otherwise.
* \pre \p stack `!= NULL`.
* \pre \p data `!= NULL`.
*/
bool cmocka_cb(AMstack** stack, void* data);
#endif /* TESTS_CMOCKA_UTILS_H */

View file

@ -0,0 +1,27 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stdlib.h>
/* third-party */
#include <cmocka.h>
/* local */
#include <automerge-c/utils/stack_callback_data.h>
#include "cmocka_utils.h"
#include "doc_state.h"
int setup_doc(void** state) {
DocState* doc_state = test_calloc(1, sizeof(DocState));
setup_base((void**)&doc_state->base_state);
AMitemToDoc(AMstackItem(&doc_state->base_state->stack, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)),
&doc_state->doc);
*state = doc_state;
return 0;
}
int teardown_doc(void** state) {
DocState* doc_state = *state;
teardown_base((void**)&doc_state->base_state);
test_free(doc_state);
return 0;
}

View file

@ -0,0 +1,17 @@
#ifndef TESTS_DOC_STATE_H
#define TESTS_DOC_STATE_H
/* local */
#include <automerge-c/automerge.h>
#include "base_state.h"
typedef struct {
BaseState* base_state;
AMdoc* doc;
} DocState;
int setup_doc(void** state);
int teardown_doc(void** state);
#endif /* TESTS_DOC_STATE_H */

View file

@ -9,12 +9,14 @@
/* local */ /* local */
#include <automerge-c/automerge.h> #include <automerge-c/automerge.h>
#include "group_state.h" #include <automerge-c/utils/stack_callback_data.h>
#include "stack_utils.h" #include "base_state.h"
#include "cmocka_utils.h"
#include "doc_state.h"
#include "str_utils.h" #include "str_utils.h"
typedef struct { typedef struct {
GroupState* group_state; DocState* doc_state;
AMbyteSpan actor_id_str; AMbyteSpan actor_id_str;
uint8_t* actor_id_bytes; uint8_t* actor_id_bytes;
size_t actor_id_size; size_t actor_id_size;
@ -22,7 +24,7 @@ typedef struct {
static int setup(void** state) { static int setup(void** state) {
TestState* test_state = test_calloc(1, sizeof(TestState)); TestState* test_state = test_calloc(1, sizeof(TestState));
group_setup((void**)&test_state->group_state); setup_doc((void**)&test_state->doc_state);
test_state->actor_id_str.src = "000102030405060708090a0b0c0d0e0f"; test_state->actor_id_str.src = "000102030405060708090a0b0c0d0e0f";
test_state->actor_id_str.count = strlen(test_state->actor_id_str.src); test_state->actor_id_str.count = strlen(test_state->actor_id_str.src);
test_state->actor_id_size = test_state->actor_id_str.count / 2; test_state->actor_id_size = test_state->actor_id_str.count / 2;
@ -34,204 +36,195 @@ static int setup(void** state) {
static int teardown(void** state) { static int teardown(void** state) {
TestState* test_state = *state; TestState* test_state = *state;
group_teardown((void**)&test_state->group_state); teardown_doc((void**)&test_state->doc_state);
test_free(test_state->actor_id_bytes); test_free(test_state->actor_id_bytes);
test_free(test_state); test_free(test_state);
return 0; return 0;
} }
static void test_AMkeys_empty() { static void test_AMkeys_empty(void** state) {
AMresultStack* stack = NULL;
AMdoc* const doc = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, cmocka_cb).doc;
AMstrs forward = AMpush(&stack,
AMkeys(doc, AM_ROOT, NULL),
AM_VALUE_STRS,
cmocka_cb).strs;
assert_int_equal(AMstrsSize(&forward), 0);
AMstrs reverse = AMstrsReversed(&forward);
assert_int_equal(AMstrsSize(&reverse), 0);
assert_null(AMstrsNext(&forward, 1).src);
assert_null(AMstrsPrev(&forward, 1).src);
assert_null(AMstrsNext(&reverse, 1).src);
assert_null(AMstrsPrev(&reverse, 1).src);
AMfreeStack(&stack);
}
static void test_AMkeys_list() {
AMresultStack* stack = NULL;
AMdoc* const doc = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, cmocka_cb).doc;
AMobjId const* const list = AMpush(
&stack,
AMmapPutObject(doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),
AM_VALUE_OBJ_ID,
cmocka_cb).obj_id;
AMfree(AMlistPutInt(doc, list, 0, true, 0));
AMfree(AMlistPutInt(doc, list, 1, true, 0));
AMfree(AMlistPutInt(doc, list, 2, true, 0));
AMstrs forward = AMpush(&stack,
AMkeys(doc, list, NULL),
AM_VALUE_STRS,
cmocka_cb).strs;
assert_int_equal(AMstrsSize(&forward), 3);
AMstrs reverse = AMstrsReversed(&forward);
assert_int_equal(AMstrsSize(&reverse), 3);
/* Forward iterator forward. */
AMbyteSpan str = AMstrsNext(&forward, 1);
assert_ptr_equal(strstr(str.src, "2@"), str.src);
str = AMstrsNext(&forward, 1);
assert_ptr_equal(strstr(str.src, "3@"), str.src);
str = AMstrsNext(&forward, 1);
assert_ptr_equal(strstr(str.src, "4@"), str.src);
assert_null(AMstrsNext(&forward, 1).src);
// /* Forward iterator reverse. */
str = AMstrsPrev(&forward, 1);
assert_ptr_equal(strstr(str.src, "4@"), str.src);
str = AMstrsPrev(&forward, 1);
assert_ptr_equal(strstr(str.src, "3@"), str.src);
str = AMstrsPrev(&forward, 1);
assert_ptr_equal(strstr(str.src, "2@"), str.src);
assert_null(AMstrsPrev(&forward, 1).src);
/* Reverse iterator forward. */
str = AMstrsNext(&reverse, 1);
assert_ptr_equal(strstr(str.src, "4@"), str.src);
str = AMstrsNext(&reverse, 1);
assert_ptr_equal(strstr(str.src, "3@"), str.src);
str = AMstrsNext(&reverse, 1);
assert_ptr_equal(strstr(str.src, "2@"), str.src);
assert_null(AMstrsNext(&reverse, 1).src);
/* Reverse iterator reverse. */
str = AMstrsPrev(&reverse, 1);
assert_ptr_equal(strstr(str.src, "2@"), str.src);
str = AMstrsPrev(&reverse, 1);
assert_ptr_equal(strstr(str.src, "3@"), str.src);
str = AMstrsPrev(&reverse, 1);
assert_ptr_equal(strstr(str.src, "4@"), str.src);
assert_null(AMstrsPrev(&reverse, 1).src);
AMfreeStack(&stack);
}
static void test_AMkeys_map() {
AMresultStack* stack = NULL;
AMdoc* const doc = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, cmocka_cb).doc;
AMfree(AMmapPutInt(doc, AM_ROOT, AMstr("one"), 1));
AMfree(AMmapPutInt(doc, AM_ROOT, AMstr("two"), 2));
AMfree(AMmapPutInt(doc, AM_ROOT, AMstr("three"), 3));
AMstrs forward = AMpush(&stack,
AMkeys(doc, AM_ROOT, NULL),
AM_VALUE_STRS,
cmocka_cb).strs;
assert_int_equal(AMstrsSize(&forward), 3);
AMstrs reverse = AMstrsReversed(&forward);
assert_int_equal(AMstrsSize(&reverse), 3);
/* Forward iterator forward. */
AMbyteSpan str = AMstrsNext(&forward, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
str = AMstrsNext(&forward, 1);
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
str = AMstrsNext(&forward, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
assert_null(AMstrsNext(&forward, 1).src);
/* Forward iterator reverse. */
str = AMstrsPrev(&forward, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
str = AMstrsPrev(&forward, 1);
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
str = AMstrsPrev(&forward, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
assert_null(AMstrsPrev(&forward, 1).src);
/* Reverse iterator forward. */
str = AMstrsNext(&reverse, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
str = AMstrsNext(&reverse, 1);
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
str = AMstrsNext(&reverse, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
assert_null(AMstrsNext(&reverse, 1).src);
/* Reverse iterator reverse. */
str = AMstrsPrev(&reverse, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
str = AMstrsPrev(&reverse, 1);
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
str = AMstrsPrev(&reverse, 1);
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
assert_null(AMstrsPrev(&reverse, 1).src);
AMfreeStack(&stack);
}
static void test_AMputActor_bytes(void **state) {
TestState* test_state = *state; TestState* test_state = *state;
AMactorId const* actor_id = AMpush(&test_state->group_state->stack, AMstack** stack_ptr = &test_state->doc_state->base_state->stack;
AMactorIdInitBytes( AMdoc* doc;
test_state->actor_id_bytes, assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc));
test_state->actor_id_size), AMitems forward = AMstackItems(stack_ptr, AMkeys(doc, AM_ROOT, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AM_VALUE_ACTOR_ID, assert_int_equal(AMitemsSize(&forward), 0);
cmocka_cb).actor_id; AMitems reverse = AMitemsReversed(&forward);
AMfree(AMsetActorId(test_state->group_state->doc, actor_id)); assert_int_equal(AMitemsSize(&reverse), 0);
actor_id = AMpush(&test_state->group_state->stack, assert_null(AMitemsNext(&forward, 1));
AMgetActorId(test_state->group_state->doc), assert_null(AMitemsPrev(&forward, 1));
AM_VALUE_ACTOR_ID, assert_null(AMitemsNext(&reverse, 1));
cmocka_cb).actor_id; assert_null(AMitemsPrev(&reverse, 1));
}
static void test_AMkeys_list(void** state) {
TestState* test_state = *state;
AMstack** stack_ptr = &test_state->doc_state->base_state->stack;
AMdoc* doc;
assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc));
AMobjId const* const list =
AMitemObjId(AMstackItem(stack_ptr, AMmapPutObject(doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), cmocka_cb,
AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
AMstackItem(NULL, AMlistPutInt(doc, list, 0, true, 0), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMstackItem(NULL, AMlistPutInt(doc, list, 1, true, 0), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMstackItem(NULL, AMlistPutInt(doc, list, 2, true, 0), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMitems forward = AMstackItems(stack_ptr, AMkeys(doc, list, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
assert_int_equal(AMitemsSize(&forward), 3);
AMitems reverse = AMitemsReversed(&forward);
assert_int_equal(AMitemsSize(&reverse), 3);
/* Forward iterator forward. */
AMbyteSpan str;
assert_true(AMitemToStr(AMitemsNext(&forward, 1), &str));
assert_ptr_equal(strstr(str.src, "2@"), str.src);
assert_true(AMitemToStr(AMitemsNext(&forward, 1), &str));
assert_ptr_equal(strstr(str.src, "3@"), str.src);
assert_true(AMitemToStr(AMitemsNext(&forward, 1), &str));
assert_ptr_equal(strstr(str.src, "4@"), str.src);
assert_null(AMitemsNext(&forward, 1));
// /* Forward iterator reverse. */
assert_true(AMitemToStr(AMitemsPrev(&forward, 1), &str));
assert_ptr_equal(strstr(str.src, "4@"), str.src);
assert_true(AMitemToStr(AMitemsPrev(&forward, 1), &str));
assert_ptr_equal(strstr(str.src, "3@"), str.src);
assert_true(AMitemToStr(AMitemsPrev(&forward, 1), &str));
assert_ptr_equal(strstr(str.src, "2@"), str.src);
assert_null(AMitemsPrev(&forward, 1));
/* Reverse iterator forward. */
assert_true(AMitemToStr(AMitemsNext(&reverse, 1), &str));
assert_ptr_equal(strstr(str.src, "4@"), str.src);
assert_true(AMitemToStr(AMitemsNext(&reverse, 1), &str));
assert_ptr_equal(strstr(str.src, "3@"), str.src);
assert_true(AMitemToStr(AMitemsNext(&reverse, 1), &str));
assert_ptr_equal(strstr(str.src, "2@"), str.src);
assert_null(AMitemsNext(&reverse, 1));
/* Reverse iterator reverse. */
assert_true(AMitemToStr(AMitemsPrev(&reverse, 1), &str));
assert_ptr_equal(strstr(str.src, "2@"), str.src);
assert_true(AMitemToStr(AMitemsPrev(&reverse, 1), &str));
assert_ptr_equal(strstr(str.src, "3@"), str.src);
assert_true(AMitemToStr(AMitemsPrev(&reverse, 1), &str));
assert_ptr_equal(strstr(str.src, "4@"), str.src);
assert_null(AMitemsPrev(&reverse, 1));
}
static void test_AMkeys_map(void** state) {
TestState* test_state = *state;
AMstack** stack_ptr = &test_state->doc_state->base_state->stack;
AMdoc* doc;
assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc));
AMstackItem(NULL, AMmapPutInt(doc, AM_ROOT, AMstr("one"), 1), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMstackItem(NULL, AMmapPutInt(doc, AM_ROOT, AMstr("two"), 2), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMstackItem(NULL, AMmapPutInt(doc, AM_ROOT, AMstr("three"), 3), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMitems forward = AMstackItems(stack_ptr, AMkeys(doc, AM_ROOT, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
assert_int_equal(AMitemsSize(&forward), 3);
AMitems reverse = AMitemsReversed(&forward);
assert_int_equal(AMitemsSize(&reverse), 3);
/* Forward iterator forward. */
AMbyteSpan str;
assert_true(AMitemToStr(AMitemsNext(&forward, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
assert_true(AMitemToStr(AMitemsNext(&forward, 1), &str));
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
assert_true(AMitemToStr(AMitemsNext(&forward, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
assert_null(AMitemsNext(&forward, 1));
/* Forward iterator reverse. */
assert_true(AMitemToStr(AMitemsPrev(&forward, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
assert_true(AMitemToStr(AMitemsPrev(&forward, 1), &str));
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
assert_true(AMitemToStr(AMitemsPrev(&forward, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
assert_null(AMitemsPrev(&forward, 1));
/* Reverse iterator forward. */
assert_true(AMitemToStr(AMitemsNext(&reverse, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
assert_true(AMitemToStr(AMitemsNext(&reverse, 1), &str));
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
assert_true(AMitemToStr(AMitemsNext(&reverse, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
assert_null(AMitemsNext(&reverse, 1));
/* Reverse iterator reverse. */
assert_true(AMitemToStr(AMitemsPrev(&reverse, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "one", str.count);
assert_true(AMitemToStr(AMitemsPrev(&reverse, 1), &str));
assert_int_equal(str.count, 5);
assert_memory_equal(str.src, "three", str.count);
assert_true(AMitemToStr(AMitemsPrev(&reverse, 1), &str));
assert_int_equal(str.count, 3);
assert_memory_equal(str.src, "two", str.count);
assert_null(AMitemsPrev(&reverse, 1));
}
static void test_AMputActor_bytes(void** state) {
TestState* test_state = *state;
AMstack** stack_ptr = &test_state->doc_state->base_state->stack;
AMactorId const* actor_id;
assert_true(AMitemToActorId(
AMstackItem(stack_ptr, AMactorIdFromBytes(test_state->actor_id_bytes, test_state->actor_id_size), cmocka_cb,
AMexpect(AM_VAL_TYPE_ACTOR_ID)),
&actor_id));
AMstackItem(NULL, AMsetActorId(test_state->doc_state->doc, actor_id), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
assert_true(AMitemToActorId(
AMstackItem(stack_ptr, AMgetActorId(test_state->doc_state->doc), cmocka_cb, AMexpect(AM_VAL_TYPE_ACTOR_ID)),
&actor_id));
AMbyteSpan const bytes = AMactorIdBytes(actor_id); AMbyteSpan const bytes = AMactorIdBytes(actor_id);
assert_int_equal(bytes.count, test_state->actor_id_size); assert_int_equal(bytes.count, test_state->actor_id_size);
assert_memory_equal(bytes.src, test_state->actor_id_bytes, bytes.count); assert_memory_equal(bytes.src, test_state->actor_id_bytes, bytes.count);
} }
static void test_AMputActor_str(void **state) { static void test_AMputActor_str(void** state) {
TestState* test_state = *state; TestState* test_state = *state;
AMactorId const* actor_id = AMpush(&test_state->group_state->stack, AMstack** stack_ptr = &test_state->doc_state->base_state->stack;
AMactorIdInitStr(test_state->actor_id_str), AMactorId const* actor_id;
AM_VALUE_ACTOR_ID, assert_true(AMitemToActorId(
cmocka_cb).actor_id; AMstackItem(stack_ptr, AMactorIdFromStr(test_state->actor_id_str), cmocka_cb, AMexpect(AM_VAL_TYPE_ACTOR_ID)),
AMfree(AMsetActorId(test_state->group_state->doc, actor_id)); &actor_id));
actor_id = AMpush(&test_state->group_state->stack, AMstackItem(NULL, AMsetActorId(test_state->doc_state->doc, actor_id), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMgetActorId(test_state->group_state->doc), assert_true(AMitemToActorId(
AM_VALUE_ACTOR_ID, AMstackItem(stack_ptr, AMgetActorId(test_state->doc_state->doc), cmocka_cb, AMexpect(AM_VAL_TYPE_ACTOR_ID)),
cmocka_cb).actor_id; &actor_id));
AMbyteSpan const str = AMactorIdStr(actor_id); AMbyteSpan const str = AMactorIdStr(actor_id);
assert_int_equal(str.count, test_state->actor_id_str.count); assert_int_equal(str.count, test_state->actor_id_str.count);
assert_memory_equal(str.src, test_state->actor_id_str.src, str.count); assert_memory_equal(str.src, test_state->actor_id_str.src, str.count);
} }
static void test_AMspliceText() { static void test_AMspliceText(void** state) {
AMresultStack* stack = NULL; TestState* test_state = *state;
AMdoc* const doc = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, cmocka_cb).doc; AMstack** stack_ptr = &test_state->doc_state->base_state->stack;
AMobjId const* const text = AMpush(&stack, AMdoc* doc;
AMmapPutObject(doc, AM_ROOT, AMstr("text"), AM_OBJ_TYPE_TEXT), assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc));
AM_VALUE_OBJ_ID, AMobjId const* const text =
cmocka_cb).obj_id; AMitemObjId(AMstackItem(stack_ptr, AMmapPutObject(doc, AM_ROOT, AMstr("text"), AM_OBJ_TYPE_TEXT), cmocka_cb,
AMfree(AMspliceText(doc, text, 0, 0, AMstr("one + "))); AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
AMfree(AMspliceText(doc, text, 4, 2, AMstr("two = "))); AMstackItem(NULL, AMspliceText(doc, text, 0, 0, AMstr("one + ")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMspliceText(doc, text, 8, 2, AMstr("three"))); AMstackItem(NULL, AMspliceText(doc, text, 4, 2, AMstr("two = ")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMbyteSpan const str = AMpush(&stack, AMstackItem(NULL, AMspliceText(doc, text, 8, 2, AMstr("three")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMtext(doc, text, NULL), AMbyteSpan str;
AM_VALUE_STR, assert_true(
cmocka_cb).str; AMitemToStr(AMstackItem(stack_ptr, AMtext(doc, text, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR)), &str));
static char const* const STR_VALUE = "one two three"; assert_int_equal(str.count, strlen("one two three"));
assert_int_equal(str.count, strlen(STR_VALUE)); assert_memory_equal(str.src, "one two three", str.count);
assert_memory_equal(str.src, STR_VALUE, str.count);
AMfreeStack(&stack);
} }
int run_doc_tests(void) { int run_doc_tests(void) {
const struct CMUnitTest tests[] = { const struct CMUnitTest tests[] = {
cmocka_unit_test(test_AMkeys_empty), cmocka_unit_test_setup_teardown(test_AMkeys_empty, setup, teardown),
cmocka_unit_test(test_AMkeys_list), cmocka_unit_test_setup_teardown(test_AMkeys_list, setup, teardown),
cmocka_unit_test(test_AMkeys_map), cmocka_unit_test_setup_teardown(test_AMkeys_map, setup, teardown),
cmocka_unit_test_setup_teardown(test_AMputActor_bytes, setup, teardown), cmocka_unit_test_setup_teardown(test_AMputActor_bytes, setup, teardown),
cmocka_unit_test_setup_teardown(test_AMputActor_str, setup, teardown), cmocka_unit_test_setup_teardown(test_AMputActor_str, setup, teardown),
cmocka_unit_test(test_AMspliceText), cmocka_unit_test_setup_teardown(test_AMspliceText, setup, teardown),
}; };
return cmocka_run_group_tests(tests, NULL, NULL); return cmocka_run_group_tests(tests, NULL, NULL);

View file

@ -0,0 +1,148 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
/* third-party */
#include <cmocka.h>
/* local */
#include <automerge-c/automerge.h>
#include <automerge-c/utils/enum_string.h>
#define assert_to_string(function, tag) assert_string_equal(function(tag), #tag)
#define assert_from_string(function, type, tag) \
do { \
type out; \
assert_true(function(&out, #tag)); \
assert_int_equal(out, tag); \
} while (0)
static void test_AMidxTypeToString(void** state) {
assert_to_string(AMidxTypeToString, AM_IDX_TYPE_DEFAULT);
assert_to_string(AMidxTypeToString, AM_IDX_TYPE_KEY);
assert_to_string(AMidxTypeToString, AM_IDX_TYPE_POS);
/* Zero tag */
assert_string_equal(AMidxTypeToString(0), "AM_IDX_TYPE_DEFAULT");
/* Invalid tag */
assert_string_equal(AMidxTypeToString(-1), "???");
}
static void test_AMidxTypeFromString(void** state) {
assert_from_string(AMidxTypeFromString, AMidxType, AM_IDX_TYPE_DEFAULT);
assert_from_string(AMidxTypeFromString, AMidxType, AM_IDX_TYPE_KEY);
assert_from_string(AMidxTypeFromString, AMidxType, AM_IDX_TYPE_POS);
/* Invalid tag */
AMidxType out = -1;
assert_false(AMidxTypeFromString(&out, "???"));
assert_int_equal(out, (AMidxType)-1);
}
static void test_AMobjTypeToString(void** state) {
assert_to_string(AMobjTypeToString, AM_OBJ_TYPE_DEFAULT);
assert_to_string(AMobjTypeToString, AM_OBJ_TYPE_LIST);
assert_to_string(AMobjTypeToString, AM_OBJ_TYPE_MAP);
assert_to_string(AMobjTypeToString, AM_OBJ_TYPE_TEXT);
/* Zero tag */
assert_string_equal(AMobjTypeToString(0), "AM_OBJ_TYPE_DEFAULT");
/* Invalid tag */
assert_string_equal(AMobjTypeToString(-1), "???");
}
static void test_AMobjTypeFromString(void** state) {
assert_from_string(AMobjTypeFromString, AMobjType, AM_OBJ_TYPE_DEFAULT);
assert_from_string(AMobjTypeFromString, AMobjType, AM_OBJ_TYPE_LIST);
assert_from_string(AMobjTypeFromString, AMobjType, AM_OBJ_TYPE_MAP);
assert_from_string(AMobjTypeFromString, AMobjType, AM_OBJ_TYPE_TEXT);
/* Invalid tag */
AMobjType out = -1;
assert_false(AMobjTypeFromString(&out, "???"));
assert_int_equal(out, (AMobjType)-1);
}
static void test_AMstatusToString(void** state) {
assert_to_string(AMstatusToString, AM_STATUS_ERROR);
assert_to_string(AMstatusToString, AM_STATUS_INVALID_RESULT);
assert_to_string(AMstatusToString, AM_STATUS_OK);
/* Zero tag */
assert_string_equal(AMstatusToString(0), "AM_STATUS_OK");
/* Invalid tag */
assert_string_equal(AMstatusToString(-1), "???");
}
static void test_AMstatusFromString(void** state) {
assert_from_string(AMstatusFromString, AMstatus, AM_STATUS_ERROR);
assert_from_string(AMstatusFromString, AMstatus, AM_STATUS_INVALID_RESULT);
assert_from_string(AMstatusFromString, AMstatus, AM_STATUS_OK);
/* Invalid tag */
AMstatus out = -1;
assert_false(AMstatusFromString(&out, "???"));
assert_int_equal(out, (AMstatus)-1);
}
static void test_AMvalTypeToString(void** state) {
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_ACTOR_ID);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_BOOL);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_BYTES);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_CHANGE);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_CHANGE_HASH);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_COUNTER);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_DEFAULT);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_DOC);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_F64);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_INT);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_NULL);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_OBJ_TYPE);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_STR);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_SYNC_HAVE);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_SYNC_MESSAGE);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_SYNC_STATE);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_TIMESTAMP);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_UINT);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_UNKNOWN);
assert_to_string(AMvalTypeToString, AM_VAL_TYPE_VOID);
/* Zero tag */
assert_string_equal(AMvalTypeToString(0), "AM_VAL_TYPE_DEFAULT");
/* Invalid tag */
assert_string_equal(AMvalTypeToString(-1), "???");
}
static void test_AMvalTypeFromString(void** state) {
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_ACTOR_ID);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_BOOL);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_BYTES);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_CHANGE);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_CHANGE_HASH);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_COUNTER);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_DEFAULT);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_DOC);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_F64);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_INT);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_NULL);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_OBJ_TYPE);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_STR);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_SYNC_HAVE);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_SYNC_MESSAGE);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_SYNC_STATE);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_TIMESTAMP);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_UINT);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_UNKNOWN);
assert_from_string(AMvalTypeFromString, AMvalType, AM_VAL_TYPE_VOID);
/* Invalid tag */
AMvalType out = -1;
assert_false(AMvalTypeFromString(&out, "???"));
assert_int_equal(out, (AMvalType)-1);
}
int run_enum_string_tests(void) {
const struct CMUnitTest tests[] = {
cmocka_unit_test(test_AMidxTypeToString), cmocka_unit_test(test_AMidxTypeFromString),
cmocka_unit_test(test_AMobjTypeToString), cmocka_unit_test(test_AMobjTypeFromString),
cmocka_unit_test(test_AMstatusToString), cmocka_unit_test(test_AMstatusFromString),
cmocka_unit_test(test_AMvalTypeToString), cmocka_unit_test(test_AMvalTypeFromString),
};
return cmocka_run_group_tests(tests, NULL, NULL);
}

View file

@ -1,27 +0,0 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stdlib.h>
/* third-party */
#include <cmocka.h>
/* local */
#include "group_state.h"
#include "stack_utils.h"
int group_setup(void** state) {
GroupState* group_state = test_calloc(1, sizeof(GroupState));
group_state->doc = AMpush(&group_state->stack,
AMcreate(NULL),
AM_VALUE_DOC,
cmocka_cb).doc;
*state = group_state;
return 0;
}
int group_teardown(void** state) {
GroupState* group_state = *state;
AMfreeStack(&group_state->stack);
test_free(group_state);
return 0;
}

View file

@ -1,16 +0,0 @@
#ifndef GROUP_STATE_H
#define GROUP_STATE_H
/* local */
#include <automerge-c/automerge.h>
typedef struct {
AMresultStack* stack;
AMdoc* doc;
} GroupState;
int group_setup(void** state);
int group_teardown(void** state);
#endif /* GROUP_STATE_H */

View file

@ -0,0 +1,94 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
/* third-party */
#include <cmocka.h>
/* local */
#include <automerge-c/automerge.h>
#include <automerge-c/utils/stack_callback_data.h>
#include "cmocka_utils.h"
#include "doc_state.h"
static void test_AMitemResult(void** state) {
enum { ITEM_COUNT = 1000 };
DocState* doc_state = *state;
AMstack** stack_ptr = &doc_state->base_state->stack;
/* Append the strings to a list so that they'll be in numerical order. */
AMobjId const* const list =
AMitemObjId(AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),
cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
for (size_t pos = 0; pos != ITEM_COUNT; ++pos) {
size_t const count = snprintf(NULL, 0, "%zu", pos);
char* const src = test_calloc(count + 1, sizeof(char));
assert_int_equal(sprintf(src, "%zu", pos), count);
AMstackItem(NULL, AMlistPutStr(doc_state->doc, list, pos, true, AMbytes(src, count)), cmocka_cb,
AMexpect(AM_VAL_TYPE_VOID));
test_free(src);
}
/* Get an item iterator. */
AMitems items = AMstackItems(stack_ptr, AMlistRange(doc_state->doc, list, 0, SIZE_MAX, NULL), cmocka_cb,
AMexpect(AM_VAL_TYPE_STR));
/* Get the item iterator's result so that it can be freed later. */
AMresult const* const items_result = (*stack_ptr)->result;
/* Iterate over all of the items and copy their pointers into an array. */
AMitem* item_ptrs[ITEM_COUNT] = {NULL};
AMitem* item = NULL;
for (size_t pos = 0; (item = AMitemsNext(&items, 1)) != NULL; ++pos) {
/* The item's reference count should be 1. */
assert_int_equal(AMitemRefCount(item), 1);
if (pos & 1) {
/* Create a redundant result for an odd item. */
AMitem* const new_item = AMstackItem(stack_ptr, AMitemResult(item), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
/* The item's old and new pointers will never match. */
assert_ptr_not_equal(new_item, item);
/* The item's reference count will have been incremented. */
assert_int_equal(AMitemRefCount(item), 2);
assert_int_equal(AMitemRefCount(new_item), 2);
/* The item's old and new indices should match. */
assert_int_equal(AMitemIdxType(item), AMitemIdxType(new_item));
assert_int_equal(AMitemIdxType(item), AM_IDX_TYPE_POS);
size_t pos, new_pos;
assert_true(AMitemPos(item, &pos));
assert_true(AMitemPos(new_item, &new_pos));
assert_int_equal(pos, new_pos);
/* The item's old and new object IDs should match. */
AMobjId const* const obj_id = AMitemObjId(item);
AMobjId const* const new_obj_id = AMitemObjId(new_item);
assert_true(AMobjIdEqual(obj_id, new_obj_id));
/* The item's old and new value types should match. */
assert_int_equal(AMitemValType(item), AMitemValType(new_item));
/* The item's old and new string values should match. */
AMbyteSpan str;
assert_true(AMitemToStr(item, &str));
AMbyteSpan new_str;
assert_true(AMitemToStr(new_item, &new_str));
assert_int_equal(str.count, new_str.count);
assert_memory_equal(str.src, new_str.src, new_str.count);
/* The item's old and new object IDs are one and the same. */
assert_ptr_equal(obj_id, new_obj_id);
/* The item's old and new string values are one and the same. */
assert_ptr_equal(str.src, new_str.src);
/* Save the item's new pointer. */
item_ptrs[pos] = new_item;
}
}
/* Free the item iterator's result. */
AMresultFree(AMstackPop(stack_ptr, items_result));
/* An odd item's reference count should be 1 again. */
for (size_t pos = 1; pos < ITEM_COUNT; pos += 2) {
assert_int_equal(AMitemRefCount(item_ptrs[pos]), 1);
}
}
int run_item_tests(void) {
const struct CMUnitTest tests[] = {
cmocka_unit_test(test_AMitemResult),
};
return cmocka_run_group_tests(tests, setup_doc, teardown_doc);
}

View file

@ -11,367 +11,417 @@
/* local */ /* local */
#include <automerge-c/automerge.h> #include <automerge-c/automerge.h>
#include <automerge-c/utils/stack_callback_data.h>
#include "base_state.h"
#include "cmocka_utils.h" #include "cmocka_utils.h"
#include "group_state.h" #include "doc_state.h"
#include "macro_utils.h" #include "macro_utils.h"
#include "stack_utils.h"
static void test_AMlistIncrement(void** state) { static void test_AMlistIncrement(void** state) {
GroupState* group_state = *state; DocState* doc_state = *state;
AMobjId const* const list = AMpush( AMstack** stack_ptr = &doc_state->base_state->stack;
&group_state->stack, AMobjId const* const list =
AMmapPutObject(group_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), AMitemObjId(AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),
AM_VALUE_OBJ_ID, cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
cmocka_cb).obj_id; AMstackItem(NULL, AMlistPutCounter(doc_state->doc, list, 0, true, 0), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutCounter(group_state->doc, list, 0, true, 0)); int64_t counter;
assert_int_equal(AMpush(&group_state->stack, assert_true(AMitemToCounter(
AMlistGet(group_state->doc, list, 0, NULL), AMstackItem(stack_ptr, AMlistGet(doc_state->doc, list, 0, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_COUNTER)),
AM_VALUE_COUNTER, &counter));
cmocka_cb).counter, 0); assert_int_equal(counter, 0);
AMfree(AMpop(&group_state->stack)); AMresultFree(AMstackPop(stack_ptr, NULL));
AMfree(AMlistIncrement(group_state->doc, list, 0, 3)); AMstackItem(NULL, AMlistIncrement(doc_state->doc, list, 0, 3), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
assert_int_equal(AMpush(&group_state->stack, assert_true(AMitemToCounter(
AMlistGet(group_state->doc, list, 0, NULL), AMstackItem(stack_ptr, AMlistGet(doc_state->doc, list, 0, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_COUNTER)),
AM_VALUE_COUNTER, &counter));
cmocka_cb).counter, 3); assert_int_equal(counter, 3);
AMfree(AMpop(&group_state->stack)); AMresultFree(AMstackPop(stack_ptr, NULL));
} }
#define test_AMlistPut(suffix, mode) test_AMlistPut ## suffix ## _ ## mode #define test_AMlistPut(suffix, mode) test_AMlistPut##suffix##_##mode
#define static_void_test_AMlistPut(suffix, mode, member, scalar_value) \ #define static_void_test_AMlistPut(suffix, mode, type, scalar_value) \
static void test_AMlistPut ## suffix ## _ ## mode(void **state) { \ static void test_AMlistPut##suffix##_##mode(void** state) { \
GroupState* group_state = *state; \ DocState* doc_state = *state; \
AMobjId const* const list = AMpush( \ AMstack** stack_ptr = &doc_state->base_state->stack; \
&group_state->stack, \ AMobjId const* const list = AMitemObjId( \
AMmapPutObject(group_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),\ AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), \
AM_VALUE_OBJ_ID, \ cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE))); \
cmocka_cb).obj_id; \ AMstackItem(NULL, AMlistPut##suffix(doc_state->doc, list, 0, !strcmp(#mode, "insert"), scalar_value), \
AMfree(AMlistPut ## suffix(group_state->doc, \ cmocka_cb, AMexpect(AM_VAL_TYPE_VOID)); \
list, \ type value; \
0, \ assert_true(AMitemTo##suffix(AMstackItem(stack_ptr, AMlistGet(doc_state->doc, list, 0, NULL), cmocka_cb, \
!strcmp(#mode, "insert"), \ AMexpect(suffix_to_val_type(#suffix))), \
scalar_value)); \ &value)); \
assert_true(AMpush( \ assert_true(value == scalar_value); \
&group_state->stack, \ AMresultFree(AMstackPop(stack_ptr, NULL)); \
AMlistGet(group_state->doc, list, 0, NULL), \ }
AMvalue_discriminant(#suffix), \
cmocka_cb).member == scalar_value); \
AMfree(AMpop(&group_state->stack)); \
}
#define test_AMlistPutBytes(mode) test_AMlistPutBytes ## _ ## mode #define test_AMlistPutBytes(mode) test_AMlistPutBytes##_##mode
#define static_void_test_AMlistPutBytes(mode, bytes_value) \ #define static_void_test_AMlistPutBytes(mode, bytes_value) \
static void test_AMlistPutBytes_ ## mode(void **state) { \ static void test_AMlistPutBytes_##mode(void** state) { \
static size_t const BYTES_SIZE = sizeof(bytes_value) / sizeof(uint8_t); \ static size_t const BYTES_SIZE = sizeof(bytes_value) / sizeof(uint8_t); \
\ \
GroupState* group_state = *state; \ DocState* doc_state = *state; \
AMobjId const* const list = AMpush( \ AMstack** stack_ptr = &doc_state->base_state->stack; \
&group_state->stack, \ AMobjId const* const list = AMitemObjId( \
AMmapPutObject(group_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),\ AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), \
AM_VALUE_OBJ_ID, \ cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE))); \
cmocka_cb).obj_id; \ AMstackItem( \
AMfree(AMlistPutBytes(group_state->doc, \ NULL, AMlistPutBytes(doc_state->doc, list, 0, !strcmp(#mode, "insert"), AMbytes(bytes_value, BYTES_SIZE)), \
list, \ cmocka_cb, AMexpect(AM_VAL_TYPE_VOID)); \
0, \ AMbyteSpan bytes; \
!strcmp(#mode, "insert"), \ assert_true(AMitemToBytes( \
AMbytes(bytes_value, BYTES_SIZE))); \ AMstackItem(stack_ptr, AMlistGet(doc_state->doc, list, 0, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_BYTES)), \
AMbyteSpan const bytes = AMpush( \ &bytes)); \
&group_state->stack, \ assert_int_equal(bytes.count, BYTES_SIZE); \
AMlistGet(group_state->doc, list, 0, NULL), \ assert_memory_equal(bytes.src, bytes_value, BYTES_SIZE); \
AM_VALUE_BYTES, \ AMresultFree(AMstackPop(stack_ptr, NULL)); \
cmocka_cb).bytes; \ }
assert_int_equal(bytes.count, BYTES_SIZE); \
assert_memory_equal(bytes.src, bytes_value, BYTES_SIZE); \
AMfree(AMpop(&group_state->stack)); \
}
#define test_AMlistPutNull(mode) test_AMlistPutNull_ ## mode #define test_AMlistPutNull(mode) test_AMlistPutNull_##mode
#define static_void_test_AMlistPutNull(mode) \ #define static_void_test_AMlistPutNull(mode) \
static void test_AMlistPutNull_ ## mode(void **state) { \ static void test_AMlistPutNull_##mode(void** state) { \
GroupState* group_state = *state; \ DocState* doc_state = *state; \
AMobjId const* const list = AMpush( \ AMstack** stack_ptr = &doc_state->base_state->stack; \
&group_state->stack, \ AMobjId const* const list = AMitemObjId( \
AMmapPutObject(group_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),\ AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), \
AM_VALUE_OBJ_ID, \ cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE))); \
cmocka_cb).obj_id; \ AMstackItem(NULL, AMlistPutNull(doc_state->doc, list, 0, !strcmp(#mode, "insert")), cmocka_cb, \
AMfree(AMlistPutNull(group_state->doc, \ AMexpect(AM_VAL_TYPE_VOID)); \
list, \ AMresult* result = AMstackResult(stack_ptr, AMlistGet(doc_state->doc, list, 0, NULL), NULL, NULL); \
0, \ if (AMresultStatus(result) != AM_STATUS_OK) { \
!strcmp(#mode, "insert"))); \ fail_msg_view("%s", AMresultError(result)); \
AMresult* const result = AMlistGet(group_state->doc, list, 0, NULL); \ } \
if (AMresultStatus(result) != AM_STATUS_OK) { \ assert_int_equal(AMresultSize(result), 1); \
fail_msg_view("%s", AMerrorMessage(result)); \ assert_int_equal(AMitemValType(AMresultItem(result)), AM_VAL_TYPE_NULL); \
} \ AMresultFree(AMstackPop(stack_ptr, NULL)); \
assert_int_equal(AMresultSize(result), 1); \ }
assert_int_equal(AMresultValue(result).tag, AM_VALUE_NULL); \
AMfree(result); \
}
#define test_AMlistPutObject(label, mode) test_AMlistPutObject_ ## label ## _ ## mode #define test_AMlistPutObject(label, mode) test_AMlistPutObject_##label##_##mode
#define static_void_test_AMlistPutObject(label, mode) \ #define static_void_test_AMlistPutObject(label, mode) \
static void test_AMlistPutObject_ ## label ## _ ## mode(void **state) { \ static void test_AMlistPutObject_##label##_##mode(void** state) { \
GroupState* group_state = *state; \ DocState* doc_state = *state; \
AMobjId const* const list = AMpush( \ AMstack** stack_ptr = &doc_state->base_state->stack; \
&group_state->stack, \ AMobjId const* const list = AMitemObjId( \
AMmapPutObject(group_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),\ AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), \
AM_VALUE_OBJ_ID, \ cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE))); \
cmocka_cb).obj_id; \ AMobjType const obj_type = suffix_to_obj_type(#label); \
AMobjType const obj_type = AMobjType_tag(#label); \ AMobjId const* const obj_id = AMitemObjId( \
if (obj_type != AM_OBJ_TYPE_VOID) { \ AMstackItem(stack_ptr, AMlistPutObject(doc_state->doc, list, 0, !strcmp(#mode, "insert"), obj_type), \
AMobjId const* const obj_id = AMpush( \ cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE))); \
&group_state->stack, \ assert_non_null(obj_id); \
AMlistPutObject(group_state->doc, \ assert_int_equal(AMobjObjType(doc_state->doc, obj_id), obj_type); \
list, \ assert_int_equal(AMobjSize(doc_state->doc, obj_id, NULL), 0); \
0, \ AMresultFree(AMstackPop(stack_ptr, NULL)); \
!strcmp(#mode, "insert"), \ }
obj_type), \
AM_VALUE_OBJ_ID, \
cmocka_cb).obj_id; \
assert_non_null(obj_id); \
assert_int_equal(AMobjObjType(group_state->doc, obj_id), obj_type); \
assert_int_equal(AMobjSize(group_state->doc, obj_id, NULL), 0); \
} \
else { \
AMpush(&group_state->stack, \
AMlistPutObject(group_state->doc, \
list, \
0, \
!strcmp(#mode, "insert"), \
obj_type), \
AM_VALUE_VOID, \
NULL); \
assert_int_not_equal(AMresultStatus(group_state->stack->result), \
AM_STATUS_OK); \
} \
AMfree(AMpop(&group_state->stack)); \
}
#define test_AMlistPutStr(mode) test_AMlistPutStr ## _ ## mode #define test_AMlistPutStr(mode) test_AMlistPutStr##_##mode
#define static_void_test_AMlistPutStr(mode, str_value) \ #define static_void_test_AMlistPutStr(mode, str_value) \
static void test_AMlistPutStr_ ## mode(void **state) { \ static void test_AMlistPutStr_##mode(void** state) { \
GroupState* group_state = *state; \ DocState* doc_state = *state; \
AMobjId const* const list = AMpush( \ AMstack** stack_ptr = &doc_state->base_state->stack; \
&group_state->stack, \ AMobjId const* const list = AMitemObjId( \
AMmapPutObject(group_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),\ AMstackItem(stack_ptr, AMmapPutObject(doc_state->doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), \
AM_VALUE_OBJ_ID, \ cmocka_cb, AMexpect(AM_VAL_TYPE_OBJ_TYPE))); \
cmocka_cb).obj_id; \ AMstackItem(NULL, AMlistPutStr(doc_state->doc, list, 0, !strcmp(#mode, "insert"), AMstr(str_value)), \
AMfree(AMlistPutStr(group_state->doc, \ cmocka_cb, AMexpect(AM_VAL_TYPE_VOID)); \
list, \ AMbyteSpan str; \
0, \ assert_true(AMitemToStr( \
!strcmp(#mode, "insert"), \ AMstackItem(stack_ptr, AMlistGet(doc_state->doc, list, 0, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR)), \
AMstr(str_value))); \ &str)); \
AMbyteSpan const str = AMpush( \ assert_int_equal(str.count, strlen(str_value)); \
&group_state->stack, \ assert_memory_equal(str.src, str_value, str.count); \
AMlistGet(group_state->doc, list, 0, NULL), \ AMresultFree(AMstackPop(stack_ptr, NULL)); \
AM_VALUE_STR, \ }
cmocka_cb).str; \
assert_int_equal(str.count, strlen(str_value)); \
assert_memory_equal(str.src, str_value, str.count); \
AMfree(AMpop(&group_state->stack)); \
}
static_void_test_AMlistPut(Bool, insert, boolean, true) static_void_test_AMlistPut(Bool, insert, bool, true);
static_void_test_AMlistPut(Bool, update, boolean, true) static_void_test_AMlistPut(Bool, update, bool, true);
static uint8_t const BYTES_VALUE[] = {INT8_MIN, INT8_MAX / 2, INT8_MAX}; static uint8_t const BYTES_VALUE[] = {INT8_MIN, INT8_MAX / 2, INT8_MAX};
static_void_test_AMlistPutBytes(insert, BYTES_VALUE) static_void_test_AMlistPutBytes(insert, BYTES_VALUE);
static_void_test_AMlistPutBytes(update, BYTES_VALUE) static_void_test_AMlistPutBytes(update, BYTES_VALUE);
static_void_test_AMlistPut(Counter, insert, counter, INT64_MAX) static_void_test_AMlistPut(Counter, insert, int64_t, INT64_MAX);
static_void_test_AMlistPut(Counter, update, counter, INT64_MAX) static_void_test_AMlistPut(Counter, update, int64_t, INT64_MAX);
static_void_test_AMlistPut(F64, insert, f64, DBL_MAX) static_void_test_AMlistPut(F64, insert, double, DBL_MAX);
static_void_test_AMlistPut(F64, update, f64, DBL_MAX) static_void_test_AMlistPut(F64, update, double, DBL_MAX);
static_void_test_AMlistPut(Int, insert, int_, INT64_MAX) static_void_test_AMlistPut(Int, insert, int64_t, INT64_MAX);
static_void_test_AMlistPut(Int, update, int_, INT64_MAX) static_void_test_AMlistPut(Int, update, int64_t, INT64_MAX);
static_void_test_AMlistPutNull(insert) static_void_test_AMlistPutNull(insert);
static_void_test_AMlistPutNull(update) static_void_test_AMlistPutNull(update);
static_void_test_AMlistPutObject(List, insert) static_void_test_AMlistPutObject(List, insert);
static_void_test_AMlistPutObject(List, update) static_void_test_AMlistPutObject(List, update);
static_void_test_AMlistPutObject(Map, insert) static_void_test_AMlistPutObject(Map, insert);
static_void_test_AMlistPutObject(Map, update) static_void_test_AMlistPutObject(Map, update);
static_void_test_AMlistPutObject(Text, insert) static_void_test_AMlistPutObject(Text, insert);
static_void_test_AMlistPutObject(Text, update) static_void_test_AMlistPutObject(Text, update);
static_void_test_AMlistPutObject(Void, insert) static_void_test_AMlistPutStr(insert,
"Hello, "
"world!");
static_void_test_AMlistPutObject(Void, update) static_void_test_AMlistPutStr(update,
"Hello,"
" world"
"!");
static_void_test_AMlistPutStr(insert, "Hello, world!") static_void_test_AMlistPut(Timestamp, insert, int64_t, INT64_MAX);
static_void_test_AMlistPutStr(update, "Hello, world!") static_void_test_AMlistPut(Timestamp, update, int64_t, INT64_MAX);
static_void_test_AMlistPut(Timestamp, insert, timestamp, INT64_MAX) static_void_test_AMlistPut(Uint, insert, uint64_t, UINT64_MAX);
static_void_test_AMlistPut(Timestamp, update, timestamp, INT64_MAX) static_void_test_AMlistPut(Uint, update, uint64_t, UINT64_MAX);
static_void_test_AMlistPut(Uint, insert, uint, UINT64_MAX) static void test_get_range_values(void** state) {
BaseState* base_state = *state;
static_void_test_AMlistPut(Uint, update, uint, UINT64_MAX) AMstack** stack_ptr = &base_state->stack;
AMdoc* doc1;
static void test_get_list_values(void** state) { assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc1));
AMresultStack* stack = *state; AMobjId const* const list =
AMdoc* const doc1 = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, cmocka_cb).doc; AMitemObjId(AMstackItem(stack_ptr, AMmapPutObject(doc1, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), cmocka_cb,
AMobjId const* const list = AMpush( AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
&stack,
AMmapPutObject(doc1, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST),
AM_VALUE_OBJ_ID,
cmocka_cb).obj_id;
/* Insert elements. */ /* Insert elements. */
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("First"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("First")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Second"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Second")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Third"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Third")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Fourth"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Fourth")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Fifth"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Fifth")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Sixth"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Sixth")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Seventh"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Seventh")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutStr(doc1, list, 0, true, AMstr("Eighth"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 0, true, AMstr("Eighth")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMcommit(doc1, AMstr(NULL), NULL)); AMstackItem(NULL, AMcommit(doc1, AMstr(NULL), NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMchangeHashes const v1 = AMpush(&stack, AMitems const v1 = AMstackItems(stack_ptr, AMgetHeads(doc1), cmocka_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMgetHeads(doc1), AMdoc* doc2;
AM_VALUE_CHANGE_HASHES, assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMfork(doc1, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc2));
cmocka_cb).change_hashes;
AMdoc* const doc2 = AMpush(&stack,
AMfork(doc1, NULL),
AM_VALUE_DOC,
cmocka_cb).doc;
AMfree(AMlistPutStr(doc1, list, 2, false, AMstr("Third V2"))); AMstackItem(NULL, AMlistPutStr(doc1, list, 2, false, AMstr("Third V2")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMcommit(doc1, AMstr(NULL), NULL)); AMstackItem(NULL, AMcommit(doc1, AMstr(NULL), NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMfree(AMlistPutStr(doc2, list, 2, false, AMstr("Third V3"))); AMstackItem(NULL, AMlistPutStr(doc2, list, 2, false, AMstr("Third V3")), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMcommit(doc2, AMstr(NULL), NULL)); AMstackItem(NULL, AMcommit(doc2, AMstr(NULL), NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMfree(AMmerge(doc1, doc2)); AMstackItem(NULL, AMmerge(doc1, doc2), cmocka_cb, AMexpect(AM_VAL_TYPE_CHANGE_HASH));
AMlistItems range = AMpush(&stack, /* Forward vs. reverse: complete current list range. */
AMlistRange(doc1, list, 0, SIZE_MAX, NULL), AMitems range =
AM_VALUE_LIST_ITEMS, AMstackItems(stack_ptr, AMlistRange(doc1, list, 0, SIZE_MAX, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
cmocka_cb).list_items; size_t size = AMitemsSize(&range);
assert_int_equal(AMlistItemsSize(&range), 8); assert_int_equal(size, 8);
AMitems range_back = AMitemsReversed(&range);
assert_int_equal(AMitemsSize(&range_back), size);
size_t pos;
assert_true(AMitemPos(AMitemsNext(&range, 1), &pos));
assert_int_equal(pos, 0);
assert_true(AMitemPos(AMitemsNext(&range_back, 1), &pos));
assert_int_equal(pos, 7);
AMlistItem const* list_item = NULL; AMitem *item1, *item_back1;
while ((list_item = AMlistItemsNext(&range, 1)) != NULL) { size_t count, middle = size / 2;
AMvalue const val1 = AMlistItemValue(list_item); range = AMitemsRewound(&range);
AMresult* result = AMlistGet(doc1, list, AMlistItemIndex(list_item), NULL); range_back = AMitemsRewound(&range_back);
AMvalue const val2 = AMresultValue(result); for (item1 = NULL, item_back1 = NULL, count = 0; item1 && item_back1;
assert_true(AMvalueEqual(&val1, &val2)); item1 = AMitemsNext(&range, 1), item_back1 = AMitemsNext(&range_back, 1), ++count) {
assert_non_null(AMlistItemObjId(list_item)); size_t pos1, pos_back1;
AMfree(result); assert_true(AMitemPos(item1, &pos1));
assert_true(AMitemPos(item_back1, &pos_back1));
if ((count == middle) && (middle & 1)) {
/* The iterators are crossing in the middle. */
assert_int_equal(pos1, pos_back1);
assert_true(AMitemEqual(item1, item_back1));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item_back1)));
} else {
assert_int_not_equal(pos1, pos_back1);
}
AMitem* item2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos1, NULL), NULL, NULL);
AMitem* item_back2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos_back1, NULL), NULL, NULL);
/** \note An item returned from an `AM...Get()` call doesn't include the
index used to retrieve it. */
assert_false(AMitemIdxType(item2));
assert_false(AMitemIdxType(item_back2));
assert_true(AMitemEqual(item1, item2));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item2)));
assert_true(AMitemEqual(item_back1, item_back2));
assert_true(AMobjIdEqual(AMitemObjId(item_back1), AMitemObjId(item_back2)));
AMresultFree(AMstackPop(stack_ptr, NULL));
} }
range = AMpush(&stack, /* Forward vs. reverse: partial current list range. */
AMlistRange(doc1, list, 3, 6, NULL), range = AMstackItems(stack_ptr, AMlistRange(doc1, list, 1, 6, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
AM_VALUE_LIST_ITEMS, size = AMitemsSize(&range);
cmocka_cb).list_items; assert_int_equal(size, 5);
AMlistItems range_back = AMlistItemsReversed(&range); range_back = AMitemsReversed(&range);
assert_int_equal(AMlistItemsSize(&range), 3); assert_int_equal(AMitemsSize(&range_back), size);
assert_int_equal(AMlistItemIndex(AMlistItemsNext(&range, 1)), 3); assert_true(AMitemPos(AMitemsNext(&range, 1), &pos));
assert_int_equal(AMlistItemIndex(AMlistItemsNext(&range_back, 1)), 5); assert_int_equal(pos, 1);
assert_true(AMitemPos(AMitemsNext(&range_back, 1), &pos));
assert_int_equal(pos, 5);
range = AMlistItemsRewound(&range); middle = size / 2;
while ((list_item = AMlistItemsNext(&range, 1)) != NULL) { range = AMitemsRewound(&range);
AMvalue const val1 = AMlistItemValue(list_item); range_back = AMitemsRewound(&range_back);
AMresult* result = AMlistGet(doc1, list, AMlistItemIndex(list_item), NULL); for (item1 = NULL, item_back1 = NULL, count = 0; item1 && item_back1;
AMvalue const val2 = AMresultValue(result); item1 = AMitemsNext(&range, 1), item_back1 = AMitemsNext(&range_back, 1), ++count) {
assert_true(AMvalueEqual(&val1, &val2)); size_t pos1, pos_back1;
assert_non_null(AMlistItemObjId(list_item)); assert_true(AMitemPos(item1, &pos1));
AMfree(result); assert_true(AMitemPos(item_back1, &pos_back1));
if ((count == middle) && (middle & 1)) {
/* The iterators are crossing in the middle. */
assert_int_equal(pos1, pos_back1);
assert_true(AMitemEqual(item1, item_back1));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item_back1)));
} else {
assert_int_not_equal(pos1, pos_back1);
}
AMitem* item2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos1, NULL), NULL, NULL);
AMitem* item_back2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos_back1, NULL), NULL, NULL);
/** \note An item returned from an `AMlistGet()` call doesn't include
the index used to retrieve it. */
assert_int_equal(AMitemIdxType(item2), 0);
assert_int_equal(AMitemIdxType(item_back2), 0);
assert_true(AMitemEqual(item1, item2));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item2)));
assert_true(AMitemEqual(item_back1, item_back2));
assert_true(AMobjIdEqual(AMitemObjId(item_back1), AMitemObjId(item_back2)));
AMresultFree(AMstackPop(stack_ptr, NULL));
} }
range = AMpush(&stack, /* Forward vs. reverse: complete historical map range. */
AMlistRange(doc1, list, 0, SIZE_MAX, &v1), range = AMstackItems(stack_ptr, AMlistRange(doc1, list, 0, SIZE_MAX, &v1), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
AM_VALUE_LIST_ITEMS, size = AMitemsSize(&range);
cmocka_cb).list_items; assert_int_equal(size, 8);
assert_int_equal(AMlistItemsSize(&range), 8); range_back = AMitemsReversed(&range);
while ((list_item = AMlistItemsNext(&range, 1)) != NULL) { assert_int_equal(AMitemsSize(&range_back), size);
AMvalue const val1 = AMlistItemValue(list_item); assert_true(AMitemPos(AMitemsNext(&range, 1), &pos));
AMresult* result = AMlistGet(doc1, list, AMlistItemIndex(list_item), &v1); assert_int_equal(pos, 0);
AMvalue const val2 = AMresultValue(result); assert_true(AMitemPos(AMitemsNext(&range_back, 1), &pos));
assert_true(AMvalueEqual(&val1, &val2)); assert_int_equal(pos, 7);
assert_non_null(AMlistItemObjId(list_item));
AMfree(result); middle = size / 2;
range = AMitemsRewound(&range);
range_back = AMitemsRewound(&range_back);
for (item1 = NULL, item_back1 = NULL, count = 0; item1 && item_back1;
item1 = AMitemsNext(&range, 1), item_back1 = AMitemsNext(&range_back, 1), ++count) {
size_t pos1, pos_back1;
assert_true(AMitemPos(item1, &pos1));
assert_true(AMitemPos(item_back1, &pos_back1));
if ((count == middle) && (middle & 1)) {
/* The iterators are crossing in the middle. */
assert_int_equal(pos1, pos_back1);
assert_true(AMitemEqual(item1, item_back1));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item_back1)));
} else {
assert_int_not_equal(pos1, pos_back1);
}
AMitem* item2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos1, &v1), NULL, NULL);
AMitem* item_back2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos_back1, &v1), NULL, NULL);
/** \note An item returned from an `AM...Get()` call doesn't include the
index used to retrieve it. */
assert_false(AMitemIdxType(item2));
assert_false(AMitemIdxType(item_back2));
assert_true(AMitemEqual(item1, item2));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item2)));
assert_true(AMitemEqual(item_back1, item_back2));
assert_true(AMobjIdEqual(AMitemObjId(item_back1), AMitemObjId(item_back2)));
AMresultFree(AMstackPop(stack_ptr, NULL));
} }
range = AMpush(&stack, /* Forward vs. reverse: partial historical map range. */
AMlistRange(doc1, list, 3, 6, &v1), range = AMstackItems(stack_ptr, AMlistRange(doc1, list, 2, 7, &v1), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
AM_VALUE_LIST_ITEMS, size = AMitemsSize(&range);
cmocka_cb).list_items; assert_int_equal(size, 5);
range_back = AMlistItemsReversed(&range); range_back = AMitemsReversed(&range);
assert_int_equal(AMlistItemsSize(&range), 3); assert_int_equal(AMitemsSize(&range_back), size);
assert_int_equal(AMlistItemIndex(AMlistItemsNext(&range, 1)), 3); assert_true(AMitemPos(AMitemsNext(&range, 1), &pos));
assert_int_equal(AMlistItemIndex(AMlistItemsNext(&range_back, 1)), 5); assert_int_equal(pos, 2);
assert_true(AMitemPos(AMitemsNext(&range_back, 1), &pos));
assert_int_equal(pos, 6);
range = AMlistItemsRewound(&range); middle = size / 2;
while ((list_item = AMlistItemsNext(&range, 1)) != NULL) { range = AMitemsRewound(&range);
AMvalue const val1 = AMlistItemValue(list_item); range_back = AMitemsRewound(&range_back);
AMresult* result = AMlistGet(doc1, list, AMlistItemIndex(list_item), &v1); for (item1 = NULL, item_back1 = NULL, count = 0; item1 && item_back1;
AMvalue const val2 = AMresultValue(result); item1 = AMitemsNext(&range, 1), item_back1 = AMitemsNext(&range_back, 1), ++count) {
assert_true(AMvalueEqual(&val1, &val2)); size_t pos1, pos_back1;
assert_non_null(AMlistItemObjId(list_item)); assert_true(AMitemPos(item1, &pos1));
AMfree(result); assert_true(AMitemPos(item_back1, &pos_back1));
if ((count == middle) && (middle & 1)) {
/* The iterators are crossing in the middle. */
assert_int_equal(pos1, pos_back1);
assert_true(AMitemEqual(item1, item_back1));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item_back1)));
} else {
assert_int_not_equal(pos1, pos_back1);
}
AMitem* item2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos1, &v1), NULL, NULL);
AMitem* item_back2 = AMstackItem(stack_ptr, AMlistGet(doc1, list, pos_back1, &v1), NULL, NULL);
/** \note An item returned from an `AM...Get()` call doesn't include the
index used to retrieve it. */
assert_false(AMitemIdxType(item2));
assert_false(AMitemIdxType(item_back2));
assert_true(AMitemEqual(item1, item2));
assert_true(AMobjIdEqual(AMitemObjId(item1), AMitemObjId(item2)));
assert_true(AMitemEqual(item_back1, item_back2));
assert_true(AMobjIdEqual(AMitemObjId(item_back1), AMitemObjId(item_back2)));
AMresultFree(AMstackPop(stack_ptr, NULL));
} }
range = AMpush(&stack, /* List range vs. object range: complete current. */
AMlistRange(doc1, list, 0, SIZE_MAX, NULL), range = AMstackItems(stack_ptr, AMlistRange(doc1, list, 0, SIZE_MAX, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
AM_VALUE_LIST_ITEMS, AMitems obj_items = AMstackItems(stack_ptr, AMobjItems(doc1, list, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
cmocka_cb).list_items; assert_int_equal(AMitemsSize(&range), AMitemsSize(&obj_items));
AMobjItems values = AMpush(&stack,
AMobjValues(doc1, list, NULL), AMitem *item, *obj_item;
AM_VALUE_OBJ_ITEMS, for (item = NULL, obj_item = NULL; item && obj_item;
cmocka_cb).obj_items; item = AMitemsNext(&range, 1), obj_item = AMitemsNext(&obj_items, 1)) {
assert_int_equal(AMlistItemsSize(&range), AMobjItemsSize(&values)); /** \note Object iteration doesn't yield any item indices. */
AMobjItem const* value = NULL; assert_true(AMitemIdxType(item));
while ((list_item = AMlistItemsNext(&range, 1)) != NULL && assert_false(AMitemIdxType(obj_item));
(value = AMobjItemsNext(&values, 1)) != NULL) { assert_true(AMitemEqual(item, obj_item));
AMvalue const val1 = AMlistItemValue(list_item); assert_true(AMobjIdEqual(AMitemObjId(item), AMitemObjId(obj_item)));
AMvalue const val2 = AMobjItemValue(value);
assert_true(AMvalueEqual(&val1, &val2));
assert_true(AMobjIdEqual(AMlistItemObjId(list_item), AMobjItemObjId(value)));
} }
range = AMpush(&stack, /* List range vs. object range: complete historical. */
AMlistRange(doc1, list, 0, SIZE_MAX, &v1), range = AMstackItems(stack_ptr, AMlistRange(doc1, list, 0, SIZE_MAX, &v1), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
AM_VALUE_LIST_ITEMS, obj_items = AMstackItems(stack_ptr, AMobjItems(doc1, list, &v1), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
cmocka_cb).list_items; assert_int_equal(AMitemsSize(&range), AMitemsSize(&obj_items));
values = AMpush(&stack,
AMobjValues(doc1, list, &v1), for (item = NULL, obj_item = NULL; item && obj_item;
AM_VALUE_OBJ_ITEMS, item = AMitemsNext(&range, 1), obj_item = AMitemsNext(&obj_items, 1)) {
cmocka_cb).obj_items; /** \note Object iteration doesn't yield any item indices. */
assert_int_equal(AMlistItemsSize(&range), AMobjItemsSize(&values)); assert_true(AMitemIdxType(item));
while ((list_item = AMlistItemsNext(&range, 1)) != NULL && assert_false(AMitemIdxType(obj_item));
(value = AMobjItemsNext(&values, 1)) != NULL) { assert_true(AMitemEqual(item, obj_item));
AMvalue const val1 = AMlistItemValue(list_item); assert_true(AMobjIdEqual(AMitemObjId(item), AMitemObjId(obj_item)));
AMvalue const val2 = AMobjItemValue(value);
assert_true(AMvalueEqual(&val1, &val2));
assert_true(AMobjIdEqual(AMlistItemObjId(list_item), AMobjItemObjId(value)));
} }
} }
/** \brief A JavaScript application can introduce NUL (`\0`) characters into a /**
* list object's string value which will truncate it in a C application. * \brief A JavaScript application can introduce NUL (`\0`) characters into a
* list object's string value which will truncate it in a C application.
*/ */
static void test_get_NUL_string_value(void** state) { static void test_get_NUL_string_value(void** state) {
/* /*
@ -381,60 +431,52 @@ static void test_get_NUL_string_value(void** state) {
doc[0] = 'o\0ps'; doc[0] = 'o\0ps';
}); });
const bytes = Automerge.save(doc); const bytes = Automerge.save(doc);
console.log("static uint8_t const SAVED_DOC[] = {" + Array.apply([], bytes).join(", ") + "};"); console.log("static uint8_t const SAVED_DOC[] = {" + Array.apply([],
bytes).join(", ") + "};");
*/ */
static uint8_t const OOPS_VALUE[] = {'o', '\0', 'p', 's'}; static uint8_t const OOPS_VALUE[] = {'o', '\0', 'p', 's'};
static size_t const OOPS_SIZE = sizeof(OOPS_VALUE) / sizeof(uint8_t); static size_t const OOPS_SIZE = sizeof(OOPS_VALUE) / sizeof(uint8_t);
static uint8_t const SAVED_DOC[] = { static uint8_t const SAVED_DOC[] = {
133, 111, 74, 131, 224, 28, 197, 17, 0, 113, 1, 16, 246, 137, 63, 193, 133, 111, 74, 131, 224, 28, 197, 17, 0, 113, 1, 16, 246, 137, 63, 193, 255, 181, 76, 79, 129,
255, 181, 76, 79, 129, 213, 133, 29, 214, 158, 164, 15, 1, 207, 184, 213, 133, 29, 214, 158, 164, 15, 1, 207, 184, 14, 57, 1, 194, 79, 247, 82, 160, 134, 227, 144,
14, 57, 1, 194, 79, 247, 82, 160, 134, 227, 144, 5, 241, 136, 205, 5, 241, 136, 205, 238, 250, 251, 54, 34, 250, 210, 96, 204, 132, 153, 203, 110, 109, 6, 6, 1,
238, 250, 251, 54, 34, 250, 210, 96, 204, 132, 153, 203, 110, 109, 6, 2, 3, 2, 19, 2, 35, 2, 64, 2, 86, 2, 8, 21, 3, 33, 2, 35, 2, 52, 1, 66,
6, 1, 2, 3, 2, 19, 2, 35, 2, 64, 2, 86, 2, 8, 21, 3, 33, 2, 35, 2, 52, 2, 86, 2, 87, 4, 128, 1, 2, 127, 0, 127, 1, 127, 1, 127, 0, 127, 0, 127, 7, 127,
1, 66, 2, 86, 2, 87, 4, 128, 1, 2, 127, 0, 127, 1, 127, 1, 127, 0, 1, 48, 127, 0, 127, 1, 1, 127, 1, 127, 70, 111, 0, 112, 115, 127, 0, 0};
127, 0, 127, 7, 127, 1, 48, 127, 0, 127, 1, 1, 127, 1, 127, 70, 111,
0, 112, 115, 127, 0, 0};
static size_t const SAVED_DOC_SIZE = sizeof(SAVED_DOC) / sizeof(uint8_t); static size_t const SAVED_DOC_SIZE = sizeof(SAVED_DOC) / sizeof(uint8_t);
AMresultStack* stack = *state; BaseState* base_state = *state;
AMdoc* const doc = AMpush(&stack, AMstack** stack_ptr = &base_state->stack;
AMload(SAVED_DOC, SAVED_DOC_SIZE), AMdoc* doc;
AM_VALUE_DOC, assert_true(AMitemToDoc(
cmocka_cb).doc; AMstackItem(stack_ptr, AMload(SAVED_DOC, SAVED_DOC_SIZE), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc));
AMbyteSpan const str = AMpush(&stack, AMbyteSpan str;
AMlistGet(doc, AM_ROOT, 0, NULL), assert_true(AMitemToStr(
AM_VALUE_STR, AMstackItem(stack_ptr, AMlistGet(doc, AM_ROOT, 0, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR)), &str));
cmocka_cb).str;
assert_int_not_equal(str.count, strlen(OOPS_VALUE)); assert_int_not_equal(str.count, strlen(OOPS_VALUE));
assert_int_equal(str.count, OOPS_SIZE); assert_int_equal(str.count, OOPS_SIZE);
assert_memory_equal(str.src, OOPS_VALUE, str.count); assert_memory_equal(str.src, OOPS_VALUE, str.count);
} }
static void test_insert_at_index(void** state) { static void test_insert_at_index(void** state) {
AMresultStack* stack = *state; BaseState* base_state = *state;
AMdoc* const doc = AMpush(&stack, AMcreate(NULL), AM_VALUE_DOC, cmocka_cb).doc; AMstack** stack_ptr = &base_state->stack;
AMdoc* doc;
AMobjId const* const list = AMpush( assert_true(AMitemToDoc(AMstackItem(stack_ptr, AMcreate(NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_DOC)), &doc));
&stack, AMobjId const* const list =
AMmapPutObject(doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), AMitemObjId(AMstackItem(stack_ptr, AMmapPutObject(doc, AM_ROOT, AMstr("list"), AM_OBJ_TYPE_LIST), cmocka_cb,
AM_VALUE_OBJ_ID, AMexpect(AM_VAL_TYPE_OBJ_TYPE)));
cmocka_cb).obj_id;
/* Insert both at the same index. */ /* Insert both at the same index. */
AMfree(AMlistPutUint(doc, list, 0, true, 0)); AMstackItem(NULL, AMlistPutUint(doc, list, 0, true, 0), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
AMfree(AMlistPutUint(doc, list, 0, true, 1)); AMstackItem(NULL, AMlistPutUint(doc, list, 0, true, 1), cmocka_cb, AMexpect(AM_VAL_TYPE_VOID));
assert_int_equal(AMobjSize(doc, list, NULL), 2); assert_int_equal(AMobjSize(doc, list, NULL), 2);
AMstrs const keys = AMpush(&stack, AMitems const keys = AMstackItems(stack_ptr, AMkeys(doc, list, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_STR));
AMkeys(doc, list, NULL), assert_int_equal(AMitemsSize(&keys), 2);
AM_VALUE_STRS, AMitems const range =
cmocka_cb).strs; AMstackItems(stack_ptr, AMlistRange(doc, list, 0, SIZE_MAX, NULL), cmocka_cb, AMexpect(AM_VAL_TYPE_UINT));
assert_int_equal(AMstrsSize(&keys), 2); assert_int_equal(AMitemsSize(&range), 2);
AMlistItems const range = AMpush(&stack,
AMlistRange(doc, list, 0, SIZE_MAX, NULL),
AM_VALUE_LIST_ITEMS,
cmocka_cb).list_items;
assert_int_equal(AMlistItemsSize(&range), 2);
} }
int run_list_tests(void) { int run_list_tests(void) {
@ -458,18 +500,16 @@ int run_list_tests(void) {
cmocka_unit_test(test_AMlistPutObject(Map, update)), cmocka_unit_test(test_AMlistPutObject(Map, update)),
cmocka_unit_test(test_AMlistPutObject(Text, insert)), cmocka_unit_test(test_AMlistPutObject(Text, insert)),
cmocka_unit_test(test_AMlistPutObject(Text, update)), cmocka_unit_test(test_AMlistPutObject(Text, update)),
cmocka_unit_test(test_AMlistPutObject(Void, insert)),
cmocka_unit_test(test_AMlistPutObject(Void, update)),
cmocka_unit_test(test_AMlistPutStr(insert)), cmocka_unit_test(test_AMlistPutStr(insert)),
cmocka_unit_test(test_AMlistPutStr(update)), cmocka_unit_test(test_AMlistPutStr(update)),
cmocka_unit_test(test_AMlistPut(Timestamp, insert)), cmocka_unit_test(test_AMlistPut(Timestamp, insert)),
cmocka_unit_test(test_AMlistPut(Timestamp, update)), cmocka_unit_test(test_AMlistPut(Timestamp, update)),
cmocka_unit_test(test_AMlistPut(Uint, insert)), cmocka_unit_test(test_AMlistPut(Uint, insert)),
cmocka_unit_test(test_AMlistPut(Uint, update)), cmocka_unit_test(test_AMlistPut(Uint, update)),
cmocka_unit_test_setup_teardown(test_get_list_values, setup_stack, teardown_stack), cmocka_unit_test_setup_teardown(test_get_range_values, setup_base, teardown_base),
cmocka_unit_test_setup_teardown(test_get_NUL_string_value, setup_stack, teardown_stack), cmocka_unit_test_setup_teardown(test_get_NUL_string_value, setup_base, teardown_base),
cmocka_unit_test_setup_teardown(test_insert_at_index, setup_stack, teardown_stack), cmocka_unit_test_setup_teardown(test_insert_at_index, setup_base, teardown_base),
}; };
return cmocka_run_group_tests(tests, group_setup, group_teardown); return cmocka_run_group_tests(tests, setup_doc, teardown_doc);
} }

View file

@ -3,23 +3,36 @@
/* local */ /* local */
#include "macro_utils.h" #include "macro_utils.h"
AMvalueVariant AMvalue_discriminant(char const* suffix) { AMobjType suffix_to_obj_type(char const* obj_type_label) {
if (!strcmp(suffix, "Bool")) return AM_VALUE_BOOLEAN; if (!strcmp(obj_type_label, "List"))
else if (!strcmp(suffix, "Bytes")) return AM_VALUE_BYTES; return AM_OBJ_TYPE_LIST;
else if (!strcmp(suffix, "Counter")) return AM_VALUE_COUNTER; else if (!strcmp(obj_type_label, "Map"))
else if (!strcmp(suffix, "F64")) return AM_VALUE_F64; return AM_OBJ_TYPE_MAP;
else if (!strcmp(suffix, "Int")) return AM_VALUE_INT; else if (!strcmp(obj_type_label, "Text"))
else if (!strcmp(suffix, "Null")) return AM_VALUE_NULL; return AM_OBJ_TYPE_TEXT;
else if (!strcmp(suffix, "Str")) return AM_VALUE_STR; else
else if (!strcmp(suffix, "Timestamp")) return AM_VALUE_TIMESTAMP; return AM_OBJ_TYPE_DEFAULT;
else if (!strcmp(suffix, "Uint")) return AM_VALUE_UINT;
else return AM_VALUE_VOID;
} }
AMobjType AMobjType_tag(char const* obj_type_label) { AMvalType suffix_to_val_type(char const* suffix) {
if (!strcmp(obj_type_label, "List")) return AM_OBJ_TYPE_LIST; if (!strcmp(suffix, "Bool"))
else if (!strcmp(obj_type_label, "Map")) return AM_OBJ_TYPE_MAP; return AM_VAL_TYPE_BOOL;
else if (!strcmp(obj_type_label, "Text")) return AM_OBJ_TYPE_TEXT; else if (!strcmp(suffix, "Bytes"))
else if (!strcmp(obj_type_label, "Void")) return AM_OBJ_TYPE_VOID; return AM_VAL_TYPE_BYTES;
else return 0; else if (!strcmp(suffix, "Counter"))
return AM_VAL_TYPE_COUNTER;
else if (!strcmp(suffix, "F64"))
return AM_VAL_TYPE_F64;
else if (!strcmp(suffix, "Int"))
return AM_VAL_TYPE_INT;
else if (!strcmp(suffix, "Null"))
return AM_VAL_TYPE_NULL;
else if (!strcmp(suffix, "Str"))
return AM_VAL_TYPE_STR;
else if (!strcmp(suffix, "Timestamp"))
return AM_VAL_TYPE_TIMESTAMP;
else if (!strcmp(suffix, "Uint"))
return AM_VAL_TYPE_UINT;
else
return AM_VAL_TYPE_DEFAULT;
} }

View file

@ -1,24 +1,23 @@
#ifndef MACRO_UTILS_H #ifndef TESTS_MACRO_UTILS_H
#define MACRO_UTILS_H #define TESTS_MACRO_UTILS_H
/* local */ /* local */
#include <automerge-c/automerge.h> #include <automerge-c/automerge.h>
/** /**
* \brief Gets the result value discriminant corresponding to a function name * \brief Gets the object type tag corresponding to an object type suffix.
* suffix.
* *
* \param[in] suffix A string. * \param[in] suffix An object type suffix string.
* \return An `AMvalue` struct discriminant.
*/
AMvalueVariant AMvalue_discriminant(char const* suffix);
/**
* \brief Gets the object type tag corresponding to an object type label.
*
* \param[in] obj_type_label A string.
* \return An `AMobjType` enum tag. * \return An `AMobjType` enum tag.
*/ */
AMobjType AMobjType_tag(char const* obj_type_label); AMobjType suffix_to_obj_type(char const* suffix);
#endif /* MACRO_UTILS_H */ /**
* \brief Gets the value type tag corresponding to a value type suffix.
*
* \param[in] suffix A value type suffix string.
* \return An `AMvalType` enum tag.
*/
AMvalType suffix_to_val_type(char const* suffix);
#endif /* TESTS_MACRO_UTILS_H */

View file

@ -1,6 +1,6 @@
#include <setjmp.h>
#include <stdarg.h> #include <stdarg.h>
#include <stddef.h> #include <stddef.h>
#include <setjmp.h>
#include <stdint.h> #include <stdint.h>
/* third-party */ /* third-party */
@ -8,8 +8,14 @@
extern int run_actor_id_tests(void); extern int run_actor_id_tests(void);
extern int run_byte_span_tests(void);
extern int run_doc_tests(void); extern int run_doc_tests(void);
extern int run_enum_string_tests(void);
extern int run_item_tests(void);
extern int run_list_tests(void); extern int run_list_tests(void);
extern int run_map_tests(void); extern int run_map_tests(void);
@ -17,11 +23,6 @@ extern int run_map_tests(void);
extern int run_ported_wasm_suite(void); extern int run_ported_wasm_suite(void);
int main(void) { int main(void) {
return ( return (run_actor_id_tests() + run_byte_span_tests() + run_doc_tests() + run_enum_string_tests() +
run_actor_id_tests() + run_item_tests() + run_list_tests() + run_map_tests() + run_ported_wasm_suite());
run_doc_tests() +
run_list_tests() +
run_map_tests() +
run_ported_wasm_suite()
);
} }

File diff suppressed because it is too large Load diff

File diff suppressed because it is too large Load diff

View file

@ -1,6 +1,6 @@
#include <setjmp.h>
#include <stdarg.h> #include <stdarg.h>
#include <stddef.h> #include <stddef.h>
#include <setjmp.h>
#include <stdint.h> #include <stdint.h>
/* third-party */ /* third-party */
@ -11,8 +11,5 @@ extern int run_ported_wasm_basic_tests(void);
extern int run_ported_wasm_sync_tests(void); extern int run_ported_wasm_sync_tests(void);
int run_ported_wasm_suite(void) { int run_ported_wasm_suite(void) {
return ( return (run_ported_wasm_basic_tests() + run_ported_wasm_sync_tests());
run_ported_wasm_basic_tests() +
run_ported_wasm_sync_tests()
);
} }

File diff suppressed because it is too large Load diff

View file

@ -1,31 +0,0 @@
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
/* third-party */
#include <cmocka.h>
/* local */
#include "cmocka_utils.h"
#include "stack_utils.h"
void cmocka_cb(AMresultStack** stack, uint8_t discriminant) {
assert_non_null(stack);
assert_non_null(*stack);
assert_non_null((*stack)->result);
if (AMresultStatus((*stack)->result) != AM_STATUS_OK) {
fail_msg_view("%s", AMerrorMessage((*stack)->result));
}
assert_int_equal(AMresultValue((*stack)->result).tag, discriminant);
}
int setup_stack(void** state) {
*state = NULL;
return 0;
}
int teardown_stack(void** state) {
AMresultStack* stack = *state;
AMfreeStack(&stack);
return 0;
}

View file

@ -1,38 +0,0 @@
#ifndef STACK_UTILS_H
#define STACK_UTILS_H
#include <stdint.h>
/* local */
#include <automerge-c/automerge.h>
/**
* \brief Reports an error through a cmocka assertion.
*
* \param[in,out] stack A pointer to a pointer to an `AMresultStack` struct.
* \param[in] discriminant An `AMvalueVariant` enum tag.
* \pre \p stack` != NULL`.
*/
void cmocka_cb(AMresultStack** stack, uint8_t discriminant);
/**
* \brief Allocates a result stack for storing the results allocated during one
* or more test cases.
*
* \param[in,out] state A pointer to a pointer to an `AMresultStack` struct.
* \pre \p state` != NULL`.
* \warning The `AMresultStack` struct returned through \p state must be
* deallocated with `teardown_stack()` in order to prevent memory leaks.
*/
int setup_stack(void** state);
/**
* \brief Deallocates a result stack after deallocating any results that were
* stored in it by one or more test cases.
*
* \param[in] state A pointer to a pointer to an `AMresultStack` struct.
* \pre \p state` != NULL`.
*/
int teardown_stack(void** state);
#endif /* STACK_UTILS_H */

View file

@ -1,5 +1,5 @@
#include <stdio.h>
#include <stdint.h> #include <stdint.h>
#include <stdio.h>
/* local */ /* local */
#include "str_utils.h" #include "str_utils.h"

View file

@ -1,14 +1,17 @@
#ifndef STR_UTILS_H #ifndef TESTS_STR_UTILS_H
#define STR_UTILS_H #define TESTS_STR_UTILS_H
/** /**
* \brief Converts a hexadecimal string into a sequence of bytes. * \brief Converts a hexadecimal string into an array of bytes.
* *
* \param[in] hex_str A string. * \param[in] hex_str A hexadecimal string.
* \param[in] src A pointer to a contiguous sequence of bytes. * \param[in] src A pointer to an array of bytes.
* \param[in] count The number of bytes to copy to \p src. * \param[in] count The count of bytes to copy into the array pointed to by
* \pre \p count `<=` length of \p src. * \p src.
* \pre \p src `!= NULL`
* \pre `sizeof(`\p src `) > 0`
* \pre \p count `<= sizeof(`\p src `)`
*/ */
void hex_to_bytes(char const* hex_str, uint8_t* src, size_t const count); void hex_to_bytes(char const* hex_str, uint8_t* src, size_t const count);
#endif /* STR_UTILS_H */ #endif /* TESTS_STR_UTILS_H */

View file

@ -1,857 +0,0 @@
# This file is automatically @generated by Cargo.
# It is not intended for manual editing.
version = 3
[[package]]
name = "adler"
version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe"
[[package]]
name = "ansi_term"
version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d52a9bb7ec0cf484c551830a7ce27bd20d67eac647e1befb56b0be4ee39a55d2"
dependencies = [
"winapi",
]
[[package]]
name = "anyhow"
version = "1.0.55"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "159bb86af3a200e19a068f4224eae4c8bb2d0fa054c7e5d1cacd5cef95e684cd"
[[package]]
name = "atty"
version = "0.2.14"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8"
dependencies = [
"hermit-abi",
"libc",
"winapi",
]
[[package]]
name = "autocfg"
version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa"
[[package]]
name = "automerge"
version = "0.1.0"
dependencies = [
"flate2",
"fxhash",
"hex",
"itertools",
"js-sys",
"leb128",
"nonzero_ext",
"rand",
"serde",
"sha2",
"smol_str",
"thiserror",
"tinyvec",
"tracing",
"unicode-segmentation",
"uuid",
"wasm-bindgen",
"web-sys",
]
[[package]]
name = "automerge-cli"
version = "0.1.0"
dependencies = [
"anyhow",
"atty",
"automerge",
"clap",
"colored_json",
"combine",
"duct",
"maplit",
"serde_json",
"thiserror",
"tracing-subscriber",
]
[[package]]
name = "bitflags"
version = "1.3.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
[[package]]
name = "block-buffer"
version = "0.10.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0bf7fe51849ea569fd452f37822f606a5cabb684dc918707a0193fd4664ff324"
dependencies = [
"generic-array",
]
[[package]]
name = "bumpalo"
version = "3.9.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a4a45a46ab1f2412e53d3a0ade76ffad2025804294569aae387231a0cd6e0899"
[[package]]
name = "byteorder"
version = "1.4.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610"
[[package]]
name = "bytes"
version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c4872d67bab6358e59559027aa3b9157c53d9358c51423c17554809a8858e0f8"
[[package]]
name = "cfg-if"
version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
[[package]]
name = "clap"
version = "3.1.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ced1892c55c910c1219e98d6fc8d71f6bddba7905866ce740066d8bfea859312"
dependencies = [
"atty",
"bitflags",
"clap_derive",
"indexmap",
"lazy_static",
"os_str_bytes",
"strsim",
"termcolor",
"textwrap",
]
[[package]]
name = "clap_derive"
version = "3.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "da95d038ede1a964ce99f49cbe27a7fb538d1da595e4b4f70b8c8f338d17bf16"
dependencies = [
"heck",
"proc-macro-error",
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "colored_json"
version = "2.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1fd32eb54d016e203b7c2600e3a7802c75843a92e38ccc4869aefeca21771a64"
dependencies = [
"ansi_term",
"atty",
"libc",
"serde",
"serde_json",
]
[[package]]
name = "combine"
version = "4.6.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "50b727aacc797f9fc28e355d21f34709ac4fc9adecfe470ad07b8f4464f53062"
dependencies = [
"bytes",
"memchr",
]
[[package]]
name = "cpufeatures"
version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "95059428f66df56b63431fdb4e1947ed2190586af5c5a8a8b71122bdf5a7f469"
dependencies = [
"libc",
]
[[package]]
name = "crc32fast"
version = "1.3.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b540bd8bc810d3885c6ea91e2018302f68baba2129ab3e88f32389ee9370880d"
dependencies = [
"cfg-if",
]
[[package]]
name = "crypto-common"
version = "0.1.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "57952ca27b5e3606ff4dd79b0020231aaf9d6aa76dc05fd30137538c50bd3ce8"
dependencies = [
"generic-array",
"typenum",
]
[[package]]
name = "digest"
version = "0.10.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f2fb860ca6fafa5552fb6d0e816a69c8e49f0908bf524e30a90d97c85892d506"
dependencies = [
"block-buffer",
"crypto-common",
]
[[package]]
name = "duct"
version = "0.13.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0fc6a0a59ed0888e0041cf708e66357b7ae1a82f1c67247e1f93b5e0818f7d8d"
dependencies = [
"libc",
"once_cell",
"os_pipe",
"shared_child",
]
[[package]]
name = "either"
version = "1.6.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457"
[[package]]
name = "flate2"
version = "1.0.22"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1e6988e897c1c9c485f43b47a529cef42fde0547f9d8d41a7062518f1d8fc53f"
dependencies = [
"cfg-if",
"crc32fast",
"libc",
"miniz_oxide",
]
[[package]]
name = "fxhash"
version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c"
dependencies = [
"byteorder",
]
[[package]]
name = "generic-array"
version = "0.14.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fd48d33ec7f05fbfa152300fdad764757cbded343c1aa1cff2fbaf4134851803"
dependencies = [
"typenum",
"version_check",
]
[[package]]
name = "getrandom"
version = "0.2.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d39cd93900197114fa1fcb7ae84ca742095eed9442088988ae74fa744e930e77"
dependencies = [
"cfg-if",
"js-sys",
"libc",
"wasi",
"wasm-bindgen",
]
[[package]]
name = "hashbrown"
version = "0.11.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e"
[[package]]
name = "heck"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2540771e65fc8cb83cd6e8a237f70c319bd5c29f78ed1084ba5d50eeac86f7f9"
[[package]]
name = "hermit-abi"
version = "0.1.19"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33"
dependencies = [
"libc",
]
[[package]]
name = "hex"
version = "0.4.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70"
[[package]]
name = "indexmap"
version = "1.8.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "282a6247722caba404c065016bbfa522806e51714c34f5dfc3e4a3a46fcb4223"
dependencies = [
"autocfg",
"hashbrown",
]
[[package]]
name = "itertools"
version = "0.10.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a9a9d19fa1e79b6215ff29b9d6880b706147f16e9b1dbb1e4e5947b5b02bc5e3"
dependencies = [
"either",
]
[[package]]
name = "itoa"
version = "1.0.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1aab8fc367588b89dcee83ab0fd66b72b50b72fa1904d7095045ace2b0c81c35"
[[package]]
name = "js-sys"
version = "0.3.56"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a38fc24e30fd564ce974c02bf1d337caddff65be6cc4735a1f7eab22a7440f04"
dependencies = [
"wasm-bindgen",
]
[[package]]
name = "lazy_static"
version = "1.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
[[package]]
name = "leb128"
version = "0.2.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "884e2677b40cc8c339eaefcb701c32ef1fd2493d71118dc0ca4b6a736c93bd67"
[[package]]
name = "libc"
version = "0.2.119"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bf2e165bb3457c8e098ea76f3e3bc9db55f87aa90d52d0e6be741470916aaa4"
[[package]]
name = "log"
version = "0.4.14"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "51b9bbe6c47d51fc3e1a9b945965946b4c44142ab8792c50835a980d362c2710"
dependencies = [
"cfg-if",
]
[[package]]
name = "maplit"
version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3e2e65a1a2e43cfcb47a895c4c8b10d1f4a61097f9f254f183aee60cad9c651d"
[[package]]
name = "memchr"
version = "2.4.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "308cc39be01b73d0d18f82a0e7b2a3df85245f84af96fdddc5d202d27e47b86a"
[[package]]
name = "miniz_oxide"
version = "0.4.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a92518e98c078586bc6c934028adcca4c92a53d6a958196de835170a01d84e4b"
dependencies = [
"adler",
"autocfg",
]
[[package]]
name = "nonzero_ext"
version = "0.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "44a1290799eababa63ea60af0cbc3f03363e328e58f32fb0294798ed3e85f444"
[[package]]
name = "once_cell"
version = "1.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "da32515d9f6e6e489d7bc9d84c71b060db7247dc035bbe44eac88cf87486d8d5"
[[package]]
name = "os_pipe"
version = "0.9.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fb233f06c2307e1f5ce2ecad9f8121cffbbee2c95428f44ea85222e460d0d213"
dependencies = [
"libc",
"winapi",
]
[[package]]
name = "os_str_bytes"
version = "6.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8e22443d1643a904602595ba1cd8f7d896afe56d26712531c5ff73a15b2fbf64"
dependencies = [
"memchr",
]
[[package]]
name = "pin-project-lite"
version = "0.2.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e280fbe77cc62c91527259e9442153f4688736748d24660126286329742b4c6c"
[[package]]
name = "ppv-lite86"
version = "0.2.16"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872"
[[package]]
name = "proc-macro-error"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c"
dependencies = [
"proc-macro-error-attr",
"proc-macro2",
"quote",
"syn",
"version_check",
]
[[package]]
name = "proc-macro-error-attr"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869"
dependencies = [
"proc-macro2",
"quote",
"version_check",
]
[[package]]
name = "proc-macro2"
version = "1.0.36"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c7342d5883fbccae1cc37a2353b09c87c9b0f3afd73f5fb9bba687a1f733b029"
dependencies = [
"unicode-xid",
]
[[package]]
name = "quote"
version = "1.0.15"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "864d3e96a899863136fc6e99f3d7cae289dafe43bf2c5ac19b70df7210c0a145"
dependencies = [
"proc-macro2",
]
[[package]]
name = "rand"
version = "0.8.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
dependencies = [
"libc",
"rand_chacha",
"rand_core",
]
[[package]]
name = "rand_chacha"
version = "0.3.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
dependencies = [
"ppv-lite86",
"rand_core",
]
[[package]]
name = "rand_core"
version = "0.6.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7"
dependencies = [
"getrandom",
]
[[package]]
name = "ryu"
version = "1.0.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "73b4b750c782965c211b42f022f59af1fbceabdd026623714f104152f1ec149f"
[[package]]
name = "serde"
version = "1.0.136"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ce31e24b01e1e524df96f1c2fdd054405f8d7376249a5110886fb4b658484789"
dependencies = [
"serde_derive",
]
[[package]]
name = "serde_derive"
version = "1.0.136"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "08597e7152fcd306f41838ed3e37be9eaeed2b61c42e2117266a554fab4662f9"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "serde_json"
version = "1.0.79"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8e8d9fa5c3b304765ce1fd9c4c8a3de2c8db365a5b91be52f186efc675681d95"
dependencies = [
"itoa",
"ryu",
"serde",
]
[[package]]
name = "sha2"
version = "0.10.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "55deaec60f81eefe3cce0dc50bda92d6d8e88f2a27df7c5033b42afeb1ed2676"
dependencies = [
"cfg-if",
"cpufeatures",
"digest",
]
[[package]]
name = "sharded-slab"
version = "0.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "900fba806f70c630b0a382d0d825e17a0f19fcd059a2ade1ff237bcddf446b31"
dependencies = [
"lazy_static",
]
[[package]]
name = "shared_child"
version = "0.3.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6be9f7d5565b1483af3e72975e2dee33879b3b86bd48c0929fccf6585d79e65a"
dependencies = [
"libc",
"winapi",
]
[[package]]
name = "smallvec"
version = "1.8.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f2dd574626839106c320a323308629dcb1acfc96e32a8cba364ddc61ac23ee83"
[[package]]
name = "smol_str"
version = "0.1.21"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "61d15c83e300cce35b7c8cd39ff567c1ef42dde6d4a1a38dbdbf9a59902261bd"
dependencies = [
"serde",
]
[[package]]
name = "strsim"
version = "0.10.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623"
[[package]]
name = "syn"
version = "1.0.86"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8a65b3f4ffa0092e9887669db0eae07941f023991ab58ea44da8fe8e2d511c6b"
dependencies = [
"proc-macro2",
"quote",
"unicode-xid",
]
[[package]]
name = "termcolor"
version = "1.1.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bab24d30b911b2376f3a13cc2cd443142f0c81dda04c118693e35b3835757755"
dependencies = [
"winapi-util",
]
[[package]]
name = "textwrap"
version = "0.15.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b1141d4d61095b28419e22cb0bbf02755f5e54e0526f97f1e3d1d160e60885fb"
[[package]]
name = "thiserror"
version = "1.0.30"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "854babe52e4df1653706b98fcfc05843010039b406875930a70e4d9644e5c417"
dependencies = [
"thiserror-impl",
]
[[package]]
name = "thiserror-impl"
version = "1.0.30"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "aa32fd3f627f367fe16f893e2597ae3c05020f8bba2666a4e6ea73d377e5714b"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "thread_local"
version = "1.1.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5516c27b78311c50bf42c071425c560ac799b11c30b31f87e3081965fe5e0180"
dependencies = [
"once_cell",
]
[[package]]
name = "tinyvec"
version = "1.5.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2c1c1d5a42b6245520c249549ec267180beaffcc0615401ac8e31853d4b6d8d2"
dependencies = [
"tinyvec_macros",
]
[[package]]
name = "tinyvec_macros"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cda74da7e1a664f795bb1f8a87ec406fb89a02522cf6e50620d016add6dbbf5c"
[[package]]
name = "tracing"
version = "0.1.31"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f6c650a8ef0cd2dd93736f033d21cbd1224c5a967aa0c258d00fcf7dafef9b9f"
dependencies = [
"cfg-if",
"log",
"pin-project-lite",
"tracing-attributes",
"tracing-core",
]
[[package]]
name = "tracing-attributes"
version = "0.1.19"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8276d9a4a3a558d7b7ad5303ad50b53d58264641b82914b7ada36bd762e7a716"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "tracing-core"
version = "0.1.22"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "03cfcb51380632a72d3111cb8d3447a8d908e577d31beeac006f836383d29a23"
dependencies = [
"lazy_static",
"valuable",
]
[[package]]
name = "tracing-log"
version = "0.1.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a6923477a48e41c1951f1999ef8bb5a3023eb723ceadafe78ffb65dc366761e3"
dependencies = [
"lazy_static",
"log",
"tracing-core",
]
[[package]]
name = "tracing-subscriber"
version = "0.3.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9e0ab7bdc962035a87fba73f3acca9b8a8d0034c2e6f60b84aeaaddddc155dce"
dependencies = [
"ansi_term",
"sharded-slab",
"smallvec",
"thread_local",
"tracing-core",
"tracing-log",
]
[[package]]
name = "typenum"
version = "1.15.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "dcf81ac59edc17cc8697ff311e8f5ef2d99fcbd9817b34cec66f90b6c3dfd987"
[[package]]
name = "unicode-segmentation"
version = "1.9.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7e8820f5d777f6224dc4be3632222971ac30164d4a258d595640799554ebfd99"
[[package]]
name = "unicode-xid"
version = "0.2.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3"
[[package]]
name = "uuid"
version = "0.8.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7"
dependencies = [
"getrandom",
"serde",
]
[[package]]
name = "valuable"
version = "0.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d"
[[package]]
name = "version_check"
version = "0.9.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f"
[[package]]
name = "wasi"
version = "0.10.2+wasi-snapshot-preview1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6"
[[package]]
name = "wasm-bindgen"
version = "0.2.79"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "25f1af7423d8588a3d840681122e72e6a24ddbcb3f0ec385cac0d12d24256c06"
dependencies = [
"cfg-if",
"wasm-bindgen-macro",
]
[[package]]
name = "wasm-bindgen-backend"
version = "0.2.79"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8b21c0df030f5a177f3cba22e9bc4322695ec43e7257d865302900290bcdedca"
dependencies = [
"bumpalo",
"lazy_static",
"log",
"proc-macro2",
"quote",
"syn",
"wasm-bindgen-shared",
]
[[package]]
name = "wasm-bindgen-macro"
version = "0.2.79"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2f4203d69e40a52ee523b2529a773d5ffc1dc0071801c87b3d270b471b80ed01"
dependencies = [
"quote",
"wasm-bindgen-macro-support",
]
[[package]]
name = "wasm-bindgen-macro-support"
version = "0.2.79"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bfa8a30d46208db204854cadbb5d4baf5fcf8071ba5bf48190c3e59937962ebc"
dependencies = [
"proc-macro2",
"quote",
"syn",
"wasm-bindgen-backend",
"wasm-bindgen-shared",
]
[[package]]
name = "wasm-bindgen-shared"
version = "0.2.79"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3d958d035c4438e28c70e4321a2911302f10135ce78a9c7834c0cab4123d06a2"
[[package]]
name = "web-sys"
version = "0.3.56"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c060b319f29dd25724f09a2ba1418f142f539b2be99fbf4d2d5a8f7330afb8eb"
dependencies = [
"js-sys",
"wasm-bindgen",
]
[[package]]
name = "winapi"
version = "0.3.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
dependencies = [
"winapi-i686-pc-windows-gnu",
"winapi-x86_64-pc-windows-gnu",
]
[[package]]
name = "winapi-i686-pc-windows-gnu"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
[[package]]
name = "winapi-util"
version = "0.1.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "70ec6ce85bb158151cae5e5c87f95a8e97d2c0c4b001223f33a334e3ce5de178"
dependencies = [
"winapi",
]
[[package]]
name = "winapi-x86_64-pc-windows-gnu"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"

Some files were not shown because too many files have changed in this diff Show more