Getting Started
The @pcd/pod
package allows you to sign, verify, and manipulate data in the POD format. This guide will walk you through how to get started with POD development. The examples below assuming you’re using TypeScript, but the same steps will work from JavaScript as well.
You can find full information about the types and functions used here in the API Reference. If you prefer to read complete working code, check out the tutorial example.
Installation
To get started with PODs, you will need to install the @pcd/pod
package, using your preferred package manager:
The package is available in CJS or ESM format, and will work either in browser or in a server. The example code makes use of bigint literals, so requries a target of at least ES2020.
Packaging for a browser requires polyfill for some Node modules, including buffer
, so watch for that if you see any dependency issues.
Imports
Next, import the types and functions you need from the package. See the API Reference for everything which is available, and import what you need as you write your code. Here’s an example covering the first few samples below:
POD Entries
A POD is made up of entries (the PODEntries type), each containing a name (PODName) and a value (PODValue).You can read more details and recommendations about POD names and values later in this guide.
In TypeScript code, all values have an explicit type specifier. Let’s declare some sample entries, modeling a digital driver’s license.
Sign a POD
A POD is always signed by some issuer. To sign a new POD you need a private key, which can be any 32 bytes, as a string in either hexadecimal or Base64. These should be randomly generated in a secure way.
You can access the signature
and signerPublicKey
fields to see the results.
Once a POD is Merklized and signed, it is immutable. The root of the Merkle tree is available in the contentID
field which uniquely identifies the (unsigned) contents of this POD. The content ID is like a hash (actually the root of a Merkle tree) in that the same entries will always result in the same content ID.
Verify a Signature
If you receive a POD from an untrusted source, it’s always a good idea to verify the signature. An invalid signature will make it impossible to generate ZK proofs of the POD.
POD Contents
Inside a POD is a PODContent object which acts as a Map-like object for accessing the POD’s entries. The PODContent class also forms the Merkle tree in which the entries are cryptographically arranged to calculate the content ID.
Merklization happens lazily on-demand, with the result cached to avoid duplicated effort. The PODContent class can also generate Merkle membership proofs for individual entries, which are needed for GPC proving.
JSON Serialization
PODs, and related types like entries are not directly stringifiable due to the use of bigints, and class objects. There are supported
conversion functions for all types to/from JSON compatible types which can be used for serialization via JSON.stringify
.
You can find TypeScript types with the JSON
prefix added to the primary type: e.g. POD
to JSONPOD
, PODValue
to JSONPODValue
, etc. The conversion functions also validate that all the data is legal and well-formed.
The JSON format is optimized to be human-readable as well as short. It omits type information in cases where the types can be properly derived from standard JSON types.
Deserialization can be performed by performing the reverse conversion after calling JSON.parse
. This conversion also fully validates the input and will throw if the data is malformed.
What’s next?
- Check out the rest of the pages of this guide for deeper dives on various aspects of PODs.
- Check the Developer Resources for references to related packages.
- Read about how GPCs make ZK proofs about PODs easy, or about how the Z-API lets you access PODs and request proofs from Zupass.
- Read or watch more about PODs in both vision and technical detail on the Learning page. In particular, check out the intro and deep dive talks for more about how PODs and GPCs work.