本文為英文版的機器翻譯版本,如內容有任何歧義或不一致之處,概以英文版為準。
AWS AppSync 解析器映射範本內容參考
注意
我們現在主要支援 APPSYNC_JS 執行期及其文件。請考慮在此處使用 APPSYNC_JS 執行期及其指南。
AWS AppSync 定義一組變數和函數,用於使用解析器映射範本。這樣可讓使用 GraphQL 對資料進行邏輯運算更簡單。本文件說明這些函式,並提供範本使用範例。
正在存取 $context
$context
變數是一種映射,保存您的解析程式叫用的所有情境資訊。其結構如下:
{ "arguments" : { ... }, "source" : { ... }, "result" : { ... }, "identity" : { ... }, "request" : { ... }, "info": { ... } }
注意
如果您嘗試透過索引鍵存取字典/地圖項目 (例如 中的項目context
) 來擷取值,則 Velocity 範本語言 (VTL) 可讓您直接使用符號 <dictionary-element>.<key-name>
。不過,這可能不是所有案例皆適用,例如當索引鍵名稱有特殊字元 (例如,底線「_」)。建議您一律使用 <dictionary-element>.get("<key-name>")
表示法。
$context
映射中每個欄位的定義如下:
$context
欄位
-
result
-
此解析程式結果的容器。此欄位僅適用於回應映射範本。
例如,如果您正在解決下列查詢
author
的欄位:query { getPost(id: 1234) { postId title content author { id name } } }
那麼,在處理回應映射範本時,可用的完整
$context
變數可能是:{ "arguments" : { id: "1234" }, "source": {}, "result" : { "postId": "1234", "title": "Some title", "content": "Some content", "author": { "id": "5678", "name": "Author Name" } }, "identity" : { "sourceIp" : ["x.x.x.x"], "userArn" : "arn:aws:iam::123456789012:user/appsync", "accountId" : "666666666666", "user" : "AIDAAAAAAAAAAAAAAAAAA" } }
-
prev.result
-
在管道解析程式中執行任何先前操作的結果。
如果先前的操作是管道解析程式的映射前範本,則
$ctx.prev.result
代表範本評估的輸出,並提供給管道中的第一個函數。如果先前操作是第一個函數,則
$ctx.prev.result
會顯示第一個函數的輸出,並將資料提供給管道中的第二個函數。如果先前的操作是最後一個函數,則
$ctx.prev.result
代表最後一個函數的輸出,並提供給管道解析程式的映射後範本。 -
info
-
包含有關 GraphQL 請求資訊的物件。如需此欄位的結構,請參閱資訊。
Identity
包含有關發起人資訊的 identity
區段。本節的形狀取決於 的授權類型 AWS AppSync API。
如需 AWS AppSync 安全選項的詳細資訊,請參閱授權和身分驗證。
-
API_KEY
授權 -
identity
欄位未填入。 AWS_LAMBDA
授權-
identity
包含resolverContext
金鑰,其中包含由 Lambda 函數傳回的相同resolverContext
內容,以授權請求。 -
AWS_IAM
授權 -
identity
具有下列表單:{ "accountId" : "string", "cognitoIdentityPoolId" : "string", "cognitoIdentityId" : "string", "sourceIp" : ["string"], "username" : "string", // IAM user principal "userArn" : "string", "cognitoIdentityAuthType" : "string", // authenticated/unauthenticated based on the identity type "cognitoIdentityAuthProvider" : "string" // the auth provider that was used to obtain the credentials }
-
AMAZON_COGNITO_USER_POOLS
授權 -
identity
具有下列表單:{ "sub" : "uuid", "issuer" : "string", "username" : "string" "claims" : { ... }, "sourceIp" : ["x.x.x.x"], "defaultAuthStrategy" : "string" }
每個欄位的定義如下:
-
accountId
-
來電者 AWS 的帳戶 ID。
-
claims
-
使用者擁有的宣告。
-
cognitoIdentityAuthType
-
經身分驗證或未經身分驗證 (根據身分類型)。
-
cognitoIdentityAuthProvider
-
以逗號分隔的外部身分提供者資訊清單,用於取得用來簽署請求的憑證。
-
cognitoIdentityId
-
來電者的 Amazon Cognito 身分 ID。
-
cognitoIdentityPoolId
-
與呼叫者相關聯的 Amazon Cognito 身分集區 ID。
-
defaultAuthStrategy
-
此發起人 (
ALLOW
或DENY
) 的預設授權策略。 -
issuer
-
字符發行者。
-
sourceIp
-
AWS AppSync 接收呼叫者的來源 IP 地址。如果請求不包含
x-forwarded-for
標頭,來源 IP 值只會包含來自TCP連線的單一 IP 地址。如果請求包含x-forwarded-for
標頭,則除了TCP連線的 IP 地址之外,來源 IP 也是來自x-forwarded-for
標頭的 IP 地址清單。 -
sub
-
已UUID驗證使用者的 。
-
user
-
IAM 使用者。
-
userArn
-
IAM 使用者的 Amazon Resource Name (ARN)。
-
username
-
已驗證使用者的使用者名稱。如果是
AMAZON_COGNITO_USER_POOLS
授權,使用者名稱的值是屬性 cognito:username 的值。在AWS_IAM
授權的情況下,使用者名稱的值是 AWS 使用者主體的值。如果您使用IAM來自 Amazon Cognito 身分集區的憑證授權,建議您使用cognitoIdentityId
。
存取請求標頭
AWS AppSync 支援從用戶端傳遞自訂標頭,並使用 存取 GraphQL 解析程式中的這些標頭$context.request.headers
。然後,您可以將標頭值用於將資料插入資料來源或授權檢查等動作。您可以使用 $curl
搭配來自命令列的 API金鑰來使用單一或多個請求標頭,如下列範例所示:
單一標頭範例
假設您設定使用 custom
值的 nadia
標頭,如下所示:
curl -XPOST -H "Content-Type:application/graphql" -H "custom:nadia" -H "x-api-key:<API-KEY-VALUE>" -d '{"query":"mutation { createEvent(name: \"demo\", when: \"Next Friday!\", where: \"Here!\") {id name when where description}}"}' https://<ENDPOINT>/graphql
可使用 $context.request.headers.custom
對此進行存取。例如,VTLDynamoDB 可能位於下列位置:
"custom": $util.dynamodb.toDynamoDBJson($context.request.headers.custom)
多個標頭範例
您也可以將多個標頭傳遞到單一要求,並在解析程式映射範本中存取這些要求。例如,如果custom
標頭設定為兩個值:
curl -XPOST -H "Content-Type:application/graphql" -H "custom:bailey" -H "custom:nadia" -H "x-api-key:<API-KEY-VALUE>" -d '{"query":"mutation { createEvent(name: \"demo\", when: \"Next Friday!\", where: \"Here!\") {id name when where description}}"}' https://<ENDPOINT>/graphql
然後,您可以將這些存取為陣列,例如 $context.request.headers.custom[1]
。
注意
AWS AppSync 不會在 中公開 Cookie 標頭$context.request.headers
。
存取請求自訂網域名稱
AWS AppSync 支援設定自訂網域,您可以使用該網域來存取 GraphQL 和 的即時端點APIs。使用自訂網域名稱提出請求時,您可以使用 取得網域名稱$context.request.domainName
。
使用預設 GraphQL 端點網域名稱時,值為 null
。
Info
info
區段包含有關 GraphQL 請求的資訊。本節包含下列表單:
{ "fieldName": "string", "parentTypeName": "string", "variables": { ... }, "selectionSetList": ["string"], "selectionSetGraphQL": "string" }
每個欄位的定義如下:
-
fieldName
-
目前正在解析的欄位名稱。
-
parentTypeName
-
目前正在解析的欄位父類型名稱。
-
variables
-
包含傳遞給 GraphQL 請求之所有變數的映射。
-
selectionSetList
-
此清單表示 GraphQL 選取範圍中的欄位。別名的欄位僅由別名名稱參考,而非欄位名稱。以下範例詳細說明這一點。
-
selectionSetGraphQL
-
選取集的字串表示法,格式為 GraphQL 結構描述定義語言 (SDL)。雖然片段不會合併至選取集,但會保留內嵌片段,如下列範例所示。
注意
在 $utils.toJson()
上使用 時context.info
,預設不會序列化 selectionSetGraphQL
和 selectionSetList
傳回的值。
例如,如果您解析下列查詢的 getPost
欄位:
query { getPost(id: $postId) { postId title secondTitle: title content author(id: $authorId) { authorId name } secondAuthor(id: "789") { authorId } ... on Post { inlineFrag: comments: { id } } ... postFrag } } fragment postFrag on Post { postFrag: comments: { id } }
那麼,在處理映射範本時,可用的完整 $context.info
變數可能是:
{ "fieldName": "getPost", "parentTypeName": "Query", "variables": { "postId": "123", "authorId": "456" }, "selectionSetList": [ "postId", "title", "secondTitle" "content", "author", "author/authorId", "author/name", "secondAuthor", "secondAuthor/authorId", "inlineFragComments", "inlineFragComments/id", "postFragComments", "postFragComments/id" ], "selectionSetGraphQL": "{\n getPost(id: $postId) {\n postId\n title\n secondTitle: title\n content\n author(id: $authorId) {\n authorId\n name\n }\n secondAuthor(id: \"789\") {\n authorId\n }\n ... on Post {\n inlineFrag: comments {\n id\n }\n }\n ... postFrag\n }\n}" }
selectionSetList
只會公開屬於目前類型的欄位。如果目前類型是介面或聯合,則只會公開屬於介面的所選欄位。例如,提供下列結構描述:
type Query { node(id: ID!): Node } interface Node { id: ID } type Post implements Node { id: ID title: String author: String } type Blog implements Node { id: ID title: String category: String }
以及下列查詢:
query { node(id: "post1") { id ... on Post { title } ... on Blog { title } } }
$ctx.info.selectionSetList
以Query.node
欄位解析度呼叫 時,只會id
公開:
"selectionSetList": [ "id" ]
處理輸入
應用程式必須處理不受信任的輸入,以防止任何外部方使用其預期用途之外的應用程式。由於 $context
包含 $context.arguments
、、$context.identity
$context.result
、 $context.info.variables
和 等屬性中的使用者輸入$context.request.headers
,因此必須小心在映射範本中清理其值。
由於映射範本代表 JSON,因此輸入消毒的形式是從代表使用者輸入的字串逸出JSON預留字元。最佳實務是使用 $util.toJson()
公用程式,在將JSON保留字元放入映射範本時,從敏感字串值中逸出保留字元。
例如,在下列 Lambda 請求映射範本中,由於我們存取了不安全的客戶輸入字串 ($context.arguments.id
),因此我們使用 包裝,$util.toJson()
以防止未逸出的JSON字元破壞JSON範本。
{ "version": "2017-02-28", "operation": "Invoke", "payload": { "field": "getPost", "postId": $util.toJson($context.arguments.id) } }
與下面的映射範本相反,我們在其中直接插入$context.arguments.id
而不進行消毒。這不適用於包含未逸出引號或其他JSON預留字元的字串,並且可能會讓您的範本保持開啟狀態失敗。
## DO NOT DO THIS { "version": "2017-02-28", "operation": "Invoke", "payload": { "field": "getPost", "postId": "$context.arguments.id" ## Unsafe! Do not insert $context string values without escaping JSON characters. } }