热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

将Apollo服务器调试为AWSLambda函数

我按照ApolloServer的指示进行了部署,将其部署为AWSlambda。

我按照Apollo Server的指示进行了部署,将其部署为AWS lambda。
https://www.apollographql.com/docs/apollo-server/deployment/lambda/
使用了无服务器框架,并且在East-2区域运行良好。

我将示例扩展为使用PostGres数据库进行查询(我使用了npm sequalize软件包)。当我作为ApolloServer和本地postresql DB运行时,可以使用相同的代码。我这样做是为了也可以切换到apollo-server-lambda。我有一个if语句,它根据是否为lambda来更改与数据库的连接。

我遇到的问题是不会打入数据库的查询工作正常。但是对数据库的graphql查询返回:

{
"error": {
"message": "Internal server error"
}
}

好的,那么现在如何调试nodejs lambda函数?

lambda管理控制台确实具有测试操作。我重新定义了hello world测试以将其用作测试:

{
"operationName": null,"variables": {},"query": "{users {id firstName lastName addressnumber streetName city email createdAt updatedAt }}"
}

但是,这似乎不是调用lambda函数的正确方法。因为日志返回:

{
"body": "Apollo Server supports only GET/POST requests.","statusCode": 405,"headers": {
"Allow": "GET,POST"
}
}





server.js

const { ApolloServer } = require('apollo-server')
const { ApolloServer: ApolloServerLambda } = require('apollo-server-lambda')
const { typeDefs,resolvers,connect } = require('./schema.js')
// The ApolloServer constructor requires two parameters: your schema
// definition and your set of resolvers.
async function setup(server) {
let { url } = await server.listen()
console.log(`? Server ready at ${url}`)
await connect("local")
}
async function awsSetup() {
await connect("aws")
}
if (process.env.username == 'ysg4206') {
const server = new ApolloServer({ typeDefs,resolvers })
setup(server)
} else {
const server = new ApolloServerLambda({ typeDefs,resolvers })
//awsSetup()
exports.graphqlHandler = server.createHandler({
playground: true,introspection: true,cors: {
origin: '*',credentials: true,},context: ({ event,context }) => { return (
{
headers: event.headers,functionName: context.functionName,event,context
})
}
})
}



schema.js

const { gql } = require('apollo-server')
const { DB } = require('./db')
const { GraphQLDateTime } = require('graphql-iso-date')
exports.typeDefs = gql`
scalar DateTime
type User {
id: Int
"English First Name"
firstName: String
lastName: String
addressnumber: Int
streetName: String
city: String
email: String
createdAt: DateTime
updatedAt: DateTime
}
input UserType {
"Hebrew First Name"
firstName: String
lastName: String
addressnumber: Int
streetName: String
city: String
email: String
}
type Query {
users: [User]
findUser(firstName: String): User
hello(reply: String): String
}
type Mutation {
addUser(user: UserType): User!
}
type Subscription {
newUser: User!
}
`
exports.resolvers = {
Query: {
// users: async () => {
// let users = await DB.findAll()
// return users
// },users: () => DB.findAll(),findUser: async (_,{ firstName }) => {
let who = await DB.findFirst(firstName)
return who
},hello: (_,{ reply },context) => {
console.log(`hello with reply ${reply}`)
console.log(`context : ${JSON.stringify(reply,null,4)}`)
return reply
}
},Mutation: {
addUser: async (_,args) => {
let who = await DB.addUser(args.user)
return who
}
}
}
exports.cOnnect= async function connect(where) {
await DB.dbSetup(where)
await DB.populate()
let users = await DB.findAll()
console.log(users)
}


调试lambda函数非常困难!如果通过CloudWatch配置了日志,则可以尝试通过日志进行挖掘,但这并不能总为您提供可行的堆栈跟踪,而且要找到您要查找的确切调用也很困难。

您是否尝试过使用无服务器框架仪表板进行部署?这将帮助您获得完整的堆栈跟踪以及日志。您只需在应用程序根目录中运行serverless命令即可​​开始使用。

更多信息可用here

全面披露-我在Serverless Framework上为Serverless Inc.工作。

,

适用于部署无服务器应用程序并在转到其AWS终端节点时收到{"message": "Internal server error"}的任何人

我整天都在碰这个问题。我做了一些更改,但我认为对我来说,所做的就是在初始化ApolloServer时包括上下文:

const server = new ApolloServer({
...serverConfig,// typeDefs and resolvers
context: ({ event,context }) => ({
headers: event.headers,functionName: context.functionName,event,context,}),playground: {
endpoint: '/dev/graphql',},});

查看此链接:https://www.apollographql.com/docs/apollo-server/deployment/lambda/#getting-request-info

调试时,我建议两件事


  1. 在AWS上检查您的lambda函数的Cloudwatch日志

  2. 运行serverless offline,转到本地端点,看看是否有任何错误(您需要安装serverless offline plugin并将其包含在serverless.yml中)

如果您使用的是纱线monorepo和打字稿,则还有一些其他相关信息-

确保编译并翻译打字稿代码。 See this article。我的代码:


  • yarn add webpack serverless-webpack

  • yarn add -D webpack-node-externals

webpack.config.js

const path = require('path');
const slsw = require('serverless-webpack');
const nodeExternals = require('webpack-node-externals');
module.exports = {
entry: slsw.lib.entries,target: 'node',mode: slsw.lib.webpack.isLocal ? 'development' : 'production',optimization: {
minimize: false,performance: {
hints: false,devtool: 'nosources-source-map',externals: [nodeExternals()],module: {
rules: [
{
test: /\.ts$/,loader: 'babel-loader',options: {
presets: [
[
'@babel/preset-env',{
targets: {
node: true,],'@babel/typescript',include: [__dirname],exclude: /node_modules/,resolve: {
extensions: ['.ts','.js'],output: {
libraryTarget: 'commonjs2',path: path.join(__dirname,'.webpack'),filename: '[name].js',};

serverless.yml

# serverless.yml
service: apollo-lambda
plugins:
- serverless-webpack
- serverless-offline
custom:
webpack:
webpackConfig: ./webpack.config.js
includeModules: true
provider:
name: aws
runtime: nodejs12.x
functions:
graphql:
# this is formatted as .
handler: dist/server.graphqlHandler
environment:
SLS_DEBUG: true
events:
- http:
path: graphql
method: post
cors: true
integration: lambda-proxy
- http:
path: graphql
method: get
cors: true
integration: lambda-proxy

tsconfig.json

{
"compilerOptions": {
"sourceMap": true,"outDir": "./dist","strict": true,"lib": ["es5"],"esModuleInterop": true,"types": ["react","jest"]
}
}

我还添加了一个名为deploy的脚本,该脚本删除了dist文件夹,并在编译打字稿后重新创建了dist文件夹(请参阅outDir中的tsconfig.json) ,然后运行serverless deploy

package.json

{
...
"main": "dist/server.js","scripts": {
...
"deploy": "rimraf dist && npx tsc && serverless deploy",...
}

注意:您需要全局安装rimraf才能使脚本正常运行(npm install -g rimraf


推荐阅读
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • WebSocket与Socket.io的理解
    WebSocketprotocol是HTML5一种新的协议。它的最大特点就是,服务器可以主动向客户端推送信息,客户端也可以主动向服务器发送信息,是真正的双向平等对话,属于服务器推送 ... [详细]
  • 如何在php中将mysql查询结果赋值给变量
    本文介绍了在php中将mysql查询结果赋值给变量的方法,包括从mysql表中查询count(学号)并赋值给一个变量,以及如何将sql中查询单条结果赋值给php页面的一个变量。同时还讨论了php调用mysql查询结果到变量的方法,并提供了示例代码。 ... [详细]
  • Python SQLAlchemy库的使用方法详解
    本文详细介绍了Python中使用SQLAlchemy库的方法。首先对SQLAlchemy进行了简介,包括其定义、适用的数据库类型等。然后讨论了SQLAlchemy提供的两种主要使用模式,即SQL表达式语言和ORM。针对不同的需求,给出了选择哪种模式的建议。最后,介绍了连接数据库的方法,包括创建SQLAlchemy引擎和执行SQL语句的接口。 ... [详细]
  • 解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法
    本文介绍了解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法,包括检查location配置是否正确、pass_proxy是否需要加“/”等。同时,还介绍了修改nginx的error.log日志级别为debug,以便查看详细日志信息。 ... [详细]
  • Oracle优化新常态的五大禁止及其性能隐患
    本文介绍了Oracle优化新常态中的五大禁止措施,包括禁止外键、禁止视图、禁止触发器、禁止存储过程和禁止JOB,并分析了这些禁止措施可能带来的性能隐患。文章还讨论了这些禁止措施在C/S架构和B/S架构中的不同应用情况,并提出了解决方案。 ... [详细]
  • 31.项目部署
    目录1一些概念1.1项目部署1.2WSGI1.3uWSGI1.4Nginx2安装环境与迁移项目2.1项目内容2.2项目配置2.2.1DEBUG2.2.2STAT ... [详细]
  • 本文介绍了sqlserver云存储和本地存储的区别,云存储是将数据存储在网络上,方便查看和调用;本地存储是将数据存储在电脑磁盘上,只能在存储的电脑上查看。同时提供了几种启动sqlserver的方法。此外,还介绍了如何导出数据库的步骤和工具。 ... [详细]
  • PDO MySQL
    PDOMySQL如果文章有成千上万篇,该怎样保存?数据保存有多种方式,比如单机文件、单机数据库(SQLite)、网络数据库(MySQL、MariaDB)等等。根据项目来选择,做We ... [详细]
  • 在Oracle11g以前版本中的的DataGuard物理备用数据库,可以以只读的方式打开数据库,但此时MediaRecovery利用日志进行数据同步的过 ... [详细]
  • 本文介绍了一个React Native新手在尝试将数据发布到服务器时遇到的问题,以及他的React Native代码和服务器端代码。他使用fetch方法将数据发送到服务器,但无法在服务器端读取/获取发布的数据。 ... [详细]
  • 程序员请收好:10 个实用的 VS Code 插件
    点击上方“编程派”,选择设为“设为星标”优质文章,第一时间送达!英文:Daan,翻译:CSDN- ... [详细]
  • 全新升级的AOP框架Dora.Interception[汇总,共6篇]
    多年之前利用ILEmit写了一个名为Dora.Interception的AOP框架。前几天利用Roslyn的SourceGenerator对自己为公司写的一个GraphQL框架进行 ... [详细]
  • SpringBoot整合GraphQL第(二)章节
    Postman请求方式Post请求JSON格式传递{query:{selectUserById(id:1){idnameecountlout}}}根据id进行查询组 ... [详细]
author-avatar
Kevinczp
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有