SubQuery開發者指南丨GraphQL 架構(GraphQL Schema) - 冷萃財經

SubQuery開發者指南丨GraphQL 架構(GraphQL Schema)

定義實體(Defining Entities)

 

schema.graphql 文件定義了各種 GraphQL 架構。 由於 GraphQL 查詢語言的工作方式,模式文件本質上決定了來自 SubQuery 的數據的形狀。 要了解有關如何使用 GraphQL 架構語言編寫的更多信息,我們建議查看架構和類型()

 

重要提示:當您對架構文件(schema file)進行任何更改時,請確保使用 yarn codegen 代碼來生成重新生成類型目錄。

 

實體(Entities)

 

每個實體(entity)都必須定義它的必填欄位id與ID的類型!它用作主鍵,並且在所有相同類型的實體中是唯一的。

 

實體(entity)中不可為空的欄位由「 ! 」來表示。 請看下面的例子:

 

type Example @entity { id: ID! # id field is always required and must look like this name: String! # This is a required field address: String # This is an optional field }

 

支持的標量和類型(Supported scalars and types)

 

我們目前支持流動標量類型:

• ID

• Int

• String

• BigInt

• Date

• Boolean

• <EntityName>  用於嵌套關係實體,您可以使用定義的實體名稱作為欄位之一。 請參閱實體關係()。

• JSON 也可以存儲結構化數據,請參閱 JSON 類型()。

 

按非主鍵欄位進行索引(Indexing by non-primary-key field)

 

為了提高查詢性能,只需在非主鍵欄位上執行 @index 注釋即可索引實體欄位。

 

但是,我們不允許用戶在任何 JSON 對象上添加 @index 注釋。 默認情況下,索引會自動添加到外鍵和資料庫中的 JSON 欄位,但只是為了增強查詢服務性能。

 

這有一個例子。

 

type User @entity { id: ID! name: String! @index(unique: true) # unique can be set to true or false title: Title! # Indexes are automatically added to foreign key field } type Title @entity { id: ID! name: String! @index(unique:true) }

 

假設我們知道這個用戶的名字,但我們不知道確切的 id 值,我們可以在名稱欄位後面添加@index,而不是提取所有用戶然後按名稱過濾。 這使得查詢速度更快,我們還可以添加 unique: true 以確保唯一性。

 

如果欄位不唯一,則最大結果集為 100

 

當代碼生成運行時,這將自動在 User 模型下創建一個 getByName,然後外鍵欄位 title 將創建一個 getByTitleId 方法,這兩者都可以在映射函數中直接訪問。

 

/* Prepare a record for title entity */ INSERT INTO titles (id, name) VALUES (‘id_1’, ‘Captain’)

 

// Handler in mapping function import {User} from “../types/models/User” import {Title} from “../types/models/Title” const jack = await User.getByName(‘Jack Sparrow’); const captainTitle = await Title.getByName(‘Captain’); const pirateLords = await User.getByTitleId(captainTitle.id); // List of all Captains

 

實體關係(Entity Relationships)

 

一個實體(entity)通常與其他實體(entity)有嵌套關係。 默認情況下,將欄位值設置為另一個實體(entity)名稱將定義這兩個實體(entity)之間的一對一關係。

 

可以使用以下示例配置不同的實體(entity)關係(一對一、一對多和多對多)。

 

  • 一對一關係(One-to-One Relationships)

 

當只有一個實體映射到另一個實體時,一對一關係是默認的。

 

例子:一本護照只屬於一個人,一個人只有一本護照(在這個例子中):

 

type Person @entity { id: ID! } type Passport @entity { id: ID! owner: Person! }

 

或者

 

type Person @entity { id: ID! passport: Passport! } type Passport @entity { id: ID! owner: Person! }

 

  • 一對多關係(One-to-Many relationships)

 

您可以使用方括弧表示一個欄位類型包括多個實體。

 

示例:一個人可以擁有多個帳戶。

 

type Person @entity { id: ID! accounts: [Account] } type Account @entity { id: ID! publicAddress: String! }

 

  • 多對多關係(Many-to-Many relationships)

 

多對多關係可以通過實現一個映射實體(mapping entity)來連接其他兩個實體(entity)來實現。

 

示例:每個人都是多個組 (PersonGroup) 的一部分,並且組有多個不同的人 (PersonGroup)。

 

type Person @entity { id: ID! name: String! groups: [PersonGroup] } type PersonGroup @entity { id: ID! person: Person! Group: Group! } type Group @entity { id: ID! name: String! persons: [PersonGroup] }

 

此外,可以在中間實體(entity)的多個欄位中創建同一實體(entity)的連接。

 

例如,一個賬戶可以實現多次轉賬,每次轉賬都有一個源賬戶和目的地賬戶。

 

這將通過 Transfer 層在兩個 Accounts(from 和 to)之間建立雙向關係。

 

type Account @entity { id: ID! publicAddress: String! } type Transfer @entity { id: ID! amount: BigInt from: Account! to: Account! }

 

反向查找(Reverse Lookups)

 

為了使一個實體(entity)能夠反向查詢到一個關係,請將 @derivedFrom 附加到該欄位並指向另一個實體(entity)的反向查找欄位。

 

這會在可以查詢的實體(entity)上創建一個虛擬欄位。

 

通過將 sentTransfer 或 receivedTransfer 設置為從相應的 from 或 to 欄位得出的值,可以從帳戶實體中訪問「來自」 賬戶的轉移。

 

type Account @entity { id: ID! publicAddress: String! sentTransfers: [Transfer] @derivedFrom(field: “from”) receivedTransfers: [Transfer] @derivedFrom(field: “to”) } type Transfer @entity { id: ID! amount: BigInt from: Account! to: Account! }

 

JSON 類型(JSON type)

 

我們支持將數據保存為 JSON 類型(JSON type),這是一種存儲結構化數據的快速方式。 我們將自動生成相應的 JSON 介面來查詢這些數據,並節省您定義和管理實體(entities)的時間。

 

我們建議用戶在以下場景中使用 JSON 類型:

 

• 在單個欄位中存儲結構化數據比創建多個單獨的實體(entities)更易於管理。

 

• 保存任意鍵/值用戶首選項(其中值可以是布爾值、文本或數字,並且不用為不同的數據類型設置單獨的列)

 

• 架構是不穩定的並且經常變化

 

定義 JSON 指令(Define JSON directive)

 

通過在實體中添加 jsonField 注釋,將該屬性定義為 JSON 類型。 這將自動為您項目中 types/interfaces.ts 下的所有 JSON 對象生成介面,您也可以在映射函數中訪問它們。

 

與實體不同,jsonField 指令對象不需要任何 id 欄位。 JSON 對象還可以與其他 JSON 對象嵌套。

 

type AddressDetail @jsonField { street: String! district: String! } type ContactCard @jsonField { phone: String! address: AddressDetail # Nested JSON } type User @entity { id: ID! contact: [ContactCard] # Store a list of JSON objects }

 

查詢 JSON 欄位(Querying JSON fields)

 

使用 JSON 類型的缺點是過濾時對查詢效率的影響很小,因為每次執行文本搜索時,都是針對整個實體來進行的。

 

但是,在我們的查詢服務中,影響仍然可以接受。 下面是一個示例,說明如何在 GraphQL 查詢中對 JSON 欄位使用 contains 運算符來查找擁有包含「0064」的電話號碼的前 5 個用戶。

 

#To find the the first 5 users own phone numbers contains ‘0064’. query{ user( first: 5, filter: { contactCard: { contains: [{ phone: “0064” }] } }){ nodes{ id contactCard } } }

冷萃財經原創,作者:Awing,轉載請註明出處:https://www.lccjd.top/2021/09/27/subquery%e5%bc%80%e5%8f%91%e8%80%85%e6%8c%87%e5%8d%97%e4%b8%a8graphql-%e6%9e%b6%e6%9e%84%ef%bc%88graphql-schema%ef%bc%89/?variant=zh-tw

0

掃一掃,分享到微信

猜你喜歡

文章評論

電子郵件地址不會被公開。 必填項已用*標註

後發表評論

    上一篇

    軟銀和 a16z 為何重金押注 NFT 市場?

    下一篇

    全景式解讀區塊鏈橋的重要性、各橋的優缺點及未來形態

    微信公眾號

    微信公眾號