Безсерверный фреймворк + AWS + Lambda + DynamoDB + GraphQL + Apollo Server = Не удается заставить POST-запрос работать - PullRequest
0 голосов
/ 05 июля 2018

Хорошо, я нахожусь в этой проблеме с нескольких дней.

Я пытаюсь взять этот урок: https://serverless.com/blog/make-serverless-graphql-api-using-lambda-dynamodb/

И заставить его работать с apollo-server-lambda. Эта учебная помощь:

https://medium.com/vessels/apollo-server-serverless-graphql-bliss-68e8e15195ac

Проблема в том, что ничего не работает (для меня), когда вы пытались использовать лямбда-сервер apollo с реальным соединением с DynamoDB. У меня есть список ошибок, которые я больше не могу вспомнить, это просто обескураживает.

Вот мой код:

# serverless.yml
service: graphql-api

provider:
  name: aws
  runtime: nodejs6.10
  region: eu-west-3
  stage: dev
  environment:
    DYNAMODB_TABLE: ${self:service}-${self:provider.stage}
  iamRoleStatements:
    - Effect: Allow
      Action:
        - dynamodb:GetItem
        - dynamodb:UpdateItem
      Resource: "arn:aws:dynamodb:${opt:region, self:provider.region}:*:table/${self:provider.environment.DYNAMODB_TABLE}"

resources:
  Resources:
    NicknamesTable:
      Type: 'AWS::DynamoDB::Table'
      Properties:
        AttributeDefinitions:
          - AttributeName: firstName
            AttributeType: S
        KeySchema:
          - AttributeName: firstName
            KeyType: HASH
        ProvisionedThroughput:
          ReadCapacityUnits: 1
          WriteCapacityUnits: 1
        TableName: ${self:provider.environment.DYNAMODB_TABLE}

functions:
  graphql:
    handler: handler.graphql
    events:
      - http:
          path: graphql
          method: post
          cors: true
      - http:
          path: graphql
          method: get
          cors: true

И мой обработчик:

# handler.js
const AWS = require('aws-sdk');
const server = require("apollo-server-lambda");
const makeExecutableSchema = require('graphql-tools').makeExecutableSchema;
const dynamoDb = new AWS.DynamoDB.DocumentClient();

const promisify = foo => new Promise((resolve, reject) => {
    foo((error, result) => {
        if (error) {
            reject(error)
        } else {
            resolve(result)
        }
    })
})

const getGreeting = firstName => promisify(callback =>
    dynamoDb.get({
        TableName: process.env.DYNAMODB_TABLE,
        Key: { firstName },
    }, callback))
    .then(result => {
        if (!result.Item) {
            return firstName
        }
        return result.Item.nickname
    })
    .then(name => `Hello, ${name}.`)

// add method for updates
const changeNickname = (firstName, nickname) => promisify(callback =>
    dynamoDb.update({
        TableName: process.env.DYNAMODB_TABLE,
        Key: { firstName },
        UpdateExpression: 'SET nickname = :nickname',
        ExpressionAttributeValues: {
            ':nickname': nickname
        }
    }, callback))
    .then(() => nickname)

const typeDefs = `
    type Query {
        greeting(firstName: String!): String
    }
    type Mutation {
        changeNickname(
            firstName: String!
            nickname: String!
        ): String
    }
`;

const resolvers = {
    Query: {
        greeting: (_, { firstName }) => getGreeting(firstName),
    },
    Mutation: {
        changeNickname: (_, { firstName, nickname }) => changeNickname(firstName, nickname),
    }
};

exports.graphql = function (event, context, callback) {
    const callbackFilter = function (error, output) {
        output.headers = output.header || {};
        output.headers['Access-Control-Allow-Origin'] = '*';
        callback(error, output);
    };
    const handler = server.graphqlLambda({ schema: makeExecutableSchema({ typeDefs, resolvers }) });

    return handler(event, context, callbackFilter);
};

Я пытался использовать Аполлон 1 и 2, ничего не получалось. Я вернулся к версии 1, так как есть больше сообщений на форуме об этом. В основном у меня «внутренняя ошибка сервера». Я пробовал разные версии сервера, которые я нашел на apollo docs, но все мои запросы не были выполнены, с помощью curl на терминале или непосредственно на API-шлюзе на AWS для тестирования функции. Я пишу тело запроса после этого документа: https://www.apollographql.com/docs/apollo-server/requests.html

Вот мой журнал cloudwatch:

(node:1) UnhandledPromiseRejectionWarning: Unhandled promise rejection (rejection id: 1): SyntaxError: Unexpected token = in JSON at position 8

Любая помощь будет очень признателен !!

1 Ответ

0 голосов
/ 08 июля 2018

Хорошо, я понял проблему. Когда вы используете функцию put, вы не можете получить новый элемент, который вы просто вставляете в DynamoDB. Вы должны поместить "ReturnValues: 'ALL_OLD'" в ваш объект params (первый аргумент функции put), чтобы не выдавалось никакой ошибки, и вы должны возвращать нужные значения, поскольку вы только что ввели их в БД.

Подробнее здесь:

https://github.com/aws/aws-sdk-js/issues/803

...