Skip to content

πŸ† 1st Place Winner – Zama Bounty : encryptoNFT leverages the power of blockchain and Zama's fhEVM to deliver a secure and decentralized solution for digital rights management (DRM). This project ensures privacy-preserving operations while maintaining transparency and security on the blockchain.

License

Notifications You must be signed in to change notification settings

El-hacen21/encryptoNFT

Repository files navigation

Documentation of the on-chain DRM system using Zama's fhEVM, by El-Hacen Diallo & Matthieu Rambaud

Link to the Application

This document gives an overview of the process of creation and management of NFTs doing access control to secret contents. The management of the NFTs is done by the unique contract and the interfaces (Mint and Gallery) enable to interact with this contract and with the storage place of the encrypted secret contents. The storage location is IPFS (InterPlanetary File System), which ensures robust and decentralized storage solutions. Each NFT (token) controls access to specific secret content. Its owner has the authority to grant or revoke access to this content for multiple users, which we call the Shared-with of the token.

I. Creation of an NFT

Summarizing, the graphical interface Mint.tsx allows a user to upload a file (image, video, document, etc.) to be saved as secret content over which only they have control. To achieve this, the following processes are automatically performed after the file has been uploaded:

  • Generation of a symmetric key (fileKey), encryption of file under fileKey, producing a ciphertext ciphFile.
  • Then, upload of ciphFile on IPFS,

    Hence, controlling access to the secret content "file" is equivalent to controlling access to the "fileKey".

  • Finally, minting of the NFT associated to the encryptedFileKey.

    The contract will do the access control to the "fileKey", thanks to the power of the fhEVM to privately decrypt the "encryptedFileKey" into "fileKey", to a predetermined list of so-called Shared-with users (identified by their Ethereum addresses).

In detail: the Creator uploads a secret content (file) in the graphical interface. This upload calls EncryptThenMint in Mint.tsx, which automatically performs all the following operations at once:

  1. Encryption of the Secret Content:

    • A symmetric 256-bit encryption key (fileKey) is generated

      const fileKey = await generateKey();
    • The secret content file is symmetrically encrypted to produce the ciphertext:

      const ciphFile = await encryptFile(file, fileKey);
      const encryptedFileKey = await fileKeyEncryption(fileKey);

      The fileKey was decomposed into an array of four 64-bit entries to be passed to instance.encrypt64. As output, the encryptedFileKey will contain 4 Uint8Array objects, each resulting from calling instance.encrypt64.

  2. Storage on IPFS :

    • ciphFile is uploaded on IPFS , and the cidHash (unique IPFS identifier of ciphFile) is retrieved.

      const cidHash = await uploadFileToIPFS(ciphFile);

      Pinata (https://www.pinata.cloud/) is a cloud-based service that simplifies uploading and managing files on IPFS. It is used to upload files to IPFS through an API key.

  3. Minting of the NFT : An NFT (token) is created (or "minted") through the contract, with an IPFS reference to the ciphFile, and with the encryptedFileKey added as metadata

    const token = await mintToken(cidHash, encryptedFileKey);

    Storing encryptedFileKey onchain is perfectly fine, thanks to Zama fhvem unique features.

    For more details refer to the function mintToken in contract.sol.

II. Management of the NFT

  • Transfer and Burn :

    • Classical operations such as transfer of ownership, or destruction (burn) of the NFT are possible via the graphical interface.
  • Addition and Revocation of Shared-with :

    • The functions shareToken and revokeTokenAccess (in contract.sol and accessible by the graphical interface) enable to add or remove the address of a user (user) in the list sharedAccess[tokenId] of the Shared-with users, i.e., those having access to the secret content associated to the token (tokenId).

    To enable a shared-with user to quickly know the tokens that it has access to, these functions also update the lists (sharedTokens[user]) of the tokens shared with each user.

    The graphical interface also enables to revoke, at once, all the shared-with of a given token.

    limitNumberOfSharedWith is used to set size limits for the maximum number of NFTs to process in a single call to revokeAllSharedAccess and burnToken. This is done to prevent DDOS issues caused by excessive gas costs when sets become too large, which could cause the functions to revert.

III. Acces to the secret content by a Shared-with

The function displayGallery (in Gallery.tsx) is executed by a user, it returns and displays all the secret contents of the tokens of which it is a shared-with. The function performs automatically all the following steps.

Summarizing, for each secret content (token) for which the user is shared-with, it queries the re-encryption of the fileKey under a public encryption key publicKey which the user owns. Then, it decrypts this re-encryption to obtain fileKey, downloads the public ciphertext (ciphFile) from IPFS and finally decrypts it into file using the fileKey.

In more detail, the function displayGallery automatically performs the following steps for each token in the ownerTokens[user] to display the NFTs owned by the user, and in sharedTokens[user] for the NFTs with which the user is shared-with. For a better user interface, two separate components are created in the gallery. One, managed by displayMyNFTs, displays only the owned NFTs; the other, displaySharedWithMeNFTs, displays the shared NFTs. These two components can be refreshed separately. The main difference between owned NFTs and shared NFTs is that the latter can only be downloaded; other functionalities are not possible.

  1. Generation of the target public Key for Reencryption :

    • The user shared-with of a token needs to receive the fileKey encrypted under an encryption key publicKey which it owns. It thus generates an encryption key pair (unless it has already one associated to the contract) using the function getSignature(contractAddress, account).

      This function is borrowed from fhevmjs.ts.

       const reencryption = await getSignature(contract.contractAddress, account);

      The output of getSignature is called reencryption : it is the tuple of the publicKey concatenated with a signature (signature) issued by the shared-with on the publicKey.

  2. Reencryption then Decryption of the fileKey :

    • Reencryption of the encryptedFileKey of the token under the encryption key publicKey, to obtain reEncryptedFileKey.

      The reencryption query to the fhEVM is done by the function :

      function reencrypt(
        uint256 tokenId,
        bytes32 publicKey,
        bytes memory signature)  public view
        onlySignedPublicKey(publicKey, signature)
        returns (bytes[] memory)

      reencrypt is defined in contract.sol and controls the access: it checks that the signer (signer) of the public key (publicKey) is member of the list (sharedAccess[tokenId]) of the shared-with of the token or the owner .

      require(
            ownerOf(tokenId) == msg.sender || sharedAccess[tokenId][msg.sender],
            "Caller is neither owner nor authorized."
        );
    • Decryption of reEncryptedFileKey with the secret decryption key of the shared-with, to obtain the fileKey.

      This is done via the call to instance.decrypt.

    • Using the fileKey to decrypt the ciphFile:

      const decryptedFile = await decryptFile(ciphFile, fileKey);

      (decryptedFile <-- AES-CTR.Decrypt("ciphFile","fileKey")) which is thus equal to the file initially uploaded by the Creator.

V. Project of improvement: Anonymization of the Shared-with, as soon as encrypted arrays will be released

We are considering adding an optional fonctionality enabling the owner of a token to declare shared-with's under the form of a list (hiddenSharedWith) of their encrypted addresses (format eaddress). It is only when a shared-with requests access to a plaintext content file that its public request to reencrypt makes it lose its anonymity. We thus need to enrich reencrypt so that its tests membership of the public signer of the publicKey, in the encrypted list (hiddenSharedWith). If it belongs, it is thus that the signer was indeed a (previously hidden) shared-with. Thus, for the arguably majority of Shared-with Alices who do not need to obtain the secret content file, their anonymity is prΓ©served. What blocks us from implementing this addition is that, according to the documentation , it is currently not advised to implement a membership test to a list of ciphertexts. The reason being that the currently available functions of the fhEVM would require to "loop over all the indexes and check equality homomorphically, however this pattern is very expensive in gas and should be avoided whenever possible." This documentation announces ("plans to make this kind of operation much more efficient in the future, by adding specialized operators for arrays"). As soon as such a membership operator will be shipped, we will be able to implement anonymization.

Overview of the Code Implementation

  • The smart contract was implemented using Remix. It has been thoroughly tested to ensure reliability and robustness before being deployed to the Zama Devnet. contract.ts handles the interactions with the deployed contract.

  • To ensure global accessibility of the created instance throughout the application, we utilize the FhevmContext.tsx. This context management strategy helps maintain state across different components effectively. Additionally, NFTContext.tsx is created to optimize how NFTs are displayed in the gallery, enhancing the visual presentation and interaction.

  • The application also includes a 'How to Work' modal that provides guidance for users, helping them navigate and utilize the application efficiently.

Running the Application

To run the application locally, follow these steps:

Environment Variables

Copy the example environment variable files and update them with your specific configurations.

cp .env.example .env

Running a Local IPFS Node

  1. Prerequisites
  • Ensure you have go-ipfs installed. If not, you can download it from the IPFS GitHub repository or using your package manager.
  1. Initialize IPFS Repository

    First, initialize your IPFS repository. Open your terminal and run:

    ipfs init
  2. Start the IPFS Daemon

    To start your IPFS node, run the IPFS daemon:

    ipfs daemon
    

    This will start your IPFS node and make it available locally at http://localhost:5001.

  3. Adding CORS for Your Application URL

    To allow your application to interact with the local IPFS node, you need to configure CORS settings.

    Open a new terminal window and run the following commands to set the appropriate CORS headers. Replace http://your-application-url.com ("http://127.0.0.1:5173", "http://localhost:5173") with your actual application URL.

    ipfs config --json API.HTTPHeaders.Access-Control-Allow-Origin '["http://localhost:3000", "http://127.0.0.1:5173", "http://localhost:5173"]'
    ipfs config --json API.HTTPHeaders.Access-Control-Allow-Methods '["PUT", "POST", "GET"]'
    ipfs config --json API.HTTPHeaders.Access-Control-Allow-Headers '["Authorization"]'
    
  4. Restart the IPFS Daemon

    After setting the CORS headers, restart your IPFS daemon to apply the changes:

    ipfs shutdown
    ipfs daemon
  5. Check IPFS Node Status

    Visit http://localhost:5001/webui in your browser to ensure your IPFS node is running correctly.

  6. Additional Resources IPFS Documentation

Run

  • Install all necessary packages: npm install
  • To start the development server: npm run dev
  • To build the application for production: npm run build
  • If you encounter the error ReferenceError: Buffer is not defined, as anticipated by Zama, here, adding the following two lines into main.tsx should resolved the issue:

    import { Buffer as BufferPolyfill } from "buffer/";
    (window as any).Buffer = BufferPolyfill; 

For more details and alternative solutions, refer to: Vite Discussion #2785.

Credits

About

πŸ† 1st Place Winner – Zama Bounty : encryptoNFT leverages the power of blockchain and Zama's fhEVM to deliver a secure and decentralized solution for digital rights management (DRM). This project ensures privacy-preserving operations while maintaining transparency and security on the blockchain.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •