2019-06-24 14:36:18 +00:00
|
|
|
nim-serialization
|
|
|
|
=================
|
|
|
|
|
|
|
|
[![License: Apache](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)
|
|
|
|
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
|
2021-06-02 23:59:50 +00:00
|
|
|
![Github action](https://github.com/status-im/nim-serialization/workflows/CI/badge.svg)
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
## Introduction
|
|
|
|
|
|
|
|
The `serialization` package aims to provide a common generic and efficient
|
|
|
|
interface for marshaling Nim values to and from various serialized formats.
|
|
|
|
Individual formats are implemented in separated packages such as
|
|
|
|
[`json_serialization`](https://github.com/status-im/nim-json-serialization)
|
|
|
|
while this package provides the common interfaces shared between all of them
|
|
|
|
and the means to customize your Nim types for the purposes of serialization.
|
|
|
|
|
|
|
|
The internal mechanisms of the library allow for implementing the required
|
|
|
|
marshaling logic in highly efficient way that goes from bytes to Nim values
|
|
|
|
and vice versa without allocating any intermediate structures.
|
|
|
|
|
|
|
|
## Defining serialization formats
|
|
|
|
|
|
|
|
A serialization format is implemented through defining a `Reader` and `Writer`
|
|
|
|
type for the format and then by providing the following type declaration:
|
|
|
|
|
|
|
|
```nim
|
2021-08-05 17:00:30 +00:00
|
|
|
serializationFormat Json, # This is the name of the format.
|
|
|
|
# Most APIs provided by the library will accept
|
|
|
|
# this identifier as a required parameter.
|
|
|
|
mimeType = "application/json" # Mime type associated with the format (Optional).
|
|
|
|
|
|
|
|
Json.setReader JsonReader # The associated Reader type.
|
|
|
|
Json.setWriter JsonWriter, # The associated Writer type.
|
|
|
|
PreferredOutput = string # APIs such as `Json.encode` will return this type.
|
2019-06-24 14:36:18 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
## Common API
|
|
|
|
|
|
|
|
Most of the time, you'll be using the following high-level APIs when encoding
|
|
|
|
and decoding values:
|
|
|
|
|
2021-03-19 02:04:14 +00:00
|
|
|
#### `Format.encode(value: auto, params: varargs): Format.PreferredOutput`
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
Encodes a value in the specified format returning the preferred output type
|
|
|
|
for the format (usually `string` or `seq[byte]`). All extra params will be
|
|
|
|
forwarded without modification to the constructor of the used `Writer` type.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
|
|
|
|
```nim
|
|
|
|
assert Json.encode(@[1, 2, 3], pretty = false) == "[1, 2, 3]"
|
|
|
|
```
|
|
|
|
|
2021-12-05 14:52:47 +00:00
|
|
|
#### `Format.decode(input: openArray[byte]|string, RecordType: type, params: varargs): RecordType`
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
Decodes and returns a value of the specified `RecordType`. All params will
|
|
|
|
be forwarded without modification to the used `Reader` type. A Format-specific
|
|
|
|
descendant of `SerializationError` may be thrown in case of error.
|
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
#### `Format.saveFile(filename: string, value: auto, params: varargs)`
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
Similar to `encode`, but saves the result in a file.
|
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
#### `Format.loadFile(filename: string, RecordType: type, params: varargs): RecordType`
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
Similar to `decode`, but treats the contents of a file as an input.
|
|
|
|
|
|
|
|
#### `reader.readValue(RecordType: type): RecordType`
|
|
|
|
|
|
|
|
Reads a single value of the designated type from the stream associated with a
|
|
|
|
particular reader.
|
|
|
|
|
|
|
|
#### `writer.writeValue(value: auto)`
|
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
Encodes a single value and writes it to the output stream of a particular writer.
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
### Custom serialization of user-defined types
|
|
|
|
|
|
|
|
By default, record types will have all of their fields serialized. You can
|
2021-08-05 17:00:30 +00:00
|
|
|
alter this behavior by attaching the `dontSerialize` pragma to exclude fields.
|
2019-06-24 14:36:18 +00:00
|
|
|
The pragma `serializedFieldName(name: string)` can be used to modify the name
|
|
|
|
of the field in formats such as Json and XML.
|
|
|
|
|
|
|
|
Alternatively, if you are not able to modify the definition of a particular
|
2021-08-05 17:00:30 +00:00
|
|
|
Nim type, you can use the `setSerializedFields` macro to achieve the same
|
|
|
|
in a less intrusive way.
|
2019-06-24 14:36:18 +00:00
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
The following two definitions can be considered equivalent:
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
```nim
|
|
|
|
type
|
|
|
|
Foo = object
|
|
|
|
a: string
|
|
|
|
b {.dontSerialize.}: int
|
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
setSerializedFields Foo:
|
2019-06-24 14:36:18 +00:00
|
|
|
a
|
|
|
|
```
|
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
As you can see, `setSerializedFields` accepts a block where each serialized
|
|
|
|
field is listed on a separate line.
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
#### `customSerialization(RecordType: type, spec)`
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-08-05 17:00:30 +00:00
|
|
|
#### `totalSerializedFields(RecordType: type)`
|
2019-06-24 14:36:18 +00:00
|
|
|
|
|
|
|
Returns the number of serialized fields in the specified format.
|
|
|
|
|
|
|
|
### Implementing Readers
|
|
|
|
|
|
|
|
### Implementing Writers
|
|
|
|
|
|
|
|
## Contributing
|
|
|
|
|
|
|
|
When submitting pull requests, please add test cases for any new features
|
|
|
|
or fixes and make sure `nimble test` is still able to execute the entire
|
|
|
|
test suite successfully.
|
|
|
|
|
|
|
|
[BOUNTIES]: https://github.com/status-im/nim-confutils/issues?q=is%3Aissue+is%3Aopen+label%3Abounty
|
|
|
|
|
|
|
|
## License
|
|
|
|
|
|
|
|
Licensed and distributed under either of
|
|
|
|
|
|
|
|
* MIT license: [LICENSE-MIT](LICENSE-MIT) or http://opensource.org/licenses/MIT
|
|
|
|
|
|
|
|
or
|
|
|
|
|
|
|
|
* Apache License, Version 2.0, ([LICENSE-APACHEv2](LICENSE-APACHEv2) or http://www.apache.org/licenses/LICENSE-2.0)
|
|
|
|
|
2019-07-13 15:05:06 +00:00
|
|
|
at your option. These files may not be copied, modified, or distributed except according to those terms.
|
2019-06-24 14:36:18 +00:00
|
|
|
|