set up Apollo server!
This commit is contained in:
parent
513b2c12e7
commit
ecebb93ec5
8 changed files with 5547 additions and 0 deletions
119
server/.gitignore
vendored
Normal file
119
server/.gitignore
vendored
Normal file
|
@ -0,0 +1,119 @@
|
||||||
|
.env
|
||||||
|
|
||||||
|
# From https://github.com/github/gitignore/blob/2a4de265d37eca626309d8e115218d18985b5435/Node.gitignore
|
||||||
|
|
||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
lerna-debug.log*
|
||||||
|
|
||||||
|
# Diagnostic reports (https://nodejs.org/api/report.html)
|
||||||
|
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
|
||||||
|
|
||||||
|
# Runtime data
|
||||||
|
pids
|
||||||
|
*.pid
|
||||||
|
*.seed
|
||||||
|
*.pid.lock
|
||||||
|
|
||||||
|
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||||
|
lib-cov
|
||||||
|
|
||||||
|
# Coverage directory used by tools like istanbul
|
||||||
|
coverage
|
||||||
|
*.lcov
|
||||||
|
|
||||||
|
# nyc test coverage
|
||||||
|
.nyc_output
|
||||||
|
|
||||||
|
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
||||||
|
.grunt
|
||||||
|
|
||||||
|
# Bower dependency directory (https://bower.io/)
|
||||||
|
bower_components
|
||||||
|
|
||||||
|
# node-waf configuration
|
||||||
|
.lock-wscript
|
||||||
|
|
||||||
|
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
||||||
|
build/Release
|
||||||
|
|
||||||
|
# Dependency directories
|
||||||
|
node_modules/
|
||||||
|
jspm_packages/
|
||||||
|
|
||||||
|
# Snowpack dependency directory (https://snowpack.dev/)
|
||||||
|
web_modules/
|
||||||
|
|
||||||
|
# TypeScript cache
|
||||||
|
*.tsbuildinfo
|
||||||
|
|
||||||
|
# Optional npm cache directory
|
||||||
|
.npm
|
||||||
|
|
||||||
|
# Optional eslint cache
|
||||||
|
.eslintcache
|
||||||
|
|
||||||
|
# Microbundle cache
|
||||||
|
.rpt2_cache/
|
||||||
|
.rts2_cache_cjs/
|
||||||
|
.rts2_cache_es/
|
||||||
|
.rts2_cache_umd/
|
||||||
|
|
||||||
|
# Optional REPL history
|
||||||
|
.node_repl_history
|
||||||
|
|
||||||
|
# Output of 'npm pack'
|
||||||
|
*.tgz
|
||||||
|
|
||||||
|
# Yarn Integrity file
|
||||||
|
.yarn-integrity
|
||||||
|
|
||||||
|
# dotenv environment variables file
|
||||||
|
.env
|
||||||
|
.env.test
|
||||||
|
|
||||||
|
# parcel-bundler cache (https://parceljs.org/)
|
||||||
|
.cache
|
||||||
|
.parcel-cache
|
||||||
|
|
||||||
|
# Next.js build output
|
||||||
|
.next
|
||||||
|
|
||||||
|
# Nuxt.js build / generate output
|
||||||
|
.nuxt
|
||||||
|
dist
|
||||||
|
|
||||||
|
# Gatsby files
|
||||||
|
.cache/
|
||||||
|
# Comment in the public line in if your project uses Gatsby and not Next.js
|
||||||
|
# https://nextjs.org/blog/next-9-1#public-directory-support
|
||||||
|
# public
|
||||||
|
|
||||||
|
# vuepress build output
|
||||||
|
.vuepress/dist
|
||||||
|
|
||||||
|
# Serverless directories
|
||||||
|
.serverless/
|
||||||
|
|
||||||
|
# FuseBox cache
|
||||||
|
.fusebox/
|
||||||
|
|
||||||
|
# DynamoDB Local files
|
||||||
|
.dynamodb/
|
||||||
|
|
||||||
|
# TernJS port file
|
||||||
|
.tern-port
|
||||||
|
|
||||||
|
# Stores VSCode versions used for testing VSCode extensions
|
||||||
|
.vscode-test
|
||||||
|
|
||||||
|
# yarn v2
|
||||||
|
|
||||||
|
.yarn/cache
|
||||||
|
.yarn/unplugged
|
||||||
|
.yarn/build-state.yml
|
||||||
|
.pnp.*
|
15
server/db.js
Normal file
15
server/db.js
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
require("dotenv").config();
|
||||||
|
const mysql = require("mysql2/promise");
|
||||||
|
|
||||||
|
async function connectToDb() {
|
||||||
|
const db = await mysql.createConnection({
|
||||||
|
host: "impress.openneo.net",
|
||||||
|
user: process.env["IMPRESS_MYSQL_USER"],
|
||||||
|
password: process.env["IMPRESS_MYSQL_PASSWORD"],
|
||||||
|
database: "openneo_impress",
|
||||||
|
});
|
||||||
|
|
||||||
|
return db;
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = connectToDb;
|
44
server/index.js
Normal file
44
server/index.js
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
const { ApolloServer, gql } = require("apollo-server");
|
||||||
|
|
||||||
|
const connectToDb = require("./db");
|
||||||
|
const { loadItems, loadItemTranslation } = require("./loaders");
|
||||||
|
|
||||||
|
const typeDefs = gql`
|
||||||
|
type Item {
|
||||||
|
id: ID!
|
||||||
|
name: String!
|
||||||
|
}
|
||||||
|
|
||||||
|
type Query {
|
||||||
|
items(ids: [ID!]!): [Item!]!
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
const resolvers = {
|
||||||
|
Item: {
|
||||||
|
name: async (item, _, { db }) => {
|
||||||
|
const translation = await loadItemTranslation(db, item.id, "en");
|
||||||
|
return translation.name;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Query: {
|
||||||
|
items: (_, { ids }, { db }) => loadItems(db, ids),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const server = new ApolloServer({
|
||||||
|
typeDefs,
|
||||||
|
resolvers,
|
||||||
|
context: async () => {
|
||||||
|
const db = await connectToDb();
|
||||||
|
return { db };
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (require.main === module) {
|
||||||
|
server.listen().then(({ url }) => {
|
||||||
|
console.log(`🚀 Server ready at ${url}`);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { server };
|
97
server/index.test.js
Normal file
97
server/index.test.js
Normal file
|
@ -0,0 +1,97 @@
|
||||||
|
const gql = require("graphql-tag");
|
||||||
|
const { createTestClient } = require("apollo-server-testing");
|
||||||
|
|
||||||
|
const connectToDb = require("./db");
|
||||||
|
const actualConnectToDb = jest.requireActual("./db");
|
||||||
|
const { server } = require("./index");
|
||||||
|
|
||||||
|
const { query } = createTestClient(server);
|
||||||
|
|
||||||
|
// Spy on db.execute, so we can snapshot the queries we run. This can help us
|
||||||
|
// keep an eye on perf - watch for tests with way too many queries!
|
||||||
|
jest.mock("./db");
|
||||||
|
let queryFn;
|
||||||
|
beforeEach(() => {
|
||||||
|
numQueries = 0;
|
||||||
|
connectToDb.mockImplementation(async (...args) => {
|
||||||
|
const db = await actualConnectToDb(...args);
|
||||||
|
queryFn = jest.spyOn(db, "execute");
|
||||||
|
return db;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
afterEach(() => {
|
||||||
|
jest.resetAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("can load items", async () => {
|
||||||
|
const res = await query({
|
||||||
|
query: gql`
|
||||||
|
query($ids: [ID!]!) {
|
||||||
|
items(ids: $ids) {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
variables: {
|
||||||
|
ids: [
|
||||||
|
38913, // Zafara Agent Gloves
|
||||||
|
38911, // Zafara Agent Hood
|
||||||
|
38912, // Zafara Agent Robe
|
||||||
|
],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(res.errors).toBeFalsy();
|
||||||
|
expect(res.data).toMatchInlineSnapshot(`
|
||||||
|
Object {
|
||||||
|
"items": Array [
|
||||||
|
Object {
|
||||||
|
"id": "38911",
|
||||||
|
"name": "Zafara Agent Hood",
|
||||||
|
},
|
||||||
|
Object {
|
||||||
|
"id": "38912",
|
||||||
|
"name": "Zafara Agent Robe",
|
||||||
|
},
|
||||||
|
Object {
|
||||||
|
"id": "38913",
|
||||||
|
"name": "Zafara Agent Gloves",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
`);
|
||||||
|
expect(queryFn.mock.calls).toMatchInlineSnapshot(`
|
||||||
|
Array [
|
||||||
|
Array [
|
||||||
|
"SELECT * FROM items WHERE id IN (?,?,?)",
|
||||||
|
Array [
|
||||||
|
"38913",
|
||||||
|
"38911",
|
||||||
|
"38912",
|
||||||
|
],
|
||||||
|
],
|
||||||
|
Array [
|
||||||
|
"SELECT * FROM item_translations WHERE item_id = ? AND locale = ? LIMIT 1",
|
||||||
|
Array [
|
||||||
|
38911,
|
||||||
|
"en",
|
||||||
|
],
|
||||||
|
],
|
||||||
|
Array [
|
||||||
|
"SELECT * FROM item_translations WHERE item_id = ? AND locale = ? LIMIT 1",
|
||||||
|
Array [
|
||||||
|
38912,
|
||||||
|
"en",
|
||||||
|
],
|
||||||
|
],
|
||||||
|
Array [
|
||||||
|
"SELECT * FROM item_translations WHERE item_id = ? AND locale = ? LIMIT 1",
|
||||||
|
Array [
|
||||||
|
38913,
|
||||||
|
"en",
|
||||||
|
],
|
||||||
|
],
|
||||||
|
]
|
||||||
|
`);
|
||||||
|
});
|
26
server/loaders.js
Normal file
26
server/loaders.js
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
async function loadItems(db, ids) {
|
||||||
|
const qs = ids.map((_) => "?").join(",");
|
||||||
|
const [rows, _] = await db.execute(
|
||||||
|
`SELECT * FROM items WHERE id IN (${qs})`,
|
||||||
|
ids
|
||||||
|
);
|
||||||
|
|
||||||
|
return rows;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadItemTranslation(db, itemId, locale) {
|
||||||
|
const [
|
||||||
|
rows,
|
||||||
|
_,
|
||||||
|
] = await db.execute(
|
||||||
|
`SELECT * FROM item_translations WHERE item_id = ? AND locale = ? LIMIT 1`,
|
||||||
|
[itemId, locale]
|
||||||
|
);
|
||||||
|
if (rows.length === 0) {
|
||||||
|
throw new Error(`could not load translation for ${itemId}, ${locale}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rows[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = { loadItems, loadItemTranslation };
|
25
server/package.json
Normal file
25
server/package.json
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
{
|
||||||
|
"name": "impress-2020-server",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "index.js",
|
||||||
|
"author": "Matchu",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"apollo-server": "^2.12.0",
|
||||||
|
"dotenv": "^8.2.0",
|
||||||
|
"graphql": "^15.0.0",
|
||||||
|
"mysql2": "^2.1.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"apollo-server-testing": "^2.12.0",
|
||||||
|
"jest": "^25.4.0",
|
||||||
|
"nodemon": "^2.0.3",
|
||||||
|
"prettier": "^2.0.5"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"start": "node index.js",
|
||||||
|
"watch": "nodemon index.js",
|
||||||
|
"test": "jest",
|
||||||
|
"setup-mysql-user": "mysql -h impress.openneo.net -u matchu -p < setup-mysql-user.sql"
|
||||||
|
}
|
||||||
|
}
|
2
server/setup-mysql-user.sql
Normal file
2
server/setup-mysql-user.sql
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
GRANT SELECT ON openneo_impress.items TO impress2020;
|
||||||
|
GRANT SELECT ON openneo_impress.item_translations TO impress2020;
|
5219
server/yarn.lock
Normal file
5219
server/yarn.lock
Normal file
File diff suppressed because it is too large
Load diff
Loading…
Reference in a new issue