AWS AppSync JavaScript référence de la fonction de résolution pour Amazon RDS - AWS AppSync

Les traductions sont fournies par des outils de traduction automatique. En cas de conflit entre le contenu d'une traduction et celui de la version originale en anglais, la version anglaise prévaudra.

AWS AppSync JavaScript référence de la fonction de résolution pour Amazon RDS

La AWS AppSync RDS fonction et le résolveur permettent aux développeurs d'envoyer SQL effectue des requêtes vers une base de données du cluster Amazon Aurora à l'aide RDS des données API et récupère le résultat de ces requêtes. Tu peux écrire SQL instructions envoyées aux données à l'aide du modèle sql étiqueté API par rds module AWS AppSync du module ou à l'aide des rds fonctionsselect, insertupdate, et d'removeassistance du module. AWS AppSync utilise l'ExecuteStatementaction du service de RDS données pour exécuter des SQL instructions sur la base de données.

SQLmodèle étiqueté

AWS AppSync le modèle sql balisé vous permet de créer une instruction statique capable de recevoir des valeurs dynamiques lors de l'exécution à l'aide d'expressions de modèle. AWS AppSync crée une carte variable à partir des valeurs d'expression pour créer une SqlParameterizedrequête envoyée aux données API sans serveur Amazon Aurora. Avec cette méthode, il n'est pas possible que les valeurs dynamiques transmises lors de l'exécution modifient l'instruction d'origine, ce qui pourrait entraîner une exécution involontaire. Toutes les valeurs dynamiques sont transmises sous forme de paramètres, ne peuvent pas modifier l'instruction d'origine et ne sont pas exécutées par la base de données. Cela rend votre requête moins vulnérable à SQL attaques par injection.

Note

Dans tous les cas, lors de la rédaction SQL déclarations, vous devez suivre les consignes de sécurité pour gérer correctement les données que vous recevez en entrée.

Note

Le modèle sql balisé prend uniquement en charge le transfert de valeurs variables. Vous ne pouvez pas utiliser d'expression pour spécifier dynamiquement les noms des colonnes ou des tables. Vous pouvez toutefois utiliser des fonctions utilitaires pour créer des instructions dynamiques.

Dans l'exemple suivant, nous créons une requête qui filtre en fonction de la valeur de l'colargument défini dynamiquement dans la requête GraphQL au moment de l'exécution. La valeur ne peut être ajoutée à l'instruction qu'à l'aide de l'expression de balise :

import { sql, createMySQLStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const query = sql` SELECT * FROM table WHERE column = ${ctx.args.col}` ; return createMySQLStatement(query); }

En faisant passer toutes les valeurs dynamiques par le biais de la carte des variables, nous nous appuyons sur le moteur de base de données pour gérer et nettoyer les valeurs en toute sécurité.

Création de déclarations

Les fonctions et les résolveurs peuvent interagir avec les bases de données My SQL et PostgreSQL. Utilisez createMySQLStatement et createPgStatement respectivement pour créer des instructions. Par exemple, createMySQLStatement vous pouvez créer une SQL requête Ma. Ces fonctions acceptent jusqu'à deux instructions, ce qui est utile lorsqu'une demande doit récupérer des résultats immédiatement. Avec MySQL, vous pouvez faire :

import { sql, createMySQLStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const { id, text } = ctx.args; const s1 = sql`insert into Post(id, text) values(${id}, ${text})`; const s2 = sql`select * from Post where id = ${id}`; return createMySQLStatement(s1, s2); }
Note

createPgStatementet createMySQLStatement n'échappe pas aux instructions créées à l'aide du modèle sql balisé et ne cite pas de citations.

Récupération des données

Le résultat de l'SQLinstruction exécutée est disponible dans le gestionnaire de réponses de l'context.resultobjet. Le résultat est une JSON chaîne contenant les éléments de réponse de l'ExecuteStatementaction. Une fois analysé, le résultat prend la forme suivante :

type SQLStatementResults = { sqlStatementResults: { records: any[]; columnMetadata: any[]; numberOfRecordsUpdated: number; generatedFields?: any[] }[] }

Vous pouvez utiliser cet toJsonObject utilitaire pour transformer le résultat en une liste d'JSONobjets représentant les lignes renvoyées. Par exemple :

import { toJsonObject } from '@aws-appsync/utils/rds'; export function response(ctx) { const { error, result } = ctx; if (error) { return util.appendError( error.message, error.type, result ) } return toJsonObject(result)[1][0] }

Notez que cela toJsonObject renvoie un tableau de résultats d'instructions. Si vous avez fourni une instruction, la longueur du tableau est1. Si vous avez fourni deux instructions, la longueur du tableau est2. Chaque résultat du tableau contient au 0 moins plusieurs lignes. toJsonObjectrenvoie null si la valeur du résultat n'est pas valide ou est inattendue.

Fonctions utilitaires

Vous pouvez utiliser les aides utilitaires du AWS AppSync RDS module pour interagir avec votre base de données.

L'selectutilitaire crée une SELECT instruction pour interroger votre base de données relationnelle.

Usage de base

Dans sa forme de base, vous pouvez spécifier la table que vous souhaitez interroger :

import { select, createPgStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { // Generates statement: // "SELECT * FROM "persons" return createPgStatement(select({table: 'persons'})); }

Notez que vous pouvez également spécifier le schéma dans l'identifiant de votre table :

import { select, createPgStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { // Generates statement: // SELECT * FROM "private"."persons" return createPgStatement(select({table: 'private.persons'})); }

Spécification des colonnes

Vous pouvez définir des colonnes à l'aide de columns cette propriété. S'il n'est pas défini sur une valeur, la valeur par défaut est : *

export function request(ctx) { // Generates statement: // SELECT "id", "name" // FROM "persons" return createPgStatement(select({ table: 'persons', columns: ['id', 'name'] })); }

Vous pouvez également spécifier le tableau d'une colonne :

export function request(ctx) { // Generates statement: // SELECT "id", "persons"."name" // FROM "persons" return createPgStatement(select({ table: 'persons', columns: ['id', 'persons.name'] })); }

Limites et compensations

Vous pouvez appliquer limit et répondre offset à la requête :

export function request(ctx) { // Generates statement: // SELECT "id", "name" // FROM "persons" // LIMIT :limit // OFFSET :offset return createPgStatement(select({ table: 'persons', columns: ['id', 'name'], limit: 10, offset: 40 })); }

Commander par

Vous pouvez trier vos résultats à l'aide de orderBy cette propriété. Fournissez un tableau d'objets spécifiant la colonne et une dir propriété facultative :

export function request(ctx) { // Generates statement: // SELECT "id", "name" FROM "persons" // ORDER BY "name", "id" DESC return createPgStatement(select({ table: 'persons', columns: ['id', 'name'], orderBy: [{column: 'name'}, {column: 'id', dir: 'DESC'}] })); }

Filtres

Vous pouvez créer des filtres à l'aide de l'objet de condition spéciale :

export function request(ctx) { // Generates statement: // SELECT "id", "name" // FROM "persons" // WHERE "name" = :NAME return createPgStatement(select({ table: 'persons', columns: ['id', 'name'], where: {name: {eq: 'Stephane'}} })); }

Vous pouvez également combiner des filtres :

export function request(ctx) { // Generates statement: // SELECT "id", "name" // FROM "persons" // WHERE "name" = :NAME and "id" > :ID return createPgStatement(select({ table: 'persons', columns: ['id', 'name'], where: {name: {eq: 'Stephane'}, id: {gt: 10}} })); }

Vous pouvez également créer des OR déclarations :

export function request(ctx) { // Generates statement: // SELECT "id", "name" // FROM "persons" // WHERE "name" = :NAME OR "id" > :ID return createPgStatement(select({ table: 'persons', columns: ['id', 'name'], where: { or: [ { name: { eq: 'Stephane'} }, { id: { gt: 10 } } ]} })); }

Vous pouvez également annuler une condition avec not :

export function request(ctx) { // Generates statement: // SELECT "id", "name" // FROM "persons" // WHERE NOT ("name" = :NAME AND "id" > :ID) return createPgStatement(select({ table: 'persons', columns: ['id', 'name'], where: { not: [ { name: { eq: 'Stephane'} }, { id: { gt: 10 } } ]} })); }

Vous pouvez également utiliser les opérateurs suivants pour comparer des valeurs :

Opérateur Description Types de valeurs possibles
eq Égal à nombre, chaîne, booléen
ne Non égal à nombre, chaîne, booléen
le Inférieur ou égal à nombre, chaîne
lt Inférieur à nombre, chaîne
gm Supérieur ou égal à nombre, chaîne
gt Supérieure à nombre, chaîne
contient Comme chaîne
notContains Pas comme chaîne
beginsWith Commence par le préfixe chaîne
between Entre deux valeurs nombre, chaîne
attributeExists L'attribut n'est pas nul nombre, chaîne, booléen
size vérifie la longueur de l'élément chaîne

L'insertutilitaire fournit un moyen simple d'insérer des éléments d'une seule ligne dans votre base de données avec l'INSERTopération.

Insertions d'un seul article

Pour insérer un élément, spécifiez le tableau, puis transmettez votre objet de valeurs. Les clés d'objet sont mappées aux colonnes de votre tableau. Les noms des colonnes sont automatiquement ignorés et les valeurs sont envoyées à la base de données à l'aide de la variable map :

import { insert, createMySQLStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const { input: values } = ctx.args; const insertStatement = insert({ table: 'persons', values }); // Generates statement: // INSERT INTO `persons`(`name`) // VALUES(:NAME) return createMySQLStatement(insertStatement) }

Mon cas SQL d'utilisation

Vous pouvez combiner un insert suivi d'un select pour récupérer la ligne que vous avez insérée :

import { insert, select, createMySQLStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const { input: values } = ctx.args; const insertStatement = insert({ table: 'persons', values }); const selectStatement = select({ table: 'persons', columns: '*', where: { id: { eq: values.id } }, limit: 1, }); // Generates statement: // INSERT INTO `persons`(`name`) // VALUES(:NAME) // and // SELECT * // FROM `persons` // WHERE `id` = :ID return createMySQLStatement(insertStatement, selectStatement) }

Cas d'utilisation de Postgres

Avec Postgres, vous pouvez l'utiliser returningpour obtenir des données à partir de la ligne que vous avez insérée. Il accepte * un tableau de noms de colonnes :

import { insert, createPgStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const { input: values } = ctx.args; const insertStatement = insert({ table: 'persons', values, returning: '*' }); // Generates statement: // INSERT INTO "persons"("name") // VALUES(:NAME) // RETURNING * return createPgStatement(insertStatement) }

L'updateutilitaire vous permet de mettre à jour les lignes existantes. Vous pouvez utiliser l'objet condition pour appliquer des modifications aux colonnes spécifiées dans toutes les lignes qui répondent à la condition. Supposons, par exemple, que nous ayons un schéma qui nous permet de réaliser cette mutation. Nous voulons mettre à jour les name de Person avec la id valeur de3, mais uniquement si nous les connaissons (known_since) depuis l'année 2000 :

mutation Update { updatePerson( input: {id: 3, name: "Jon"}, condition: {known_since: {ge: "2000"}} ) { id name } }

Notre résolveur de mises à jour ressemble à ceci :

import { update, createPgStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const { input: { id, ...values }, condition } = ctx.args; const where = { ...condition, id: { eq: id }, }; const updateStatement = update({ table: 'persons', values, where, returning: ['id', 'name'], }); // Generates statement: // UPDATE "persons" // SET "name" = :NAME, "birthday" = :BDAY, "country" = :COUNTRY // WHERE "id" = :ID // RETURNING "id", "name" return createPgStatement(updateStatement) }

Nous pouvons ajouter une vérification à notre condition pour nous assurer que seule la ligne dont la clé primaire est id égale à 3 est mise à jour. De même, pour Postgresinserts, vous pouvez utiliser returning pour renvoyer les données modifiées.

L'removeutilitaire vous permet de supprimer des lignes existantes. Vous pouvez utiliser l'objet de condition sur toutes les lignes qui répondent à la condition. Notez qu'il delete s'agit d'un mot clé réservé dans JavaScript. removedoit être utilisé à la place :

import { remove, createPgStatement } from '@aws-appsync/utils/rds'; export function request(ctx) { const { input: { id }, condition } = ctx.args; const where = { ...condition, id: { eq: id } }; const deleteStatement = remove({ table: 'persons', where, returning: ['id', 'name'], }); // Generates statement: // DELETE "persons" // WHERE "id" = :ID // RETURNING "id", "name" return createPgStatement(updateStatement) }

Forçage de type

Dans certains cas, vous souhaiterez peut-être plus de précisions quant au type d'objet approprié à utiliser dans votre déclaration. Vous pouvez utiliser les indications de type fournies pour spécifier le type de vos paramètres. AWS AppSync prend en charge les mêmes indications de type que les donnéesAPI. Vous pouvez convertir vos paramètres en utilisant les typeHint fonctions du AWS AppSync rds module.

L'exemple suivant vous permet d'envoyer un tableau sous forme de valeur qui est convertie en JSON objet. Nous utilisons l'->opérateur pour récupérer l'élément situé index 2 dans le JSON tableau :

import { sql, createPgStatement, toJsonObject, typeHint } from '@aws-appsync/utils/rds'; export function request(ctx) { const arr = ctx.args.list_of_ids const statement = sql`select ${typeHint.JSON(arr)}->2 as value` return createPgStatement(statement) } export function response(ctx) { return toJsonObject(ctx.result)[0][0].value }

Le casting est également utile lors de la manipulation et de DATE la comparaisonTIME, et TIMESTAMP :

import { select, createPgStatement, typeHint } from '@aws-appsync/utils/rds'; export function request(ctx) { const when = ctx.args.when const statement = select({ table: 'persons', where: { createdAt : { gt: typeHint.DATETIME(when) } } }) return createPgStatement(statement) }

Voici un autre exemple montrant comment vous pouvez envoyer la date et l'heure actuelles :

import { sql, createPgStatement, typeHint } from '@aws-appsync/utils/rds'; export function request(ctx) { const now = util.time.nowFormatted('YYYY-MM-dd HH:mm:ss') return createPgStatement(sql`select ${typeHint.TIMESTAMP(now)}`) }

Indications de type disponibles

  • typeHint.DATE- Le paramètre correspondant est envoyé sous forme d'objet de DATE ce type à la base de données. Le format accepté est YYYY-MM-DD.

  • typeHint.DECIMAL- Le paramètre correspondant est envoyé sous forme d'objet de DECIMAL ce type à la base de données.

  • typeHint.JSON- Le paramètre correspondant est envoyé sous forme d'objet de JSON ce type à la base de données.

  • typeHint.TIME- La valeur de paramètre de chaîne correspondante est envoyée sous forme d'objet de TIME ce type à la base de données. Le format accepté est HH:MM:SS[.FFF].

  • typeHint.TIMESTAMP- La valeur de paramètre de chaîne correspondante est envoyée sous forme d'objet de TIMESTAMP ce type à la base de données. Le format accepté est YYYY-MM-DD HH:MM:SS[.FFF].

  • typeHint.UUID- La valeur de paramètre de chaîne correspondante est envoyée sous forme d'objet de UUID ce type à la base de données.