Compare commits
34 commits
faster_syn
...
main
Author | SHA1 | Date | |
---|---|---|---|
|
cb409b6ffe | ||
|
b34b46fa16 | ||
|
7b747b8341 | ||
|
2c1970f664 | ||
|
63b761c0d1 | ||
|
44fa7ac416 | ||
|
8de2fa9bd4 | ||
|
407faefa6e | ||
|
1425af43cd | ||
|
c92d042c87 | ||
|
9271b20cf5 | ||
|
5e82dbc3c8 | ||
|
2cd7427f35 | ||
|
11f063cbfe | ||
|
a24d536d16 | ||
|
c5fde2802f | ||
|
13a775ed9a | ||
|
1e33c9d9e0 | ||
|
c3c04128f5 | ||
|
da55dfac7a | ||
|
9195e9cb76 | ||
|
f8d5a8ea98 | ||
|
2a9652e642 | ||
|
a6959e70e8 | ||
|
de5af2fffa | ||
|
08801ab580 | ||
|
89a0866272 | ||
|
9b6a3c8691 | ||
|
58a7a06b75 | ||
|
f428fe0169 | ||
|
931ee7e77b | ||
|
819767cc33 | ||
|
78adbc4ff9 | ||
|
1f7b109dcd |
196 changed files with 13749 additions and 11256 deletions
22
.github/workflows/ci.yaml
vendored
22
.github/workflows/ci.yaml
vendored
|
@ -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
|
||||||
|
|
18
README.md
18
README.md
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -54,6 +54,7 @@
|
||||||
|
|
||||||
nodejs
|
nodejs
|
||||||
yarn
|
yarn
|
||||||
|
deno
|
||||||
|
|
||||||
# c deps
|
# c deps
|
||||||
cmake
|
cmake
|
||||||
|
|
|
@ -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: "^_",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
|
@ -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"
|
||||||
|
|
|
@ -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
100
javascript/src/conflicts.ts
Normal 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
|
||||||
|
}
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -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 */
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
30
javascript/src/unstable_types.ts
Normal file
30
javascript/src/unstable_types.ts
Normal 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
|
|
@ -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] })
|
||||||
|
|
|
@ -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" },
|
||||||
])
|
])
|
||||||
})
|
})
|
||||||
|
|
|
@ -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"])
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -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
|
|
||||||
|
|
250
rust/automerge-c/.clang-format
Normal file
250
rust/automerge-c/.clang-format
Normal 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
|
||||||
|
...
|
||||||
|
|
8
rust/automerge-c/.gitignore
vendored
8
rust/automerge-c/.gitignore
vendored
|
@ -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
|
|
||||||
|
|
|
@ -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}
|
|
||||||
)
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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!
|
||||||
|
|
|
@ -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
|
||||||
|
|
22
rust/automerge-c/cmake/Cargo.toml.in
Normal file
22
rust/automerge-c/cmake/Cargo.toml.in
Normal 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"
|
48
rust/automerge-c/cmake/cbindgen.toml.in
Normal file
48
rust/automerge-c/cmake/cbindgen.toml.in
Normal 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"]
|
|
@ -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 */
|
||||||
|
|
183
rust/automerge-c/cmake/enum-string-functions-gen.cmake
Normal file
183
rust/automerge-c/cmake/enum-string-functions-gen.cmake
Normal 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()
|
|
@ -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.")
|
|
@ -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.")
|
35
rust/automerge-c/docs/CMakeLists.txt
Normal file
35
rust/automerge-c/docs/CMakeLists.txt
Normal 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()
|
Before Width: | Height: | Size: 1.4 KiB After Width: | Height: | Size: 1.4 KiB |
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
```
|
```
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
30
rust/automerge-c/include/automerge-c/utils/result.h
Normal file
30
rust/automerge-c/include/automerge-c/utils/result.h
Normal 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 */
|
130
rust/automerge-c/include/automerge-c/utils/stack.h
Normal file
130
rust/automerge-c/include/automerge-c/utils/stack.h
Normal 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 */
|
|
@ -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 */
|
29
rust/automerge-c/include/automerge-c/utils/string.h
Normal file
29
rust/automerge-c/include/automerge-c/utils/string.h
Normal 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 */
|
|
@ -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()
|
|
|
@ -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
|
||||||
|
|
|
@ -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,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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())),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
|
@ -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()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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
|
@ -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(),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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))
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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;
|
||||||
|
|
84
rust/automerge-c/src/index.rs
Normal file
84
rust/automerge-c/src/index.rs
Normal 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
1963
rust/automerge-c/src/item.rs
Normal file
File diff suppressed because it is too large
Load diff
401
rust/automerge-c/src/items.rs
Normal file
401
rust/automerge-c/src/items.rs
Normal 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()
|
||||||
|
}
|
|
@ -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"));
|
||||||
|
|
|
@ -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(),
|
||||||
|
}),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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
|
@ -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
|
|
||||||
}
|
|
|
@ -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()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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;
|
||||||
|
|
|
@ -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()
|
})
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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()
|
})
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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())
|
||||||
}
|
}
|
||||||
|
|
33
rust/automerge-c/src/utils/result.c
Normal file
33
rust/automerge-c/src/utils/result.c
Normal 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;
|
||||||
|
}
|
106
rust/automerge-c/src/utils/stack.c
Normal file
106
rust/automerge-c/src/utils/stack.c
Normal 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;
|
||||||
|
}
|
9
rust/automerge-c/src/utils/stack_callback_data.c
Normal file
9
rust/automerge-c/src/utils/stack_callback_data.c
Normal 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;
|
||||||
|
}
|
46
rust/automerge-c/src/utils/string.c
Normal file
46
rust/automerge-c/src/utils/string.c
Normal 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;
|
||||||
|
}
|
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
|
|
17
rust/automerge-c/test/base_state.c
Normal file
17
rust/automerge-c/test/base_state.c
Normal 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;
|
||||||
|
}
|
39
rust/automerge-c/test/base_state.h
Normal file
39
rust/automerge-c/test/base_state.h
Normal 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 */
|
119
rust/automerge-c/test/byte_span_tests.c
Normal file
119
rust/automerge-c/test/byte_span_tests.c
Normal 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);
|
||||||
|
}
|
88
rust/automerge-c/test/cmocka_utils.c
Normal file
88
rust/automerge-c/test/cmocka_utils.c
Normal 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;
|
||||||
|
}
|
|
@ -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 */
|
||||||
|
|
27
rust/automerge-c/test/doc_state.c
Normal file
27
rust/automerge-c/test/doc_state.c
Normal 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;
|
||||||
|
}
|
17
rust/automerge-c/test/doc_state.h
Normal file
17
rust/automerge-c/test/doc_state.h
Normal 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 */
|
|
@ -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);
|
||||||
|
|
148
rust/automerge-c/test/enum_string_tests.c
Normal file
148
rust/automerge-c/test/enum_string_tests.c
Normal 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);
|
||||||
|
}
|
|
@ -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;
|
|
||||||
}
|
|
|
@ -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 */
|
|
94
rust/automerge-c/test/item_tests.c
Normal file
94
rust/automerge-c/test/item_tests.c
Normal 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);
|
||||||
|
}
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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 */
|
||||||
|
|
|
@ -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
|
@ -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
|
@ -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;
|
|
||||||
}
|
|
|
@ -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 */
|
|
|
@ -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"
|
||||||
|
|
|
@ -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 */
|
||||||
|
|
857
rust/automerge-cli/Cargo.lock
generated
857
rust/automerge-cli/Cargo.lock
generated
|
@ -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
Loading…
Reference in a new issue