Como vimos en la publicación anterior, los contratos de Olive Oil Trust emiten muchos eventos con datos importantes que deben recopilarse, junto con metadatos que deben agregarse.
Suscribirse a todos estos eventos en una aplicación de front end sería ineficiente y cualquier dato que estos eventos hubiesen emitido antes de la suscripción no se consideraría.
Además, procesar todos los eventos emitidos desde el bloque en el que se desplegó el contrato en adelante y filtrarlos también sería inconveniente.
Es por eso que utilizo un subgrafo, el cual actúa en la práctica como una capa de indexación entre los contratos y la interfaz de usuario.
En esta publicación trato de explicar cómo crear un subgrafo utilizando TheGraph que se puede usar para consultar datos de Olive Oil Trust de manera eficiente.
En este artículo
El manifiesto del subgrafo
El manifiesto del subgrafo especifica qué contratos indexa el subgrafo, a qué eventos reaccionar y cómo se asignan los datos de eventos a entidades que Graph Node almacena y permite consultar.
El workspacehardhat-env en el monorepositorio de Olive Oil Trust genera una carpeta deployments que contiene una carpeta para cada red en la que se despliegan los contratos.
Cada carpeta de una red contiene un archivo JSON para cada contrato que se despliega en esa red.
Estos archivos JSON reúnen información clave que se necesita para generar un manifiesto que permita que el subgrafo recopile datos de la cadena de bloques de manera eficiente.
El manifiesto se genera automáticamente a partir de una plantilla, utilizando el scriptgenerateSubgraph en subgraph/scripts/index.ts, para admitir múltiples despliegues en diferentes redes de forma dinámica.
Esto se realiza en el script npm "pre" npm run precodegen.El scriptnpm run codegen ejecuta el comando graph codegen:
generateSubgraph reemplazará los campos clave con valores de un archivo JSON en subgraph/src/generated/config con el nombre que coincida con la red en el parámetro--despliegue.
Ese archivo JSON se genera cuando se ejecutanpm ejecutar hardhat: compartir. Si hay implementaciones en varias redes, se generará un archivo JSON para cada red.
Campos
subgraph/templates/subgraph.template.yaml contiene cinco campos con datos estáticos:
subgraph/templates/subgraph.template.yaml
Puede encontrar la especificación completa para los manifiestos de subgrafos aquí.
La plantilla de subgrafo también contiene el campo dataSources, que presenta algunos campos con claves entre corchetes que son reemplazadas por valores por generateSubgraph.
dataSources define los datos que se incorporan, así como la lógica de transformación para derivar el estado de las entidades del subgrafo en función de los datos de origen.
Hay una fuente de datos por contrato. Por ejemplo, veamos la fuente de datos del contrato de Bottling Company:
subgraph/templates/subgraph.template.yaml
Las claves {{BottlingCompanyModule}}, {{network}}, {{BottlingCompanyAddress}} y {{BottlingCompanyStartBlock}} serán reemplazadas por el módulo de ese contrato, el nombre de la red, la dirección del contrato y el bloque de inicio respectivamente.
El manifiesto del subgrafo de Olive Oil Trust está diseñado de tal manera que mappings se pueden reutilizar entre los contratos de miembros que heredan del mismo contrato de Olive Oil Trust.
Por ejemplo, Bottling Company 2 -siendo una planta embotelladora- usaría el mismo mapping que Bottling Company ya que ambas heredarían de BottlingPlantUpgradeable.sol.
Los ABIs están en src/generated/abis y también se generan cuando se ejecuta npm run hardhat:share.
La primera fuente de datos de los contratos del mismo tipo (por ejemplo, tokens dependientes) contiene una clave del nombre del módulo del que hereda en lugar del nombre de ese contrato:
subgraph/templates/subgraph.template.yaml
Esto es para que existan tipos generados con la nomenclatura de los módulos de Olive Oil Trust que se pueden importar en los archivos AssemblyScript de mapping.
El esquema GraphQL
Para llegar a un buen concepto sobre cómo diseñar el esquema GraphQL, me he centrado en cómo se definen las entidades en los subgrafos de OpenZeppelin.
La idea es construir un subgrafo denso, vinculando tantos tipos de datos como tenga sentido para hacer consultas complejas tan fácil y eficientemente como sea posible.
He seguido algunas pautas de este artículo del blog de OpenZeppelin para diseñar el esquema.
Todo comienza con cómo se diseñan los contratos para que todo pueda ser indexado únicamente usando eventos, pero también hay algunos otros aspectos que se resumen en las siguientes secciones.
Creación de entidades para conceptos de alto nivel
En Olive Oil Trust existen conceptos de alto nivel como tokens, escrows, certificados, cuentas, saldos, etc.
Por ejemplo, las entidades Account son objetos que se utilizan para recopilar datos sobre cualquier dirección que aparece en un evento (incluida la dirección del contrato que emitió el evento).
Podemos ver el tipo Account abajo:
subgraph/schema.graphql
Creación de entidades para conceptos de bajo nivel
En Olive Oil Trust, los conceptos de bajo nivel se refieren a los eventos que emiten los contratos de Olive Oil Trust.
Por ejemplo, el contrato de certificado emite un evento de tipo TokenCertified cuando un certificador certifica un tipo de token y entidad TokenCertification dará la estructura del objeto que contendrá los datos de la emisión del evento:
subgraph/schema.graphql
Todas las entidades para conceptos de bajo nivel son inmutables e implementan la interfaz IEvent ya que todos comparten algunos campos.
Proporcionando enlaces cruzados entre entidades
Como mencionamos anteriormente, los subgrafos densos ayudan a manejar consultas complejas ya que hay muchas relaciones entre entidades.
Digamos que iniciamos sesión como certificador en una aplicación de front end que consume datos de este subgrafo y querríamos que la aplicación reciba en una sola consulta el nombre del miembro que inició sesión, su función, todos los certificados que ha emitido junto con los tipos de token que estos certificados certifican y todos los tipos de token que existen en Olive Oil Trust.
Un ejemplo de esa consulta podría ser:
Un certificado puede certificar varios tipos de tokens y un tipo de token puede estar certificado por varios certificados.
Podemos ver en la consulta anterior que el campo resaltado certificates, que representa un array de entidades Certificate, contiene el campo tokenTypes.
Este campo representa un array de entidades TokenTypeCertificateMapping que es una relación de muchos a muchos.
Se recomienda utilizar tablas de mapping para almacenar datos de relaciones de muchos a muchos de manera eficaz
El tipo TokenTypeCertificateMapping se puede ver a continuación:
subgraph/schema.graphql
Escribiendo mappings
Los mappings son funciones que asignan datos de Ethereum a objetos definidos como entidades.
Como vimos anteriormente, las entidades pueden verse como objetos que contienen datos, estos datos se mapean desde la cadena de bloques a través de funciones de mapeo.
Todas los mappings en el subgrafo de Olive Oil Trust se pueden encontrar en subgraph/src/mappings.
npm run subgraph:codegen genera el manifiesto de subgrafo y los tipos que se necesitan en las funciones de mapeo.
Digamos que Bottling Company acuña un nuevo lote de tokens de botellas de aceite de oliva de cierto tipo.
La función mint en hardhat-env/contracts/tokens/BottlingCompanyOliveOilBottle.sol emitirá un evento TokenTransferred que contiene los siguientes datos:
Para almacenar estos datos en el Graph Node y dado que BottlingCompanyOliveOilBottle es un token dependiente, el subgrafo manejará este evento con la función de mapeo handleTokenTransferred en subgraph/src/mappings/DependentTokenUpgradeable.ts:
subgraph/templates/subgraph.template.yaml
La función handleTokenTransferred luego escribirá varias entidades con los datos procesados en el store del Graph Node:
Como podemos ver, se crearán varias entidades nuevas en este proceso, incluida una entidad TokenTransfer que tiene una relación con la entidad Balance a través del campo balance.
Estas entidades contendrán datos emitidos por el evento y reflejarán los cambios en el saldo de la dirección de Bottling Company en BottlingCompanyOliveOilBottle.sol.
Metadatos
Para simular el flujo de trabajo de la larga cadena de valor del aceite de oliva en la aplicación de front-end (que veremos en la próxima publicación), agregamos metadatos figurados a los datos escritos en las entidades de Certificate y TokenType:
subgraph/src/utils/entities/TokenType.ts
En este ejemplo, los metadatos son constantes locales, pero podrían almacenarse en IPFS.
@graphprotocol/graph-ts nos proporciona soporte para tratar con ellos. Una vez que tuviéramos el hash de IPFS, simplemente podríamos hacer ipfs.cat(hash) para obtener los datos JSON.
Desplegando el subgrafo
Una vez que el nodo de gráfico local está en funcionamiento, el proceso de creación y despliegue del subgrafo es bastante sencillo:
Ahora la interfaz de usuario podrá consumir datos del subgrafo una vez que se desplieguen los contratos y el subgrafo indexe los datos de los eventos a los que reacciona.
Introducción a una serie de artículos acerca de Olive Oil Trust
¿Preparado para #buidl?
¿Está interesado en Web3 o en las sinergias entre la tecnología blockchain, la inteligencia artificial y el conocimiento cero?. Entonces, no dude en contactarme por e-mail o en mi perfil de LinkedIn. También me puede encontrar en GitHub.