Duplication Counting : Ensuring one proof is used only once, without doxxing the user

Madhavan Malolan
Jan 16, 2024
Duplication Counting : Ensuring one proof is used only once, without doxxing the user
CodeNot Code

Introduction

For many applications, it will be critical that multiple users are not able to claim the same credential, at least without being detected. For example, suppose a user proves they own some funds in a certain bank account, and then later shares that their banking login to another user who then proves the same credential. An application that requires users t have some minimum collateral will need to be able to detect if the same bank account is being claimed multiple times.

Furthermore, the two users may be using their duplicated credentials for different applications, but these applications may still want to know that these users' credentials are duplicated. Note that detecting duplicate credentials is trivial if all users must post their credentials to a public chain in an unencrypted format. However, for many kinds of credentials this public chain would leak sensitive information.

Model

We now fix some notation. Consider a single kind of credential, where each user ii proves their credential using data DiD_i. We divide the (useful) parts of each DiD_i into the triple Di=(di1,di2,di3)D_i = (d^1_i, d^2_i, d^3_i) where:

For example, if the data is returned from a banking website, then di1d_i^1 could be the amount of money in the bank account, di2d_i^2 the name of the account owner, and di3d_i^3 the routing number of the account. This is because users might be comfortable with the amount of money in their account being public, as nobody can link that information to them personally. Additionally, users might be comfortable sharing their name to specific applications that need this information, but will not want to share the routing number of their account to anyone.

Finally, we denote the set of all possible DiD_i values as DD.

Problem

We will assume that users are not required to have continued interaction with the system after they prove and share their credentials with applications. We will call this the non-interactive assumption. Therefore, all information about a user's credential that will be used for future duplication counting must be saved to a public database at the time that the credential is created. So, without loss of generality we can assume that for some fixed function HH, when a user creates a credential using data DiD_i , they must post to a public chain the data

ci=(di1,H(di2,di3))c_i = (d_i^1 ,H(d_i^2,d_i^3))

as well as the signature of the attestor. When a user shows a credential to an application, they will also have to share the true value of di2d_i^2 .

For HH to work as a feasible solution, it must satisfy the following two conditions:

(1) HH sufficiently obscures di2d_i^2 and di3d_i^3 so that no sensitive information is leaked.

(2) HH outputs enough information so that distinct credentials are not counted as duplicates. That is, if DiDjD_i \neq D_j then cicjc_i \neq c_j.

Later in this article, we consider relaxing the non-interactive assumption. That is, users can be asked at a future date to cooperate in a protocol to show that some new credential is not a duplicate. We discuss how multi-party computation (MPC), at least theoretically, could be used to solve the problem given this relaxation. However, since MPCs of this size are not currently practical, we otherwise will keep the non-interactive assumption.

Entropy

For a given HH function to satisfy condition (1) is the entropy in the space of possible credentials. Informally, the entropy of this space is a measure of how much random variation there will be between credentials.

Let D\mathcal D' be the set of all possible (di2,di3)(d_i^2,d_i^3) values. If it is reasonably inexpensive for someone to compute the hashed value of every (likely) element of D\mathcal D', then they can determine the value of every (likely) credential that has been posted on the chain. Condition (1) is then violated. So, the entropy of D\mathcal D' is critical for the security: if it is too low relative to the computational cost of HH then the scheme will not be secure. Letting T(H)T(H) be the average cost (in time or memory) of computing the hashed value of some input, the cost to compute the hashed value of all elements of D\mathcal D will be T(H)D.T(H)|\mathcal D'|.

The two ways to alleviate this problem is to either increase the entropy of D\mathcal D' by expanding the scope of the data that must be used to create credentials (i.e. increasing D|\mathcal D'|), or to increase the computational cost of HH (i.e. increasing T(H)T(H)). We will further discuss both of these strategies below in the Solutions section.

Credential Uniqueness

Condition (2) on HH says that credentials that are actually unique must not be erroneously counted as being duplicates. For example, suppose users are proving credentials using data from a government website, and multiple users have proven the credential that just says ‘My government login says I am at least 18.' These credentials do not contain enough information to positively confirm that they are not duplicates. Enough information needs to be added to the credentials so that:

Individual Applications

The problem becomes somewhat simpler when, for some kind of credential, each application is only concerned with detecting duplicates among the credentials that are used for it. This is because the credentials do not need to be posted to a public chain, and so users share credentials of the form (di1,di2,H(di3))(d_i^1,d_i^2,H(d_i^3)) to applications. Otherwise, the same two conditions still apply to HH.

Solutions

Recall that D\mathcal D' is the set of all possible (di2,di3)(d_i^2,d_i^3) values. If D\mathcal |D'| is sufficiently large, then we claim that setting HH to be any reasonable hash function with unknown inverse will satisfy condition (1). This is because it is too expensive for someone to find inverse values of HH through brute force. Secondly, if every user credential will have a unique (di2,di3)(d_i^2,d_i^3) values, then this choice of HH will also satisfy condition (2) as long as the probability of collisions of HH is sufficiently low.

But what if D\mathcal D' does not contain enough entropy, or the (di2,di3)(d_i^2,d_i^3) will not be all unique? We now explores different strategies to overcome this problem.

Trusted Entity or TEEs

More options become available if there is an entity that can be trusted to participate in the protocol honestly, or a trusted execution environment (TEE) can be used to emulate a trusted entity. We will assume that the entity is trusted to respond honestly to certain requests, but still is not allowed to learn the private data (i.e di3d_i^3 values) of the users.

Consider using the trusted entity in the following scheme. The entity manages a database, and when a user ii proves a credential, they submit the credential to the entity's database in the form ci=(di1,di2,H(di3))c_i = (d_i^1,d_i^2,H(d_i^3)), as they would in the Individual Application case discussed above. Now, when the user shows cic_i to an application, the application can query the trusted entity with cic_i and the entity will respond with the number of times cic_i appears in the database. See that this will achieve similar security and privacy as the Individual Applications case.

Importantly, the entity must only respond to a query from an application if the application provides evidence of knowledge of a legitimate credential. This is because if people can make arbitrary queries to the entity's database, and the entropy of the credential space is low, then they can discover some or all of the credentials in the database by brute force querying.

Secondary Data Source

For some data sources it may be possible to increase the entropy of D\mathcal D' or the uniqueness of (di2,di3)(d_i^2,d_i^3) values by incorporating another piece of data from the source. This secondary data source will have to be found on a case-by-case basis, and must satisfy the following two conditions:

Here are some examples.

Suppose the type of credential that users are proving is ownership of an email account, where di2d_i^2 is the address of the email account and di3d_i^3 is empty. Since many email addresses have fewer than eight characters (before the domain name), there may not be enough entropy to satisfy condition (1). Depending on what data is available on the email client, the secondary data source could be the date the email account was created, for example.

Now suppose the type of credential that users are proving is that they are over 18, using data from their government login. The data fields could be that di2d_i^2 is just a Boolean value of whether the user is over 18 or not, and di3d_i^3 is the user's birthday. Since the number of possible birthdays is low, a secondary data source is needed. Possible data that could be added that may be available on the government website is the user's social security number, or other private identifier.

Memory-Expensive Hashing

To make brute force searching more difficult, a memory-expensive hash function can be chosen for HH. One standard choice for this hash is Argon2, which takes as input parameters the execution time and memory required to compute each output. These parameters can be tuned to be as low as possible while still making brute force searching infeasible.

MPC

This last solution would be an entirely different method of duplication counting. Multi-party computation (MPC) allows parties to ‘jointly compute a function over their inputs while keeping those inputs private'. That is, each user has some private data, and the users perform a protocol together that outputs a function of all the private data, but no user learns the private data of any other user.

Current MPC implementations are too computationally expensive for our purposes. Furthermore, the use of MPCs would require use to lose to non-interactive assumption we described in the Problem section. However, theoretically MPCs could be used achieve stronger privacy guarantees for users.

Consider the following scheme. When a user proves a credential, they post the credential on chain but encrypted with their private key. The user then engages in an MPC protocol with each other user who has proven a credential previously. This protocol outputs the number of credentials that are duplicates of the new credential, as well as a ZK-proof that the output is correct.

Copyright © 2025 Reclaim Protocol. All rights reserved.