Internet-Draft | VDAF | July 2022 |
Barnes, et al. | Expires 12 January 2023 | [Page] |
This document describes Verifiable Distributed Aggregation Functions (VDAFs), a family of multi-party protocols for computing aggregate statistics over user measurements. These protocols are designed to ensure that, as long as at least one aggregation server executes the protocol honestly, individual measurements are never seen by any server in the clear. At the same time, VDAFs allow the servers to detect if a malicious or misconfigured client submitted an input that would result in an incorrect aggregate result.¶
This note is to be removed before publishing as an RFC.¶
Discussion of this document takes place on the Crypto Forum Research Group mailing list (cfrg@ietf.org), which is archived at https://mailarchive.ietf.org/arch/search/?email_list=cfrg.¶
Source for this draft and an issue tracker can be found at https://github.com/cjpatton/vdaf.¶
This Internet-Draft is submitted in full conformance with the provisions of BCP 78 and BCP 79.¶
Internet-Drafts are working documents of the Internet Engineering Task Force (IETF). Note that other groups may also distribute working documents as Internet-Drafts. The list of current Internet-Drafts is at https://datatracker.ietf.org/drafts/current/.¶
Internet-Drafts are draft documents valid for a maximum of six months and may be updated, replaced, or obsoleted by other documents at any time. It is inappropriate to use Internet-Drafts as reference material or to cite them other than as "work in progress."¶
This Internet-Draft will expire on 12 January 2023.¶
Copyright (c) 2022 IETF Trust and the persons identified as the document authors. All rights reserved.¶
This document is subject to BCP 78 and the IETF Trust's Legal Provisions Relating to IETF Documents (https://trustee.ietf.org/license-info) in effect on the date of publication of this document. Please review these documents carefully, as they describe your rights and restrictions with respect to this document. Code Components extracted from this document must include Revised BSD License text as described in Section 4.e of the Trust Legal Provisions and are provided without warranty as described in the Revised BSD License.¶
The ubiquity of the Internet makes it an ideal platform for measurement of large-scale phenomena, whether public health trends or the behavior of computer systems at scale. There is substantial overlap, however, between information that is valuable to measure and information that users consider private.¶
For example, consider an application that provides health information to users. The operator of an application might want to know which parts of their application are used most often, as a way to guide future development of the application. Specific users' patterns of usage, though, could reveal sensitive things about them, such as which users are researching a given health condition.¶
In many situations, the measurement collector is only interested in aggregate statistics, e.g., which portions of an application are most used or what fraction of people have experienced a given disease. Thus systems that provide aggregate statistics while protecting individual measurements can deliver the value of the measurements while protecting users' privacy.¶
Most prior approaches to this problem fall under the rubric of "differential privacy (DP)" [Dwo06]. Roughly speaking, a data aggregation system that is differentially private ensures that the degree to which any individual measurement influences the value of the aggregate result can be precisely controlled. For example, in systems like RAPPOR [EPK14], each user samples noise from a well-known distribution and adds it to their input before submitting to the aggregation server. The aggregation server then adds up the noisy inputs, and because it knows the distribution from whence the noise was sampled, it can estimate the true sum with reasonable precision.¶
Differentially private systems like RAPPOR are easy to deploy and provide a useful guarantee. On its own, however, DP falls short of the strongest privacy property one could hope for. Specifically, depending on the "amount" of noise a client adds to its input, it may be possible for a curious aggregator to make a reasonable guess of the input's true value. Indeed, the more noise the clients add, the less reliable will be the server's estimate of the output. Thus systems employing DP techniques alone must strike a delicate balance between privacy and utility.¶
The ideal goal for a privacy-preserving measurement system is that of secure multi-party computation: No participant in the protocol should learn anything about an individual input beyond what it can deduce from the aggregate. In this document, we describe Verifiable Distributed Aggregation Functions (VDAFs) as a general class of protocols that achieve this goal.¶
VDAF schemes achieve their privacy goal by distributing the computation of the aggregate among a number of non-colluding aggregation servers. As long as a subset of the servers executes the protocol honestly, VDAFs guarantee that no input is ever accessible to any party besides the client that submitted it. At the same time, VDAFs are "verifiable" in the sense that malformed inputs that would otherwise garble the output of the computation can be detected and removed from the set of inputs.¶
The cost of achieving these security properties is the need for multiple servers to participate in the protocol, and the need to ensure they do not collude to undermine the VDAF's privacy guarantees. Recent implementation experience has shown that practical challenges of coordinating multiple servers can be overcome. The Prio system [CGB17] (essentially a VDAF) has been deployed in systems supporting hundreds of millions of users: The Mozilla Origin Telemetry project [OriginTelemetry] and the Exposure Notification Private Analytics collaboration among the Internet Security Research Group (ISRG), Google, Apple, and others [ENPA].¶
The VDAF abstraction laid out in Section 5 represents a class of multi-party protocols for privacy-preserving measurement proposed in the literature. These protocols vary in their operational and security considerations, sometimes in subtle but consequential ways. This document therefore has two important goals:¶
Providing higher-level protocols like [DAP] with a simple, uniform interface for accessing privacy-preserving measurement schemes, and documenting relevant operational and security bounds for that interface:¶
This document also specifies two concrete VDAF schemes, each based on a protocol from the literature.¶
The aforementioned Prio system [CGB17] allows for the privacy-preserving computation of a variety aggregate statistics. The basic idea underlying Prio is fairly simple:¶
The difficult part of this system is ensuring that the servers hold shares of a valid input, e.g., the input is an integer in a specific range. Thus Prio specifies a multi-party protocol for accomplishing this task.¶
In Section 7 we describe Prio3, a VDAF that follows the same overall framework as the original Prio protocol, but incorporates techniques introduced in [BBCGGI19] that result in significant performance gains.¶
More recently, Boneh et al. [BBCGGI21] described a protocol called Poplar
for solving the t
-heavy-hitters problem in a privacy-preserving manner. Here
each client holds a bit-string of length n
, and the goal of the aggregation
servers is to compute the set of inputs that occur at least t
times. The
core primitive used in their protocol is a specialized Distributed Point
Function (DPF) [GI14] that allows the servers to "query" their DPF shares on
any bit-string of length shorter than or equal to n
. As a result of this
query, each of the servers has an additive share of a bit indicating whether
the string is a prefix of the client's input. The protocol also specifies a
multi-party computation for verifying that at most one string among a set of
candidates is a prefix of the client's input.¶
In Section 8 we describe a VDAF called Poplar1 that implements this functionality.¶
Finally, perhaps the most complex aspect of schemes like Prio3 and Poplar1 is the process by which the client-generated measurements are prepared for aggregation. Because these constructions are based on secret sharing, the servers will be required to exchange some amount of information in order to verify the measurement is valid and can be aggregated. Depending on the construction, this process may require multiple round trips over the network.¶
There are applications in which this verification step may not be necessary, e.g., when the client's software is run a trusted execution environment. To support these applications, this document also defines Distributed Aggregation Functions (DAFs) as a simpler class of protocols that aim to provide the same privacy guarantee as VDAFs but fall short of being verifiable.¶
OPEN ISSUE Decide if we should give one or two example DAFs. There are natural variants of Prio3 and Poplar1 that might be worth describing.¶
The remainder of this document is organized as follows: Section 3 gives a brief overview of DAFs and VDAFs; Section 4 defines the syntax for DAFs; Section 5 defines the syntax for VDAFs; Section 6 defines various functionalities that are common to our constructions; Section 7 describes the Prio3 construction; Section 8 describes the Poplar1 construction; and Section 9 enumerates the security considerations for VDAFs.¶
(*) Indicates a change that breaks wire compatibility with the previous draft.¶
02:¶
01:¶
prep_next()
to
prep_shares_to_prep()
. (*)¶
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "NOT RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in BCP 14 [RFC2119] [RFC8174] when, and only when, they appear in all capitals, as shown here.¶
Algorithms in this document are written in Python 3. Type hints are used to define input and output types. A fatal error in a program (e.g., failure to parse one of the function parameters) is usually handled by raising an exception.¶
A variable with type Bytes
is a byte string. This document defines several
byte-string constants. When comprised of printable ASCII characters, they are
written as Python 3 byte-string literals (e.g., b'some constant string'
).¶
A global constant VERSION
is defined, which algorithms are free to use as
desired. Its value SHALL be b'vdaf-02'
.¶
This document describes algorithms for multi-party computations in which the parties typically communicate over a network. Wherever a quantity is defined that must be be transmitted from one party to another, this document prescribes a particular encoding of that quantity as a byte string.¶
OPEN ISSUE It might be better to not be prescriptive about how quantities are encoded on the wire. See issue #58.¶
Some common functionalities:¶
zeros(len: Unsigned) -> Bytes
returns an array of zero bytes. The length of
output
MUST be len
.¶
gen_rand(len: Unsigned) -> Bytes
returns an array of random bytes. The
length of output
MUST be len
.¶
byte(int: Unsigned) -> Bytes
returns the representation of int
as a byte
string. The value of int
MUST be in [0,256)
.¶
xor(left: Bytes, right: Bytes) -> Bytes
returns the bitwise XOR of left
and right
. An exception is raised if the inputs are not the same length.¶
I2OSP
and OS2IP
from [RFC8017], which are used, respectively, to
convert a non-negative integer to a byte string and convert a byte string to a
non-negative integer.¶
next_power_of_2(n: Unsigned) -> Unsigned
returns the smallest integer
greater than or equal to n
that is also a power of two.¶
In a DAF- or VDAF-based private measurement system, we distinguish three types of actors: Clients, Aggregators, and Collectors. The overall flow of the measurement process is as follows:¶
The Aggregators convert their input shares into "output shares".¶
Aggregators are a new class of actor relative to traditional measurement systems where clients submit measurements to a single server. They are critical for both the privacy properties of the system and, in the case of VDAFs, the correctness of the measurements obtained. The privacy properties of the system are assured by non-collusion among Aggregators, and Aggregators are the entities that perform validation of Client measurements. Thus clients trust Aggregators not to collude (typically it is required that at least one Aggregator is honest), and Collectors trust Aggregators to correctly run the protocol.¶
Within the bounds of the non-collusion requirements of a given (V)DAF instance, it is possible for the same entity to play more than one role. For example, the Collector could also act as an Aggregator, effectively using the other Aggregator(s) to augment a basic client-server protocol.¶
In this document, we describe the computations performed by the actors in this system. It is up to the higher-level protocol making use of the (V)DAF to arrange for the required information to be delivered to the proper actors in the proper sequence. In general, we assume that all communications are confidential and mutually authenticated, with the exception that Clients submitting measurements may be anonymous.¶
By way of a gentle introduction to VDAFs, this section describes a simpler class of schemes called Distributed Aggregation Functions (DAFs). Unlike VDAFs, DAFs do not provide verifiability of the computation. Clients must therefore be trusted to compute their input shares correctly. Because of this fact, the use of a DAF is NOT RECOMMENDED for most applications. See Section 9 for additional discussion.¶
A DAF scheme is used to compute a particular "aggregation function" over a set of measurements generated by Clients. Depending on the aggregation function, the Collector might select an "aggregation parameter" and disseminates it to the Aggregators. The semantics of this parameter is specific to the aggregation function, but in general it is used to represent the set of "queries" that can be made on the measurement set. For example, the aggregation parameter is used to represent the candidate prefixes in Poplar1 Section 8.¶
Execution of a DAF has four distinct stages:¶
Sharding and Preparation are done once per measurement. Aggregation and Unsharding are done over a batch of measurements (more precisely, over the recovered output shares).¶
A concrete DAF specifies an algorithm for the computation needed in each of these stages. The interface of each algorithm is defined in the remainder of this section. In addition, a concrete DAF defines the associated constants and types enumerated in the following table.¶
Parameter | Description |
---|---|
SHARES
|
Number of input shares into which each measurement is sharded |
Measurement
|
Type of each measurement |
AggParam
|
Type of aggregation parameter |
OutShare
|
Type of each output share |
AggResult
|
Type of the aggregate result |
These types define some of the inputs and outputs of DAF methods at various
stages of the computation. Observe that only the measurements, output shares,
the aggregate result, and the aggregation parameter have an explicit type. All
other values --- in particular, the input shares and the aggregate shares ---
have type Bytes
and are treated as opaque byte strings. This is because these
values must be transmitted between parties over a network.¶
OPEN ISSUE It might be cleaner to define a type for each value, then have that type implement an encoding where necessary. This way each method parameter has a meaningful type hint. See issue#58.¶
In order to protect the privacy of its measurements, a DAF Client shards its
measurements into a sequence of input shares. The measurement_to_input_shares
method is used for this purpose.¶
Daf.measurement_to_input_shares(input: Measurement) -> (Bytes, Vec[Bytes])
is the randomized input-distribution algorithm run by each Client. It consumes
the measurement and produces a "public share", distributed to each of the
Aggregators, and a corresponding sequence of input shares, one for each
Aggregator. The length of the output vector MUST be SHARES
.¶
Once an Aggregator has received the public share and one of the input shares, the next step is to prepare the input share for aggregation. This is accomplished using the following algorithm:¶
Daf.prep(agg_id: Unsigned, agg_param: AggParam, public_share: Bytes,
input_share: Bytes) -> OutShare
is the deterministic preparation algorithm.
It takes as input the public share and one of the input shares generated by a
Client, the Aggregator's unique identifier, and the aggregation parameter
selected by the Collector and returns an output share.¶
The protocol in which the DAF is used MUST ensure that the Aggregator's
identifier is equal to the integer in range [0, SHARES)
that matches the index
of input_share
in the sequence of input shares output by the Client.¶
Once an Aggregator holds output shares for a batch of measurements (where batches are defined by the application), it combines them into a share of the desired aggregate result:¶
Daf.out_shares_to_agg_share(agg_param: AggParam, out_shares: Vec[OutShare])
-> agg_share: Bytes
is the deterministic aggregation algorithm. It is run by
each Aggregator a set of recovered output shares.¶
For simplicity, we have written this algorithm in a "one-shot" form, where all output shares for a batch are provided at the same time. Many DAFs may also support a "streaming" form, where shares are processed one at a time.¶
OPEN ISSUE It may be worthwhile to explicitly define the "streaming" API. See issue#47.¶
After the Aggregators have aggregated a sufficient number of output shares, each sends its aggregate share to the Collector, who runs the following algorithm to recover the following output:¶
Daf.agg_shares_to_result(agg_param: AggParam,
agg_shares: Vec[Bytes], num_measurements: Unsigned) -> AggResult
is
run by the Collector in order to compute the aggregate result from
the Aggregators' shares. The length of agg_shares
MUST be SHARES
.
num_measurements
is the number of measurements that contributed to
each of the aggregate shares. This algorithm is deterministic.¶
QUESTION Maybe the aggregation algorithms should be randomized in order to allow the Aggregators (or the Collector) to add noise for differential privacy. (See the security considerations of [DAP].) Or is this out-of-scope of this document? See https://github.com/ietf-wg-ppm/ppm-specification/issues/19.¶
Securely executing a DAF involves emulating the following procedure.¶
The inputs to this procedure are the same as the aggregation function computed by the DAF: An aggregation parameter and a sequence of measurements. The procedure prescribes how a DAF is executed in a "benign" environment in which there is no adversary and the messages are passed among the protocol participants over secure point-to-point channels. In reality, these channels need to be instantiated by some "wrapper protocol", such as [DAP], that realizes these channels using suitable cryptographic mechanisms. Moreover, some fraction of the Aggregators (or Clients) may be malicious and diverge from their prescribed behaviors. Section 9 describes the execution of the DAF in various adversarial environments and what properties the wrapper protocol needs to provide in each.¶
Like DAFs described in the previous section, a VDAF scheme is used to compute a particular aggregation function over a set of Client-generated measurements. Evaluation of a VDAF involves the same four stages as for DAFs: Sharding, Preparation, Aggregation, and Unsharding. However, the Preparation stage will require interaction among the Aggregators in order to facilitate verifiability of the computation's correctness. Accommodating this interaction will require syntactic changes.¶
Overall execution of a VDAF comprises the following stages:¶
In contrast to DAFs, the Preparation stage for VDAFs now performs an additional task: Verification of the validity of the recovered output shares. This process ensures that aggregating the output shares will not lead to a garbled aggregate result.¶
The remainder of this section defines the VDAF interface. The attributes are listed in Table 2 are defined by each concrete VDAF.¶
Parameter | Description |
---|---|
VERIFY_KEY_SIZE
|
Size (in bytes) of the verification key (Section 5.2) |
ROUNDS
|
Number of rounds of communication during the Preparation stage (Section 5.2) |
SHARES
|
Number of input shares into which each measurement is sharded (Section 5.1) |
Measurement
|
Type of each measurement |
AggParam
|
Type of aggregation parameter |
Prep
|
State of each Aggregator during Preparation (Section 5.2) |
OutShare
|
Type of each output share |
AggResult
|
Type of the aggregate result |
Similarly to DAFs (see {[sec-daf}}), any output of a VDAF method that must be transmitted from one party to another is treated as an opaque byte string. All other quantities are given a concrete type.¶
OPEN ISSUE It might be cleaner to define a type for each value, then have that type implement an encoding where necessary. See issue#58.¶
Sharding is syntactically identical to DAFs (cf. Section 4.1):¶
Vdaf.measurement_to_input_shares(measurement: Measurement) -> (Bytes,
Vec[Bytes])
is the randomized input-distribution algorithm run by each
Client. It consumes the measurement and produces a public share, distributed
to each of Aggregators, and the corresponding sequence of input shares, one
for each Aggregator. Depending on the VDAF, the input shares may encode
additional information used to verify the recovered output shares (e.g., the
"proof shares" in Prio3 Section 7). The length of the output vector MUST be
SHARES
.¶
To recover and verify output shares, the Aggregators interact with one another
over ROUNDS
rounds. Prior to each round, each Aggregator constructs an
outbound message. Next, the sequence of outbound messages is combined into a
single message, called a "preparation message". (Each of the outbound messages
are called "preparation-message shares".) Finally, the preparation message is
distributed to the Aggregators to begin the next round.¶
An Aggregator begins the first round with its input share and it begins each subsequent round with the previous preparation message. Its output in the last round is its output share and its output in each of the preceding rounds is a preparation-message share.¶
This process involves a value called the "aggregation parameter" used to map the input shares to output shares. The Aggregators need to agree on this parameter before they can begin preparing inputs for aggregation.¶
To facilitate the preparation process, a concrete VDAF implements the following class methods:¶
Vdaf.prep_init(verify_key: Bytes, agg_id: Unsigned, agg_param: AggParam,
nonce: Bytes, public_share: Bytes, input_share: Bytes) -> Prep
is the
deterministic preparation-state initialization algorithm run by each
Aggregator to begin processing its input share into an output share. Its
inputs are the shared verification key (verify_key
), the Aggregator's unique
identifier (agg_id
), the aggregation parameter (agg_param
), the nonce
provided by the environment (nonce
, see Figure 7), and the public share
(public_share
) and one of the input shares generated by the client
(input_share
). Its output is the Aggregator's initial preparation state.¶
The length of verify_key
MUST be VERIFY_KEY_SIZE
. It is up to the high
level protocol in which the VDAF is used to arrange for the distribution of
the verification key among the Aggregators prior to the start of this phase of
VDAF evaluation.¶
OPEN ISSUE What security properties do we need for this key exchange? See issue#18.¶
Protocols using the VDAF MUST ensure that the Aggregator's identifier is equal
to the integer in range [0, SHARES)
that matches the index of input_share
in the sequence of input shares output by the Client. In addition, protocols
MUST ensure that public share consumed by each of the Aggregators is
identical. This is security critical for VDAFs such as Poplar1 that require an
extractable distributed point function. (See Section 8 for details.)¶
Vdaf.prep_next(prep: Prep, inbound: Optional[Bytes]) -> Union[Tuple[Prep,
Bytes], OutShare]
is the deterministic preparation-state update algorithm run
by each Aggregator. It updates the Aggregator's preparation state (prep
) and
returns either its next preparation state and its message share for the
current round or, if this is the last round, its output share. An exception is
raised if a valid output share could not be recovered. The input of this
algorithm is the inbound preparation message or, if this is the first round,
None
.¶
Vdaf.prep_shares_to_prep(agg_param: AggParam, prep_shares: Vec[Bytes]) ->
Bytes
is the deterministic preparation-message pre-processing algorithm. It
combines the preparation-message shares generated by the Aggregators in the
previous round into the preparation message consumed by each in the next
round.¶
In effect, each Aggregator moves through a linear state machine with ROUNDS+1
states. The Aggregator enters the first state on using the initialization
algorithm, and the update algorithm advances the Aggregator to the next state.
Thus, in addition to defining the number of rounds (ROUNDS
), a VDAF instance
defines the state of the Aggregator after each round.¶
TODO Consider how to bake this "linear state machine" condition into the syntax. Given that Python 3 is used as our pseudocode, it's easier to specify the preparation state using a class.¶
The preparation-state update accomplishes two tasks: recovery of output shares from the input shares and ensuring that the recovered output shares are valid. The abstraction boundary is drawn so that an Aggregator only recovers an output share if it is deemed valid (at least, based on the Aggregator's view of the protocol). Another way to draw this boundary would be to have the Aggregators recover output shares first, then verify that they are valid. However, this would allow the possibility of misusing the API by, say, aggregating an invalid output share. Moreover, in protocols like Prio+ [AGJOP21] based on oblivious transfer, it is necessary for the Aggregators to interact in order to recover aggregatable output shares at all.¶
Note that it is possible for a VDAF to specify ROUNDS == 0
, in which case each
Aggregator runs the preparation-state update algorithm once and immediately
recovers its output share without interacting with the other Aggregators.
However, most, if not all, constructions will require some amount of interaction
in order to ensure validity of the output shares (while also maintaining
privacy).¶
OPEN ISSUE accommodating 0-round VDAFs may require syntax changes if, for example, public keys are required. On the other hand, we could consider defining this class of schemes as a different primitive. See issue#77.¶
VDAF Aggregation is identical to DAF Aggregation (cf. Section 4.3):¶
Vdaf.out_shares_to_agg_share(agg_param: AggParam, out_shares: Vec[OutShare])
-> agg_share: Bytes
is the deterministic aggregation algorithm. It is run by
each Aggregator over the output shares it has computed over a batch of
measurement inputs.¶
The data flow for this stage is illustrated in Figure 3. Here again, we have the aggregation algorithm in a "one-shot" form, where all shares for a batch are provided at the same time. VDAFs typically also support a "streaming" form, where shares are processed one at a time.¶
VDAF Unsharding is identical to DAF Unsharding (cf. Section 4.4):¶
Vdaf.agg_shares_to_result(agg_param: AggParam,
agg_shares: Vec[Bytes], num_measurements: Unsigned) -> AggResult
is
run by the Collector in order to compute the aggregate result from
the Aggregators' shares. The length of agg_shares
MUST be SHARES
.
num_measurements
is the number of measurements that contributed to
each of the aggregate shares. This algorithm is deterministic.¶
Secure execution of a VDAF involves simulating the following procedure.¶
The inputs to this algorithm are the aggregation parameter, a list of measurements, and a nonce for each measurement. This document does not specify how the nonces are chosen, but security requires that the nonces be unique. See Section 9 for details. As explained in Section 4.5, the secure execution of a VDAF requires the application to instantiate secure channels between each of the protocol participants.¶
This section describes the primitives that are common to the VDAFs specified in this document.¶
Both Prio3 and Poplar1 use finite fields of prime order. Finite field
elements are represented by a class Field
with the following associated
parameters:¶
MODULUS: Unsigned
is the prime modulus that defines the field.¶
ENCODED_SIZE: Unsigned
is the number of bytes used to encode a field element
as a byte string.¶
A concrete Field
also implements the following class methods:¶
Field.zeros(length: Unsigned) -> output: Vec[Field]
returns a vector of
zeros. The length of output
MUST be length
.¶
Field.rand_vec(length: Unsigned) -> output: Vec[Field]
returns a vector of
random field elements. The length of output
MUST be length
.¶
A field element is an instance of a concrete Field
. The concrete class defines
the usual arithmetic operations on field elements. In addition, it defines the
following instance method for converting a field element to an unsigned integer:¶
elem.as_unsigned() -> Unsigned
returns the integer representation of
field element elem
.¶
Likewise, each concrete Field
implements a constructor for converting an
unsigned integer into a field element:¶
Field(integer: Unsigned)
returns integer
represented as a field element.
The value of integer
MUST be less than Field.MODULUS
.¶
Finally, each concrete Field
has two derived class methods, one for encoding
a vector of field elements as a byte string and another for decoding a vector of
field elements.¶
The following auxiliary functions on vectors of field elements are used in the remainder of this document. Note that an exception is raised by each function if the operands are not the same length.¶
Some VDAFs require fields that are suitable for efficient computation of the discrete Fourier transform. (One example is Prio3 (Section 7) when instantiated with the generic FLP of Section 7.3.3.) Specifically, a field is said to be "FFT-friendly" if, in addition to satisfying the interface described in Section 6.1, it implements the following method:¶
Field.gen() -> Field
returns the generator of a large subgroup of the
multiplicative group.¶
FFT-friendly fields also define the following parameter:¶
GEN_ORDER: Unsigned
is the order of a multiplicative subgroup generated by
Field.gen()
. This value MUST be a power of 2.¶
The tables below define finite fields used in the remainder of this document.¶
Parameter | Value |
---|---|
MODULUS | 2^32 * 4294967295 + 1 |
ENCODED_SIZE | 8 |
Generator | 7^4294967295 |
GEN_ORDER | 2^32 |
Parameter | Value |
---|---|
MODULUS | 2^66 * 4611686018427387897 + 1 |
ENCODED_SIZE | 16 |
Generator | 7^4611686018427387897 |
GEN_ORDER | 2^66 |
OPEN ISSUE We currently use big-endian for encoding field elements. However,
for implementations of GF(2^255-19)
, little endian is more common. See
issue#90.¶
A pseudorandom generator (PRG) is used to expand a short, (pseudo)random seed into a long string of pseudorandom bits. A PRG suitable for this document implements the interface specified in this section. Concrete constructions are described in the subsections that follow.¶
PRGs are defined by a class Prg
with the following associated parameter:¶
SEED_SIZE: Unsigned
is the size (in bytes) of a seed.¶
A concrete Prg
implements the following class method:¶
Prg(seed: Bytes, info: Bytes)
constructs an instance of Prg
from the given
seed and info string. The seed MUST be of length SEED_SIZE
and MUST be
generated securely (i.e., it is either the output of gen_rand
or a previous
invocation of the PRG). The info string is used for domain separation.¶
prg.next(length: Unsigned)
returns the next length
bytes of output of PRG.
If the seed was securely generated, the output can be treated as pseudorandom.¶
Each Prg
has two derived class methods. The first is used to derive a fresh
seed from an existing one. The second is used to compute a sequence of
pseudorandom field elements. For each method, the seed MUST be of length
SEED_SIZE
and MUST be generated securely (i.e., it is either the output of
gen_rand
or a previous invocation of the PRG).¶
OPEN ISSUE Phillipp points out that a fixed-key mode of AES may be more performant (https://eprint.iacr.org/2019/074.pdf). See issue#32.¶
Our first construction, PrgAes128
, converts a blockcipher, namely AES-128,
into a PRG. Seed expansion involves two steps. In the first step, CMAC
[RFC4493] is applied to the seed and info string to get a fresh key. In the
second step, the fresh key is used in CTR-mode to produce a key stream for
generating the output. A fixed initialization vector (IV) is used.¶
NOTE This construction has not undergone significant security analysis.¶
This section describes Prio3, a VDAF for Prio [CGB17]. Prio is suitable for a wide variety of aggregation functions, including (but not limited to) sum, mean, standard deviation, estimation of quantiles (e.g., median), and linear regression. In fact, the scheme described in this section is compatible with any aggregation function that has the following structure:¶
At a high level, Prio3 distributes this computation as follows. Each Client first shards its measurement by first encoding it, then splitting the vector into secret shares and sending a share to each Aggregator. Next, in the preparation phase, the Aggregators carry out a multi-party computation to determine if their shares correspond to a valid input (as determined by the arithmetic circuit). This computation involves a "proof" of validity generated by the Client. Next, each Aggregator sums up its input shares locally. Finally, the Collector sums up the aggregate shares and computes the aggregate result.¶
This VDAF does not have an aggregation parameter. Instead, the output share is derived from the input share by applying a fixed map. See Section 8 for an example of a VDAF that makes meaningful use of the aggregation parameter.¶
As the name implies, Prio3 is a descendant of the original Prio construction. A second iteration was deployed in the [ENPA] system, and like the VDAF described here, the ENPA system was built from techniques introduced in [BBCGGI19] that significantly improve communication cost. That system was specialized for a particular aggregation function; the goal of Prio3 is to provide the same level of generality as the original construction.¶
The core component of Prio3 is a "Fully Linear Proof (FLP)" system. Introduced by [BBCGGI19], the FLP encapsulates the functionality required for encoding and validating inputs. Prio3 can be thought of as a transformation of a particular class of FLPs into a VDAF.¶
The remainder of this section is structured as follows. The syntax for FLPs is described in Section 7.1. The generic transformation of an FLP into Prio3 is specified in Section 7.2. Next, a concrete FLP suitable for any validity circuit is specified in Section 7.3. Finally, instantiations of Prio3 for various types of measurements are specified in Section 7.4. Test vectors can be found in Appendix "Test Vectors".¶
Conceptually, an FLP is a two-party protocol executed by a prover and a verifier. In actual use, however, the prover's computation is carried out by the Client, and the verifier's computation is distributed among the Aggregators. The Client generates a "proof" of its input's validity and distributes shares of the proof to the Aggregators. Each Aggregator then performs some a computation on its input share and proof share locally and sends the result to the other Aggregators. Combining the exchanged messages allows each Aggregator to decide if it holds a share of a valid input. (See Section 7.2 for details.)¶
As usual, we will describe the interface implemented by a concrete FLP in terms
of an abstract base class Flp
that specifies the set of methods and parameters
a concrete FLP must provide.¶
The parameters provided by a concrete FLP are listed in Table 6.¶
Parameter | Description |
---|---|
PROVE_RAND_LEN
|
Length of the prover randomness, the number of random field elements consumed by the prover when generating a proof |
QUERY_RAND_LEN
|
Length of the query randomness, the number of random field elements consumed by the verifier |
JOINT_RAND_LEN
|
Length of the joint randomness, the number of random field elements consumed by both the prover and verifier |
INPUT_LEN
|
Length of the encoded measurement (Section 7.1.1) |
OUTPUT_LEN
|
Length of the aggregatable output (Section 7.1.1) |
PROOF_LEN
|
Length of the proof |
VERIFIER_LEN
|
Length of the verifier message generated by querying the input and proof |
Measurement
|
Type of the measurement |
AggResult
|
Type of the aggregate result |
Field
|
As defined in (Section 6.1) |
An FLP specifies the following algorithms for generating and verifying proofs of validity (encoding is described below in Section 7.1.1):¶
Flp.prove(input: Vec[Field], prove_rand: Vec[Field], joint_rand: Vec[Field])
-> Vec[Field]
is the deterministic proof-generation algorithm run by the
prover. Its inputs are the encoded input, the "prover randomness"
prove_rand
, and the "joint randomness" joint_rand
. The prover randomness is
used only by the prover, but the joint randomness is shared by both the prover
and verifier.¶
Flp.query(input: Vec[Field], proof: Vec[Field], query_rand: Vec[Field],
joint_rand: Vec[Field], num_shares: Unsigned) -> Vec[Field]
is the
query-generation algorithm run by the verifier. This is used to "query" the
input and proof. The result of the query (i.e., the output of this function)
is called the "verifier message". In addition to the input and proof, this
algorithm takes as input the query randomness query_rand
and the joint
randomness joint_rand
. The former is used only by the verifier. num_shares
specifies how many input and proof shares were generated.¶
Flp.decide(verifier: Vec[Field]) -> Bool
is the deterministic decision
algorithm run by the verifier. It takes as input the verifier message and
outputs a boolean indicating if the input from which it was generated is
valid.¶
Our application requires that the FLP is "fully linear" in the sense defined in [BBCGGI19]. As a practical matter, what this property implies is that, when run on a share of the input and proof, the query-generation algorithm outputs a share of the verifier message. Furthermore, the "strong zero-knowledge" property of the FLP system ensures that the verifier message reveals nothing about the input's validity. Therefore, to decide if an input is valid, the Aggregators will run the query-generation algorithm locally, exchange verifier shares, combine them to recover the verifier message, and run the decision algorithm.¶
The query-generation algorithm includes a parameter num_shares
that specifies
the number of shares of the input and proof that were generated. If these data
are not secret shared, then num_shares == 1
. This parameter is useful for
certain FLP constructions. For example, the FLP in Section 7.3 is defined in
terms of an arithmetic circuit; when the circuit contains constants, it is
sometimes necessary to normalize those constants to ensure that the circuit's
output, when run on a valid input, is the same regardless of the number of
shares.¶
An FLP is executed by the prover and verifier as follows:¶
The proof system is constructed so that, if input
is a valid input, then
run_flp(Flp, input, 1)
always returns True
. On the other hand, if input
is
invalid, then as long as joint_rand
and query_rand
are generated uniform
randomly, the output is False
with overwhelming probability.¶
We remark that [BBCGGI19] defines a much larger class of fully linear proof systems than we consider here. In particular, what is called an "FLP" here is called a 1.5-round, public-coin, interactive oracle proof system in their paper.¶
The type of measurement being aggregated is defined by the FLP. Hence, the FLP also specifies a method of encoding raw measurements as a vector of field elements:¶
Flp.encode(measurement: Measurement) -> Vec[Field]
encodes a raw measurement
as a vector of field elements. The return value MUST be of length INPUT_LEN
.¶
For some FLPs, the encoded input also includes redundant field elements that are
useful for checking the proof, but which are not needed after the proof has been
checked. An example is the "integer sum" data type from [CGB17] in which an
integer in range [0, 2^k)
is encoded as a vector of k
field elements (this
type is also defined in Section 7.4.2). After consuming this vector,
all that is needed is the integer it represents. Thus the FLP defines an
algorithm for truncating the input to the length of the aggregated output:¶
Flp.truncate(input: Vec[Field]) -> Vec[Field]
maps an encoded input to an
aggregatable output. The length of the input MUST be INPUT_LEN
and the length
of the output MUST be OUTPUT_LEN
.¶
Once the aggregate shares have been computed and combined together, their sum can be converted into the aggregate result. This could be a projection from the FLP's field to the integers, or it could include additional post-processing.¶
Flp.decode(output: Vec[Field], num_measurements: Unsigned) -> AggResult
maps a sum of aggregate shares to an aggregate result. The length of the
input MUST be OUTPUT_LEN
. num_measurements
is the number of measurements
that contributed to the aggregated output.¶
We remark that, taken together, these three functionalities correspond roughly to the notion of "Affine-aggregatable encodings (AFEs)" from [CGB17].¶
This section specifies Prio3
, an implementation of the Vdaf
interface
(Section 5). It has two generic parameters: an Flp
(Section 7.1) and a Prg
(Section 6.2). The associated constants and types required by the Vdaf
interface
are defined in Table 7. The methods required for sharding, preparation,
aggregation, and unsharding are described in the remaining subsections.¶
Parameter | Value |
---|---|
VERIFY_KEY_SIZE
|
Prg.SEED_SIZE
|
ROUNDS
|
1
|
SHARES
|
in [2, 255)
|
Measurement
|
Flp.Measurement
|
AggParam
|
None
|
Prep
|
Tuple[Vec[Flp.Field], Optional[Bytes], Bytes]
|
OutShare
|
Vec[Flp.Field]
|
AggResult
|
Flp.AggResult
|
This section describes the process of recovering output shares from the input shares. The high-level idea is that each Aggregator first queries its input and proof share locally, then exchanges its verifier share with the other Aggregators. The verifier shares are then combined into the verifier message, which is used to decide whether to accept.¶
In addition, the Aggregators must ensure that they have all used the same joint randomness for the query-generation algorithm. The joint randomness is generated by a PRG seed. Each Aggregator derives an XOR secret share of this seed from its input share and the "blind" generated by the client. Thus, before running the query-generation algorithm, it must first gather the XOR secret shares derived by the other Aggregators.¶
In order to avoid extra round of communication, the Client sends each Aggregator a "hint" equal to the XOR of the other Aggregators' shares of the joint randomness seed. This leaves open the possibility that the Client cheated by, say, forcing the Aggregators to use joint randomness that biases the proof check procedure some way in its favor. To mitigate this, the Aggregators also check that they have all computed the same joint randomness seed before accepting their output shares. To do so, they exchange their XOR shares of the PRG seed along with their verifier shares.¶
NOTE This optimization somewhat diverges from Section 6.2.3 of [BBCGGI19]. Security analysis is needed.¶
The algorithms required for preparation are defined as follows. These algorithms make use of encoding and decoding methods defined in Section 7.2.5.¶
Aggregating a set of output shares is simply a matter of adding up the vectors element-wise.¶
To unshard a set of aggregate shares, the Collector first adds up the vectors element-wise. It then converts each element of the vector into an integer.¶
This section describes an FLP based on the construction from in [BBCGGI19], Section 4.2. We begin in Section 7.3.1 with an overview of their proof system and the extensions to their proof system made here. The construction is specified in Section 7.3.3.¶
OPEN ISSUE We're not yet sure if specifying this general-purpose FLP is desirable. It might be preferable to specify specialized FLPs for each data type that we want to standardize, for two reasons. First, clear and concise specifications are likely easier to write for specialized FLPs rather than the general one. Second, we may end up tailoring each FLP to the measurement type in a way that improves performance, but breaks compatibility with the general-purpose FLP.¶
In any case, we can't make this decision until we know which data types to standardize, so for now, we'll stick with the general-purpose construction. The reference implementation can be found at https://github.com/cfrg/draft-irtf-cfrg-vdaf/tree/main/poc.¶
OPEN ISSUE Chris Wood points out that the this section reads more like a paper than a standard. Eventually we'll want to work this into something that is readily consumable by the CFRG.¶
In the proof system of [BBCGGI19], validity is defined via an arithmetic circuit evaluated over the input: If the circuit output is zero, then the input is deemed valid; otherwise, if the circuit output is non-zero, then the input is deemed invalid. Thus the goal of the proof system is merely to allow the verifier to evaluate the validity circuit over the input. For our application (Section 7), this computation is distributed among multiple Aggregators, each of which has only a share of the input.¶
Suppose for a moment that the validity circuit C
is affine, meaning its only
operations are addition and multiplication-by-constant. In particular, suppose
the circuit does not contain a multiplication gate whose operands are both
non-constant. Then to decide if an input x
is valid, each Aggregator could
evaluate C
on its share of x
locally, broadcast the output share to its
peers, then combine the output shares locally to recover C(x)
. This is true
because for any SHARES
-way secret sharing of x
it holds that¶
C(x_shares[0] + ... + x_shares[SHARES-1]) = C(x_shares[0]) + ... + C(x_shares[SHARES-1])¶
(Note that, for this equality to hold, it may be necessary to scale any
constants in the circuit by SHARES
.) However this is not the case if C
is
not-affine (i.e., it contains at least one multiplication gate whose operands
are non-constant). In the proof system of [BBCGGI19], the proof is designed to
allow the (distributed) verifier to compute the non-affine operations using only
linear operations on (its share of) the input and proof.¶
To make this work, the proof system is restricted to validity circuits that
exhibit a special structure. Specifically, an arithmetic circuit with "G-gates"
(see [BBCGGI19], Definition 5.2) is composed of affine gates and any number of
instances of a distinguished gate G
, which may be non-affine. We will refer to
this class of circuits as 'gadget circuits' and to G
as the "gadget".¶
As an illustrative example, consider a validity circuit C
that recognizes the
set L = set([0], [1])
. That is, C
takes as input a length-1 vector x
and
returns 0 if x[0]
is in [0,2)
and outputs something else otherwise. This
circuit can be expressed as the following degree-2 polynomial:¶
C(x) = (x[0] - 1) * x[0] = x[0]^2 - x[0]¶
This polynomial recognizes L
because x[0]^2 = x[0]
is only true if x[0] ==
0
or x[0] == 1
. Notice that the polynomial involves a non-affine operation,
x[0]^2
. In order to apply [BBCGGI19], Theorem 4.3, the circuit needs to be
rewritten in terms of a gadget that subsumes this non-affine operation. For
example, the gadget might be multiplication:¶
Mul(left, right) = left * right¶
The validity circuit can then be rewritten in terms of Mul
like so:¶
C(x[0]) = Mul(x[0], x[0]) - x[0]¶
The proof system of [BBCGGI19] allows the verifier to evaluate each instance
of the gadget (i.e., Mul(x[0], x[0])
in our example) using a linear function
of the input and proof. The proof is constructed roughly as follows. Let C
be
the validity circuit and suppose the gadget is arity-L
(i.e., it has L
input
wires.). Let wire[j-1,k-1]
denote the value of the j
th wire of the k
th
call to the gadget during the evaluation of C(x)
. Suppose there are M
such
calls and fix distinct field elements alpha[0], ..., alpha[M-1]
. (We will
require these points to have a special property, as we'll discuss in
Section 7.3.1.1; but for the moment it is only important
that they are distinct.)¶
The prover constructs from wire
and alpha
a polynomial that, when evaluated
at alpha[k-1]
, produces the output of the k
th call to the gadget. Let us
call this the "gadget polynomial". Polynomial evaluation is linear, which means
that, in the distributed setting, the Client can disseminate additive shares of
the gadget polynomial that the Aggregators then use to compute additive shares
of each gadget output, allowing each Aggregator to compute its share of C(x)
locally.¶
There is one more wrinkle, however: It is still possible for a malicious prover
to produce a gadget polynomial that would result in C(x)
being computed
incorrectly, potentially resulting in an invalid input being accepted. To
prevent this, the verifier performs a probabilistic test to check that the
gadget polynomial is well-formed. This test, and the procedure for constructing
the gadget polynomial, are described in detail in Section 7.3.3.¶
The FLP described in the next section extends the proof system [BBCGGI19], Section 4.2 in three ways.¶
First, the validity circuit in our construction includes an additional, random
input (this is the "joint randomness" derived from the input shares in Prio3;
see Section 7.2). This allows for circuit optimizations that trade a
small soundness error for a shorter proof. For example, consider a circuit that
recognizes the set of length-N
vectors for which each element is either one or
zero. A deterministic circuit could be constructed for this language, but it
would involve a large number of multiplications that would result in a large
proof. (See the discussion in [BBCGGI19], Section 5.2 for details). A much
shorter proof can be constructed for the following randomized circuit:¶
C(inp, r) = r * Range2(inp[0]) + ... + r^N * Range2(inp[N-1])¶
(Note that this is a special case of [BBCGGI19], Theorem 5.2.) Here inp
is
the length-N
input and r
is a random field element. The gadget circuit
Range2
is the "range-check" polynomial described above, i.e., Range2(x) = x^2 -
x
. The idea is that, if inp
is valid (i.e., each inp[j]
is in [0,2)
),
then the circuit will evaluate to 0 regardless of the value of r
; but if
inp[j]
is not in [0,2)
for some j
, the output will be non-zero with high
probability.¶
The second extension implemented by our FLP allows the validity circuit to
contain multiple gadget types. (This generalization was suggested in
[BBCGGI19], Remark 4.5.) For example, the following circuit is allowed, where
Mul
and Range2
are the gadgets defined above (the input has length N+1
):¶
C(inp, r) = r * Range2(inp[0]) + ... + r^N * Range2(inp[N-1]) + \ 2^0 * inp[0] + ... + 2^(N-1) * inp[N-1] - \ Mul(inp[N], inp[N])¶
Finally, [BBCGGI19], Theorem 4.3 makes no restrictions on the choice of the
fixed points alpha[0], ..., alpha[M-1]
, other than to require that the points
are distinct. In this document, the fixed points are chosen so that the gadget
polynomial can be constructed efficiently using the Cooley-Tukey FFT ("Fast
Fourier Transform") algorithm. Note that this requires the field to be
"FFT-friendly" as defined in Section 6.1.2.¶
The FLP described in Section 7.3.3 is defined in terms of a
validity circuit Valid
that implements the interface described here.¶
A concrete Valid
defines the following parameters:¶
Parameter | Description |
---|---|
GADGETS
|
A list of gadgets |
GADGET_CALLS
|
Number of times each gadget is called |
INPUT_LEN
|
Length of the input |
OUTPUT_LEN
|
Length of the aggregatable output |
JOINT_RAND_LEN
|
Length of the random input |
Measurement
|
The type of measurement |
AggResult
|
Type of the aggregate result |
Field
|
An FFT-friendly finite field as defined in Section 6.1.2 |
Each gadget G
in GADGETS
defines a constant DEGREE
that specifies the
circuit's "arithmetic degree". This is defined to be the degree of the
polynomial that computes it. For example, the Mul
circuit in
Section 7.3.1 is defined by the polynomial Mul(x) = x * x
, which
has degree 2
. Hence, the arithmetic degree of this gadget is 2
.¶
Each gadget also defines a parameter ARITY
that specifies the circuit's arity
(i.e., the number of input wires).¶
A concrete Valid
provides the following methods for encoding a measurement as
an input vector, truncating an input vector to the length of an aggregatable
output, and converting an aggregated output to an aggregate result:¶
Valid.encode(measurement: Measurement) -> Vec[Field]
returns a vector of
length INPUT_LEN
representing a measurement.¶
Valid.truncate(input: Vec[Field]) -> Vec[Field]
returns a vector of length
OUTPUT_LEN
representing an aggregatable output.¶
Valid.decode(output: Vec[Field], num_measurements: Unsigned) -> AggResult
returns an aggregate result.¶
Finally, the following class methods are derived for each concrete Valid
:¶
This section specifies FlpGeneric
, an implementation of the Flp
interface
(Section 7.1). It has as a generic parameter a validity circuit Valid
implementing
the interface defined in Section 7.3.2.¶
NOTE A reference implementation can be found in https://github.com/cfrg/draft-irtf-cfrg-vdaf/blob/main/poc/flp_generic.sage.¶
The FLP parameters for FlpGeneric
are defined in Table 9. The
required methods for generating the proof, generating the verifier, and deciding
validity are specified in the remaining subsections.¶
In the remainder, we let [n]
denote the set {1, ..., n}
for positive integer
n
. We also define the following constants:¶
Parameter | Value |
---|---|
PROVE_RAND_LEN
|
Valid.prove_rand_len() (see Section 7.3.2) |
QUERY_RAND_LEN
|
Valid.query_rand_len() (see Section 7.3.2) |
JOINT_RAND_LEN
|
Valid.JOINT_RAND_LEN
|
INPUT_LEN
|
Valid.INPUT_LEN
|
OUTPUT_LEN
|
Valid.OUTPUT_LEN
|
PROOF_LEN
|
Valid.proof_len() (see Section 7.3.2) |
VERIFIER_LEN
|
Valid.verifier_len() (see Section 7.3.2) |
Measurement
|
Valid.Measurement
|
Field
|
Valid.Field
|
On input inp
, prove_rand
, and joint_rand
, the proof is computed as
follows:¶
i
in [H]
create an empty table wire_i
.¶
prove_rand
into subvectors seed_1, ...,
seed_H
where len(seed_i) == L_i
for all i
in [H]
. Let us call these
the "wire seeds" of each gadget.¶
Valid
on input of inp
and joint_rand
, recording the inputs of
each gadget in the corresponding table. Specifically, for every i
in [H]
,
set wire_i[j-1,k-1]
to the value on the j
th wire into the k
th call to
gadget G_i
.¶
Compute the "wire polynomials". That is, for every i
in [H]
and j
in
[L_i]
, construct poly_wire_i[j-1]
, the j
th wire polynomial for the
i
th gadget, as follows:¶
w = [seed_i[j-1], wire_i[j-1,0], ..., wire_i[j-1,M_i-1]]
.¶
padded_w = w + Field.zeros(P_i - len(w))
.¶
NOTE We pad w
to the nearest power of 2 so that we can use FFT for
interpolating the wire polynomials. Perhaps there is some clever math for
picking wire_inp
in a way that avoids having to pad.¶
poly_wire_i[j-1]
be the lowest degree polynomial for which
poly_wire_i[j-1](alpha_i^k) == padded_w[k]
for all k
in [P_i]
.¶
Compute the "gadget polynomials". That is, for every i
in [H]
:¶
poly_gadget_i = G_i(poly_wire_i[0], ..., poly_wire_i[L_i-1])
. That
is, evaluate the circuit G_i
on the wire polynomials for the i
th
gadget. (Arithmetic is in the ring of polynomials over Field
.)¶
The proof is the vector proof = seed_1 + coeff_1 + ... + seed_H + coeff_H
,
where coeff_i
is the vector of coefficients of poly_gadget_i
for each i
in
[H]
.¶
On input of inp
, proof
, query_rand
, and joint_rand
, the verifier message
is generated as follows:¶
i
in [H]
create an empty table wire_i
.¶
proof
into the subvectors seed_1
, coeff_1
, ..., seed_H
,
coeff_H
defined in Section 7.3.3.1.¶
Valid
on input of inp
and joint_rand
, recording the inputs of
each gadget in the corresponding table. This step is similar to the prover's
step (3.) except the verifier does not evaluate the gadgets. Instead, it
computes the output of the k
th call to G_i
by evaluating
poly_gadget_i(alpha_i^k)
. Let v
denote the output of the circuit
evaluation.¶
Compute the tests for well-formedness of the gadget polynomials. That is, for
every i
in [H]
:¶
The verifier message is the vector verifier = [v] + x_1 + [y_1] + ... + x_H +
[y_H]
.¶
This section specifies instantiations of Prio3 for various measurement types.
Each uses FlpGeneric
as the FLP (Section 7.3) and is determined by a
validity circuit (Section 7.3.2) and a PRG (Section 6.2). Test vectors for
each can be found in Appendix "Test Vectors".¶
NOTE Reference implementations of each of these VDAFs can be found in https://github.com/cfrg/draft-irtf-cfrg-vdaf/blob/main/poc/vdaf_prio3.sage.¶
Our first instance of Prio3 is for a simple counter: Each measurement is either one or zero and the aggregate result is the sum of the measurements.¶
This instance uses PrgAes128
(Section 6.2.1) as its PRG. Its validity
circuit, denoted Count
, uses Field64
(Table 3) as its finite field. Its
gadget, denoted Mul
, is the degree-2, arity-2 gadget defined as¶
def Mul(x, y): return x * y¶
The validity circuit is defined as¶
def Count(inp: Vec[Field64]): return Mul(inp[0], inp[0]) - inp[0]¶
The measurement is encoded and decoded as a singleton vector in the natural way. The parameters for this circuit are summarized below.¶
Parameter | Value |
---|---|
GADGETS
|
[Mul]
|
GADGET_CALLS
|
[1]
|
INPUT_LEN
|
1
|
OUTPUT_LEN
|
1
|
JOINT_RAND_LEN
|
0
|
Measurement
|
Unsigned , in range [0,2)
|
AggResult
|
Unsigned
|
Field
|
Field64 (Table 3) |
The next instance of Prio3 supports summing of integers in a pre-determined
range. Each measurement is an integer in range [0, 2^bits)
, where bits
is an
associated parameter.¶
This instance of Prio3 uses PrgAes128
(Section 6.2.1) as its PRG.
Its validity circuit, denoted Sum
, uses Field128
(Table 4) as its
finite field. The measurement is encoded as a length-bits
vector of field
elements, where the l
th element of the vector represents the l
th bit of the
summand:¶
def encode(Sum, measurement: Integer): if 0 > measurement or measurement >= 2^Sum.INPUT_LEN: raise ERR_INPUT encoded = [] for l in range(Sum.INPUT_LEN): encoded.append(Sum.Field((measurement >> l) & 1)) return encoded def truncate(Sum, inp): decoded = Sum.Field(0) for (l, b) in enumerate(inp): w = Sum.Field(1 << l) decoded += w * b return [decoded] def decode(Sum, output, _num_measurements): return output[0].as_unsigned()¶
The validity circuit checks that the input consists of ones and zeros. Its
gadget, denoted Range2
, is the degree-2, arity-1 gadget defined as¶
def Range2(x): return x^2 - x¶
The validity circuit is defined as¶
def Sum(inp: Vec[Field128], joint_rand: Vec[Field128]): out = Field128(0) r = joint_rand[0] for x in inp: out += r * Range2(x) r *= joint_rand[0] return out¶
Parameter | Value |
---|---|
GADGETS
|
[Range2]
|
GADGET_CALLS
|
[bits]
|
INPUT_LEN
|
bits
|
OUTPUT_LEN
|
1
|
JOINT_RAND_LEN
|
1
|
Measurement
|
Unsigned , in range [0, 2^bits)
|
AggResult
|
Unsigned
|
Field
|
Field128 (Table 4) |
This instance of Prio3 allows for estimating the distribution of the measurements by computing a simple histogram. Each measurement is an arbitrary integer and the aggregate result counts the number of measurements that fall in a set of fixed buckets.¶
This instance of Prio3 uses PrgAes128
(Section 6.2.1) as its PRG. Its
validity circuit, denoted Histogram
, uses Field128
(Table 4) as its
finite field. The measurement is encoded as a one-hot vector representing the
bucket into which the measurement falls (let bucket
denote a sequence of
monotonically increasing integers):¶
def encode(Histogram, measurement: Integer): boundaries = buckets + [Infinity] encoded = [Field128(0) for _ in range(len(boundaries))] for i in range(len(boundaries)): if measurement <= boundaries[i]: encoded[i] = Field128(1) return encoded def truncate(Histogram, inp: Vec[Field128]): return inp def decode(Histogram, output: Vec[Field128], _num_measurements): return [bucket_count.as_unsigned() for bucket_count in output]¶
The validity circuit uses Range2
(see Section 7.4.2) as its single gadget. It
checks for one-hotness in two steps, as follows:¶
def Histogram(inp: Vec[Field128], joint_rand: Vec[Field128], num_shares: Unsigned): # Check that each bucket is one or zero. range_check = Field128(0) r = joint_rand[0] for x in inp: range_check += r * Range2(x) r *= joint_rand[0] # Check that the buckets sum to 1. sum_check = -Field128(1) * Field128(num_shares).inv() for b in inp: sum_check += b out = joint_rand[1] * range_check + \ joint_rand[1]^2 * sum_check return out¶
Note that this circuit depends on the number of shares into which the input is sharded. This is provided to the FLP by Prio3.¶
Parameter | Value |
---|---|
GADGETS
|
[Range2]
|
GADGET_CALLS
|
[buckets + 1]
|
INPUT_LEN
|
buckets + 1
|
OUTPUT_LEN
|
buckets + 1
|
JOINT_RAND_LEN
|
2
|
Measurement
|
Integer
|
AggResult
|
Vec[Unsigned]
|
Field
|
Field128 (Table 4) |
NOTE This construction has not undergone significant security analysis.¶
This section specifies Poplar1, a VDAF for the following task. Each Client holds
a string of length BITS
and the Aggregators hold a set of l
-bit strings,
where l <= BITS
. We will refer to the latter as the set of "candidate
prefixes". The Aggregators' goal is to count how many inputs are prefixed by
each candidate prefix.¶
This functionality is the core component of the Poplar protocol [BBCGGI21]. At a high level, the protocol works as follows.¶
0
and
1
.¶
H
denote the set of prefixes that occurred at least t
times. If the
prefixes all have length BITS
, then H
is the set of t
-heavy-hitters.
Otherwise compute the next set of candidate prefixes as follows. For each p
in H
, add add p || 0
and p || 1
to the set. Repeat step 3 with the new
set of candidate prefixes.¶
Poplar1 is constructed from an "Incremental Distributed Point Function (IDPF)", a primitive described by [BBCGGI21] that generalizes the notion of a Distributed Point Function (DPF) [GI14]. Briefly, a DPF is used to distribute the computation of a "point function", a function that evaluates to zero on every input except at a programmable "point". The computation is distributed in such a way that no one party knows either the point or what it evaluates to.¶
An IDPF generalizes this "point" to a path on a full binary tree from the root to one of the leaves. It is evaluated on an "index" representing a unique node of the tree. If the node is on the programmed path, then function evaluates to to a non-zero value; otherwise it evaluates to zero. This structure allows an IDPF to provide the functionality required for the above protocol, while at the same time ensuring the same degree of privacy as a DPF.¶
Poplar1 composes an IDPF with the "secure sketching" protocol of [BBCGGI21]. This protocol ensures that evaluating a set of input shares on a unique set of candidate prefixes results in shares of a "one-hot" vector, i.e., a vector that is zero everywhere except for one element, which is equal to one.¶
The remainder of this section is structured as follows. IDPFs are defined in Section 8.1; a concrete instantiation is given Section 8.3. The Poplar1 VDAF is defined in Section 8.2 in terms of a generic IDPF. Finally, a concrete instantiation of Poplar1 is specified in Section 8.4; test vectors can be found in Appendix "Test Vectors".¶
An IDPF is defined over a domain of size 2^BITS
, where BITS
is constant
defined by the IDPF. Indexes into the IDPF tree are encoded as integers in range
[0, 2^BITS)
. The Client specifies an index alpha
and a vector of
values beta
, one for each "level" L
in range [0, BITS)
. The key generation
algorithm generates one IDPF "key" for each Aggregator. When evaluated at level
L
and index 0 <= prefix < 2^L
, each IDPF key returns an additive share of
beta[L]
if prefix
is the L
-bit prefix of alpha
and shares of zero
otherwise.¶
An index x
is defined to be a prefix of another index y
as follows. Let
LSB(x, N)
denote the least significant N
bits of positive integer x
. By
definition, a positive integer 0 <= x < 2^L
is said to be the length-L
prefix of positive integer 0 <= y < 2^BITS
if LSB(x, L)
is equal to the most
significant L
bits of LSB(y, BITS)
, For example, 6 (110 in binary) is the
length-3 prefix of 25 (11001), but 7 (111) is not.¶
Each of the programmed points beta
is a vector of elements of some finite
field. We distinguish two types of fields: One for inner nodes (denoted
Idpf.FieldInner
), and one for leaf nodes (Idpf.FieldLeaf
). (Our
instantiation of Poplar1 (Section 8.4) will use a much larger
field for leaf nodes than for inner nodes. This is to ensure the IDPF is
"extractable" as defined in [BBCGGI21], Definition 1.)¶
A concrete IDPF defines the types and constants enumerated in Table 13. In
the remainder we write Idpf.Vec
as shorthand for the type
Union[Vec[Vec[Idpf.FieldInner]], Vec[Vec[Idpf.FieldLeaf]]]
. (This type denotes
either a vector of inner node field elements or leaf node field elements.) The
scheme is comprised of the following algorithms:¶
Idpf.gen(alpha: Unsigned, beta_inner: Vec[Vec[Idpf.FieldInner]], beta_leaf:
Vec[Idpf.FieldLeaf]) -> (Bytes, Vec[Bytes])
is the randomized IDPF-key
generation algorithm. Its inputs are the index alpha
and the values beta
.
The value of alpha
MUST be in range [0, 2^BITS)
. The output is a public
part that is sent to all aggregators and a vector of private IDPF keys, one
for each aggregator.¶
Idpf.eval(agg_id: Unsigned, public_share: Bytes, key: Bytes, level: Unsigned,
prefixes: Vec[Unsigned]) -> Idpf.Vec
is the deterministic, stateless
IDPF-key evaluation algorithm run by each Aggregator. Its inputs are the
Aggregator's unique identifier, the public share distributed to all of the
Aggregators, the Aggregator's IDPF key, the "level" at which to evaluate the
IDPF, and the sequence of candidate prefixes. It returns the share of the
value corresponding to each candidate prefix.¶
The output type depends on the value of level
: If level < Idpf.BITS-1
, the
output is the value for an inner node, which has type
Vec[Vec[Idpf.FieldInner]]
; otherwise, if level == Idpf.BITS-1
, then the
output is the value for a leaf node, which has type
Vec[Vec[Idpf.FieldLeaf]]
.¶
The value of level
MUST be in range [0, BITS)
. The indexes in prefixes
MUST all be distinct and in range [0, 2^level)
.¶
Applications MUST ensure that the Aggregator's identifier is equal to the
integer in range [0, SHARES)
that matches the index of key
in the sequence
of IDPF keys output by the Client.¶
In addition, the following method is derived for each concrete Idpf
:¶
def current_field(Idpf, level): return Idpf.FieldInner if level < Idpf.BITS-1 \ else Idpf.FieldLeaf¶
Finally, an implementation note. The interface for IDPFs specified here is stateless, in the sense that there is no state carried between IDPF evaluations. This is to align the IDPF syntax with the VDAF abstraction boundary, which does not include shared state across across VDAF evaluations. In practice, of course, it will often be beneficial to expose a stateful API for IDPFs and carry the state across evaluations. See Section 8.3 for details.¶
Parameter | Description |
---|---|
SHARES | Number of IDPF keys output by IDPF-key generator |
BITS | Length in bits of each input string |
VALUE_LEN | Number of field elements of each output value |
KEY_SIZE | Size in bytes of each IDPF key |
FieldInner | Implementation of Field (Section 6.1) used for values of inner nodes |
FieldLeaf | Implementation of Field used for values of leaf nodes |
Prg | Implementation of Prg (Section 6.2) |
This section specifies Poplar1
, an implementation of the Vdaf
interface
(Section 5). It is defined in terms of any Idpf
(Section 8.1) for which
Idpf.SHARES == 2
and Idpf.VALUE_LEN == 2
. The associated constants and types
required by the Vdaf
interface are defined in Table 14. The methods
required for sharding, preparation, aggregation, and unsharding are described in
the remaining subsections.¶
Parameter | Value |
---|---|
VERIFY_KEY_SIZE
|
Idpf.Prg.SEED_SIZE
|
ROUNDS
|
2
|
SHARES
|
2
|
Measurement
|
Unsigned
|
AggParam
|
Tuple[Unsigned, Vec[Unsigned]]
|
Prep
|
Tuple[Bytes, Unsigned, Idpf.Vec]
|
OutShare
|
Idpf.Vec
|
AggResult
|
Vec[Unsigned]
|
The client's input is an IDPF index, denoted alpha
. The programmed IDPF values
are pairs of field elements (1, k)
where each k
is chosen at random. This
random value is used as part of the secure sketching protocol of [BBCGGI21],
Appendix C.4. After evaluating their IDPF key shares on a given sequence of
candidate prefixes, the sketching protocol is used by the Aggregators to verify
that they hold shares of a one-hot vector. In addition, for each level of the
tree, the prover generates random elements a
, b
, and c
and computes¶
A = -2*a + k B = a^2 + b - k*a + c¶
and sends additive shares of a
, b
, c
, A
and B
to the Aggregators.
Putting everything together, the input-distribution algorithm is defined as
follows. Function encode_input_shares
is defined in Section 8.2.5.¶
The aggregation parameter encodes a sequence of candidate prefixes. When an
Aggregator receives an input share from the Client, it begins by evaluating its
IDPF share on each candidate prefix, recovering a data_share
and auth_share
for each. The Aggregators use these and the correlation shares provided by the
Client to verify that the sequence of data_share
values are additive shares of
a one-hot vector.¶
The algorithms below make use of auxiliary functions verify_context()
and
decode_input_share()
defined in Section 8.2.5.¶
Aggregation involves simply adding up the output shares.¶
Finally, the Collector unshards the aggregate result by adding up the aggregate shares.¶
In this section we specify a concrete IDPF, called IdpfPoplar, suitable for instantiating Poplar1. The scheme gets its name from the name of the protocol of [BBCGGI21].¶
TODO We should consider giving IdpfPoplar
a more distinctive name.¶
The constant and type definitions required by the Idpf
interface are given in
Table 15.¶
Parameter | Value |
---|---|
SHARES |
2
|
BITS | any positive integer |
VALUE_LEN | any positive integer |
KEY_SIZE |
Prg.SEED_SIZE
|
FieldInner |
Field64 (Table 3) |
FieldLeaf |
Field255 (Table 5) |
Prg | any implementation of Prg (Section 6.2) |
TODO Describe the construction in prose, beginning with a gentle introduction to the high level idea.¶
The description of the IDPF-key generation algorithm makes use of auxiliary
functions extend()
, convert()
, and encode_public_share()
defined in
Section 8.3.3. In the following, we let Field2
denote the
field GF(2)
.¶
TODO Describe in prose how IDPF-key evaluation algorithm works.¶
The description of the IDPF-evaluation algorithm makes use of auxiliary
functions extend()
, convert()
, and decode_public_share()
defined in
Section 8.3.3.¶
We refer to Poplar1 instantiated with IdpfPoplar (VALUE_LEN == 2
)
and PrgAes128 (Section 6.2.1) as Poplar1Aes128. This VDAF is suitable
for any positive value of BITS
. Test vectors can be found in
Appendix "Test Vectors".¶
NOTE: This is a brief outline of the security considerations. This section will be filled out more as the draft matures and security analyses are completed.¶
VDAFs have two essential security goals:¶
Note that, to achieve robustness, it is important to ensure that the
verification key distributed to the Aggregators (verify_key
, see Section 5.2) is
never revealed to the Clients.¶
It is also possible to consider a stronger form of robustness, where the
attacker also controls a subset of Aggregators (see [BBCGGI19], Section 6.3).
To satisfy this stronger notion of robustness, it is necessary to prevent the
attacker from sharing the verification key with the Clients. It is therefore
RECOMMENDED that the Aggregators generate verify_key
only after a set of
Client inputs has been collected for verification, and re-generate them for each
such set of inputs.¶
In order to achieve robustness, the Aggregators MUST ensure that the nonces used to process the measurements in a batch are all unique.¶
A VDAF is the core cryptographic primitive of a protocol that achieves the above privacy and robustness goals. It is not sufficient on its own, however. The application will need to assure a few security properties, for example:¶
Establishing secure channels:¶
In such an environment, a VDAF provides the high-level privacy property described above: The Collector learns only the aggregate measurement, and nothing about individual measurements aside from what can be inferred from the aggregate result. The Aggregators learn neither individual measurements nor the aggregate result. The Collector is assured that the aggregate statistic accurately reflects the inputs as long as the Aggregators correctly executed their role in the VDAF.¶
On their own, VDAFs do not mitigate Sybil attacks [Dou02]. In this attack, the adversary observes a subset of input shares transmitted by a Client it is interested in. It allows the input shares to be processed, but corrupts and picks bogus inputs for the remaining Clients. Applications can guard against these risks by adding additional controls on measurement submission, such as client authentication and rate limits.¶
VDAFs do not inherently provide differential privacy [Dwo06]. The VDAF approach to private measurement can be viewed as complementary to differential privacy, relying on non-collusion instead of statistical noise to protect the privacy of the inputs. It is possible that a future VDAF could incorporate differential privacy features, e.g., by injecting noise before the sharding stage and removing it after unsharding.¶
This document makes no request of IANA.¶
Thanks to David Cook, Henry Corrigan-Gibbs, Armando Faz-Hernandez, Simon Friedberger, Tim Geoghegan, Mariana Raykova, Jacob Rothstein, and Christopher Wood for useful feedback on and contributions to the spec.¶
NOTE Machine-readable test vectors can be found at https://github.com/cfrg/draft-irtf-cfrg-vdaf/tree/main/poc/test_vec.¶
Test vectors cover the generation of input shares and the conversion of input
shares into output shares. Vectors specify the verification key, measurements,
aggregation parameter, and any parameters needed to construct the VDAF. (For
example, for Prio3AesSum
, the user specifies the number of bits for
representing each summand.)¶
Byte strings are encoded in hexadecimal To make the tests deterministic,
gen_rand()
was replaced with a function that returns the requested number of
0x01
octets.¶
verify_key: "01010101010101010101010101010101" upload_0: measurement: 1 nonce: "01010101010101010101010101010101" public_share: >- input_share_0: >- a46091d07a930ace79bb11d1d03596bf4f921f7e70b853cf6b8d4969287dfb504a1d ea6b0eb01a5f50ad24225186c104 input_share_1: >- 0101010101010101010101010101010101010101010101010101010101010101 round_0: prep_share_0: >- cdbbf14ef0c19728578ce1e50411e1e26eb49eea4dadfada9ed3d13c71d846b8 prep_share_1: >- 32440eb00f3e68d95b9f6e2e856cf534948881bf63c24d315e4954159611ebb5 prep_message: >- out_share_0: - 11844627344580086478 out_share_1: - 6602116724834497844 agg_share_0: >- a46091d07a930ace agg_share_1: >- 5b9f6e2e856cf534 agg_result: 1¶
bits: 8 verify_key: "01010101010101010101010101010101" upload_0: measurement: 100 nonce: "01010101010101010101010101010101" public_share: >- input_share_0: >- a46091d17a930aaf4d02cff822f817a3d10d4d74d526daf00112da46b9b33deaa222 0ff24fe104b34007f47c4531dc7471b6832358f5d6859fb3b2112e79ec6eb940251d 14855b16689ab3cfd9100b568729aadfb5558a3bd0277d3ae7c5d967e3525839f26f 5cb210e4fb0b0aeb0394ce0791de9c4c07f8840dcdd1ce09bb2179bb11d1d03596be 4f921f7f70b853ce99c2bd80daf08029b7bb7820b7a7632e3b9200ff98adde77a833 6469ba7fa61ebf135e7f303568ddf63e39978954c40a3fcfa08412f5f03f61699643 85f167c68c3ae5205e1c3dd075fc76224bf9407ef3fd54334bec36d453460381ee54 4027bb520f55394873f5dff846794007cc7d77e903b1474095c61140d1e488a98e00 f433c22491825191261e982affc44e20756db024e0dcdda9746aba73d2ec9a083953 4a1d0178802fc0ee9393160b34dca94999c91fa788b8308b15880ea3146aabf38e2f a4d715c0be7e183fe7f158f47438eaa5a55406da5be56d724afd86e579bc076c669c 334c93fb724891e0b72c8737a9f65c2d3d56bae20dce052f43293712a603f8a8b9e8 bcdafad8420ae14201f9a8b4891e9b60dd418d0f71ee2395b9026cb4978f901765e0 155b127a4fa95a0e0751b95d9ada245c78df820e4bc7f0c9329a5bbad2ee85688cea 47336da7a798705b6c465678363574e844d1974f6f369e9ef13e920d383f580aa382 8b7641a77f352b73134886c58e822040747311a9b094137c5f46d6e9ed05bb17d9c7 0f01b5491805ac83c447035560b00b8e7049c334fcfa003941fa0e307cf52884e461 8f794848936bfcd8d7f895781d27b006673a6349bdfa04c8c3506b2b64efff5335f7 6723bce0f1092571cae3d125dcabbb6da4867963ffd29172ef699ecf010101010101 0101010101010101010132dd9fa5604fadc81ff1636918a6db69 input_share_1: >- 01010101010101010101010101010101010101010101010101010101010101010101 0101010101010101010101010101646430cb3212da0c23bff5618647609c round_0: prep_share_0: >- ede97ab0d3d2dc256a1f82261078cc06b5a423c5f221e9fa029ba023bc71e70fd1 9e2db541507b4171409bc06c3ac292646430cb3212da0c23bff5618647609c prep_share_1: >- 1216854f2c2d23be95e07dd9ef8733fb779dce023a0e4f7147b74c75c5fd1723a1 1b7a116c0a89c6b97471597eb2a83c32dd9fa5604fadc81ff1636918a6db69 prep_message: >- 56b9af6e525d77c43c4e96089ee1bbf5 out_share_0: - 231691668211413349799981210015670190003 out_share_1: - 108590698709525113146884563352230576306 agg_share_0: >- ae4e3131e0f5bea01ba67704684fbfb3 agg_share_1: >- 51b1cece1f0a4143e45988fb97b040b2 agg_result: 100¶
buckets: [1, 10, 100] verify_key: "01010101010101010101010101010101" upload_0: measurement: 50 nonce: "01010101010101010101010101010101" public_share: >- input_share_0: >- a46091d17a930aaf4d02cff822f817a3d10d4d74d526daf00112da46b9b33deaa222 0ff24fe104b34007f47c4531dc7471b6832358f5d6859fb3b2112e79ec6e79bb11d1 d03596be4f921f7f70b853cec138b5a4f1a798810bde46a7265fd6bed9bb8cc8d058 bb224cc75b17f2ebc7f7b6c7a708b11022329fab97ed7d23309116a6fb53d1d9ff45 9a67eafcddbd190e69069ef82f361a6d4c9616ec3396776834a7d3d5848409cdd89f ec7920a858074db9c2e7adfc4b3397339532f80e327f25f31975e3f5d149ba8c8d5b f101488f4882ab2b38ddedc84f814435c2b227ffeee7affc5076d78819d7ce68f247 355af67c52fc7ef9bbcbf8bc910af7c047a28dfeb4f92158934ce93dac747c1b55a8 f5efb18e944e0579602c918e670f8d543b530d8ba3730d0a43e8bcd2d54f736c4db1 ac64c9c49dc5ab20c0ac0cf6530b01010101010101010101010101010101d451ca17 ca18344ac4925510c71e5ac8 input_share_1: >- 01010101010101010101010101010101010101010101010101010101010101010101 0101010101010101010101010101ef377f05b86ffa85ed011213230bb1d5 round_0: prep_share_0: >- a1a7093768a9f8a3c01e31f7e9c6dc7925407d03831c510799fcf30a03c61bcd35 78329d502e06cd01e994c227d3d1afef377f05b86ffa85ed011213230bb1d5 prep_share_1: >- 5e58f6c8975607403fe1ce081639238809c0227ef8c371e897e7950d763d8a37f7 ee6acb826a97f3ef37fd59e6a95511d451ca17ca18344ac4925510c71e5ac8 prep_message: >- 3b66b5127277cecf29934703e415eb1d out_share_0: - 218494809353158975333103031759951894435 - 277877721980181983205633439801202130410 - 215511796844427285058159435311010143348 - 151150421348124479376036084370309246062 out_share_1: - 121787557567779487613762741607948871774 - 62404644940756479741232333566698635799 - 124770570076511177888706338056890622862 - 189131945572813983570829688997591520147 agg_share_0: >- a46091d17a930aaf4d02cff822f817a3d10d4d74d526daf00112da46b9b33deaa2220f f24fe104b34007f47c4531dc7471b6832358f5d6859fb3b2112e79ec6e agg_share_1: >- 5b9f6e2e856cf534b2fd3007dd07e85e2ef2b28b2ad924f3feed25b9464cc2175dddf0 0db01efb30bff80b83bace238e8e497cdca70a295e604c4deed1861393 agg_result: [0, 0, 1, 0]¶
verify_key: "01010101010101010101010101010101" agg_param: (0, [0, 1]) upload_0: round_0: prep_share_0: >- 48c3581799a109398b1943c10957b8c2536f292900de6436 prep_share_1: >- 8377f3fcd4d2d1f33233948d19fe2c385f97c52d8e007f76 prep_message: >- cc3b4c146e73db2cbd4cd84e2355e4fab306ee568edee3ac round_1: prep_share_0: >- 64ce1f88d77b2a65 prep_share_1: >- 9b31e0762884d59c prep_message: >- out_share_0: - 18352387916526274078 - 14559030685141423577 out_share_1: - 94356152888310243 - 3887713384273160745 agg_share_0: >- feb0c79330b8461eca0c114565ddd5d9 agg_share_1: >- 014f386bcf47b9e335f3eeb99a222a29 agg_result: [0, 1]¶
verify_key: "01010101010101010101010101010101" agg_param: (1, [0, 1, 2, 3]) upload_0: round_0: prep_share_0: >- 37c72ca578a032eaeffbd9e63d29e6eaa205d9d67628d055 prep_share_1: >- ab956611d15cfcef0773a4e435fdcb83909785aa50c17513 prep_message: >- e35c92b749fd2fd9f76f7eca7327b26d329d5f81c6ea4567 round_1: prep_share_0: >- ee63dfb515814411 prep_share_1: >- 119c2049ea7ebbf0 prep_message: >- out_share_0: - 7623875273889432259 - 4753532626118505607 - 9999595461072379379 - 1320535832873937471 out_share_1: - 10822868795525152062 - 13693211443296078714 - 8447148608342204942 - 17126208236540646851 agg_share_0: >- 69cd722b281042c341f7f0bcf805f4878ac5b3177ad3ddf312537c57efa02a3f agg_share_1: >- 96328dd3d7efbd3ebe080f4207fa0b7a753a4ce7852c220eedac83a7105fd5c3 agg_result: [0, 0, 0, 1]¶
verify_key: "01010101010101010101010101010101" agg_param: (2, [0, 2, 4, 6]) upload_0: round_0: prep_share_0: >- 7eb8dbda5ea6807e53f22d3f74a43f59ed83ef578b070654 prep_share_1: >- 567632dc63ec2294c6f030077726f87ec11e2fc53a38f640 prep_message: >- d52f0eb6c292a3121ae25d47ebcb37d6aea21f1dc53ffc93 round_1: prep_share_0: >- 9e92c793980229fd prep_share_1: >- 616d386b67fdd604 prep_message: >- out_share_0: - 1646250657834468215 - 16923505979406688621 - 18330814903122515861 - 12155602082232726549 out_share_1: - 16800493411580116106 - 1523238090007895700 - 115929166292068460 - 6291141987181857773 agg_share_0: >- 16d8a840477a0377eadc5f0610bc216dfe642308980dd795a8b15fb0ce7af415 agg_share_1: >- e92757beb885fc8a1523a0f8ef43de94019bdcf667f2286c574ea04e31850bed agg_result: [0, 0, 0, 1]¶
verify_key: "01010101010101010101010101010101" agg_param: (3, [1, 3, 5, 7, 9, 13, 15]) upload_0: round_0: prep_share_0: >- 246777985de3e9fd0ea9e8ea429ae3c7255c0b2aa84cff6bd5c0079a122be38b53 4e75cbfe8abb2c54b5b277089bb2708b274a17ce26df0ea9881487ac852eee30c1 8ba6dc40beb7f33e5003fdf065eec844e320d0b9ff49b0b29f5bbbcbcc8d prep_share_1: >- 3dea0b9e72389ea0d62521e91214c654ebf9b2a3d855a32219755216f4cb8eca0d 864838efc99f1a5d2adecf87e4ff0b839fb7191d4c58c5fd6753c0c6a2fb155071 efff9719f0e9645d70b398ca24ecb2fa8fbb56a6a90ce6d95b675e1e28f2 prep_message: >- 62518336d01c889de4cf0ad354afaa1c1155bdce80a2a28def3559b106f7725560 d4be04ee545a46b1e091469080b17c0ec70130eb7337d4a6ef684873282a030133 7ba6735aafa1579bc0b796ba8adb7b3f72dc2760a856978bfac319e9f592 round_1: prep_share_0: >- 40521213baba54c83b4ddc0cb1e3e4c16897bb8831cc12ea91a82927dd30ae58 prep_share_1: >- 3fadedec4545ab37c4b223f34e1c1b3e97684477ce33ed156e57d6d822cf5195 prep_message: >- out_share_0: - 27022242238524926041754024638978288940369802983157338113702795433392718442891 - 8433446103891669428510589487794410319530091467635619512122120272876905203470 - 31927171851194534364500488903327032909992419490289770138187660233081601268509 - 13674389066590417186009193737398514883303633790292716630453797389670854762435 - 49245655988555337792760994399532100867010365510706353464574012401106060986059 - 38418164104898327452350816690091778787588836972103923142413897241001677348897 - 17473652484574293745985786122460988541722879368489070298827855759184581179349 out_share_1: - 30873802380133171670031467865365664986265189349662943906025996570563846377058 - 49462598514766428283274903016549543607104900865184662507606671731079659616479 - 25968872767463563347285003601016921016642572842530511881541131770874963551440 - 44221655552067680525776298766945439043331358542527565389274994614285710057514 - 8650388630102759919024498104811853059624626822113928555154779602850503833890 - 19477880513759770259434675814252175139046155360716358877314894762954887471053 - 40422392134083803965799706381882965384912112964331211720900936244771983640600 agg_share_0: >- 3bbe0c0f2a3ecf406419e5a8636f13c1102df2df86b4dec82daa4669b24f718b12a529 4d99b7a604a6944dbe1b81cfcc98e4a5a1fa0d08ac727f620ee0b4870e4696238e0b88 71bb6d93ab6c8fc1680da244dfc59bd0b6f70b8dae1f4c1a1b1d1e3b6e1053afb851cc ae5d9ddabb8bff59afea5f6f905e0b2dd1b8a9cc911fc36ce00db4dfb57d404c6bddc9 0a93a35fa5a476634aab5c2c46b467e2a6ce46cb54efe909f5c78891ac58d782afb5a4 4dbc5b2f0c190d4ebe8ca7eb1e4af8e02126a1bca0b523a4c67c7176bebb04afed7b50 6fc5c343e3e7727d3027d68937d5 agg_share_1: >- 4441f3f0d5c130bf9be61a579c90ec3eefd20d20794b2137d255b9964db08e626d5ad6 b2664859fb596bb241e47e3033671b5a5e05f2f7538d809df11f4b78df3969dc71f477 8e44926c5493703e97f25dbb203a642f4908f47251e0b3e5e4d061c491efac5047ae33 51a26225447400a65015a0906fa1f4d22e4756336ee02a131ff24b204a82bfb3942236 f56c5ca05a5b899cb554a3d3b94b981d5931b9222b1016f60a38776e53a7287d504a5b b243a4d0f3e6f2b141735814e1b5071fcd595e435f4adc5b39838e894144fb501284af 903a3cbc1c188d82cfd82976c818 agg_result: [0, 0, 0, 0, 0, 1, 0]¶