Skip to content
This repository was archived by the owner on Sep 3, 2021. It is now read-only.

Commit db5fda4

Browse files
committed
initial commit
0 parents  commit db5fda4

File tree

10 files changed

+550
-0
lines changed

10 files changed

+550
-0
lines changed

.babelrc

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
presets: ['latest'],
3+
plugins: [
4+
'transform-runtime',
5+
'transform-async-generator-functions',
6+
'transform-object-rest-spread',
7+
],
8+
}

.gitignore

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
# Logs
2+
logs
3+
*.log
4+
npm-debug.log*
5+
yarn-debug.log*
6+
yarn-error.log*
7+
8+
# Runtime data
9+
pids
10+
*.pid
11+
*.seed
12+
*.pid.lock
13+
14+
# Directory for instrumented libs generated by jscoverage/JSCover
15+
lib-cov
16+
17+
# Coverage directory used by tools like istanbul
18+
coverage
19+
20+
# nyc test coverage
21+
.nyc_output
22+
23+
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
24+
.grunt
25+
26+
# Bower dependency directory (https://bower.io/)
27+
bower_components
28+
29+
# node-waf configuration
30+
.lock-wscript
31+
32+
# Compiled binary addons (https://nodejs.org/api/addons.html)
33+
build/Release
34+
35+
# Dependency directories
36+
node_modules/
37+
jspm_packages/
38+
39+
# Typescript v1 declaration files
40+
typings/
41+
42+
# Optional npm cache directory
43+
.npm
44+
45+
# Optional eslint cache
46+
.eslintcache
47+
48+
# Optional REPL history
49+
.node_repl_history
50+
51+
# Output of 'npm pack'
52+
*.tgz
53+
54+
# Yarn Integrity file
55+
.yarn-integrity
56+
57+
# dotenv environment variables file
58+
.env
59+
60+
.idea
61+
62+
.DS_Store
63+
64+
dist/

.npmignore

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
src
2+
test
3+
.DS_Store
4+
.idea

LICENSE

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
Licensed under the Apache License, Version 2.0 (the "License");
2+
you may not use this file except in compliance with the License.
3+
You may obtain a copy of the License at
4+
5+
http://www.apache.org/licenses/LICENSE-2.0
6+
7+
Unless required by applicable law or agreed to in writing, software
8+
distributed under the License is distributed on an "AS IS" BASIS,
9+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
10+
See the License for the specific language governing permissions and
11+
limitations under the License.

README.md

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
# neo4j-graphql-js
2+
3+
A GraphQL to Cypher query execution layer for Neo4j and JavaScript GraphQL implementations.
4+
5+
## What is `neo4j-graphql-js`
6+
7+
A package to make it easier to use GraphQL and Neo4j together. `neo4j-graphql-js` translates GraphQL queries to a single Cypher query, eliminating the need to write queries in GraphQL resolvers. It also exposes the Cypher query language through GraphQL via the `@cypher` schema directive.
8+
9+
### Goals
10+
11+
* Translate GraphQL queries to Cypher to simply the process of writing GraphQL resolvers
12+
* Allow for custom logic by overriding of any resolver function
13+
* Work with `graphl-tools`, `graphql-js`, and `apollo-server`
14+
* Support GraphQL servers that need to resolve data from multiple data services/databases
15+
* Expose the power of Cypher through GraphQL via the `@cypher` directive
16+
17+
## `@cypher` directive
18+
19+
GraphQL is fairly limited when it comes to expressing complex queries such as filtering, or aggregations. We expose the graph querying language Cypher through GraphQL via the `@cypher` directive. Annotate a field in your schema with the `@cypher` directive to map the results of that query to the annotated GraphQL field. For example:
20+
21+
~~~
22+
type Movie {
23+
movieId: ID!
24+
title: String
25+
year: Int
26+
plot: String
27+
28+
similar(first: Int = 3, offset: Int = 0): [Movie] @cypher(statement: "WITH {this} AS this MATCH (this)-->(:Genre)<--(o:Movie) RETURN o ORDER BY COUNT(*) DESC")
29+
}
30+
~~~
31+
32+
The field `similar` will be resolved using the Cypher query
33+
34+
~~~
35+
MATCH (this)-->(:Genre)<--(o:Movie) RETURN o ORDER BY COUNT(*) DESC
36+
~~~
37+
38+
to find movies with overlapping Genres. See the [examples](example/graphql-tools/movies.js) for more information.
39+
40+
## Test
41+
42+
We use the `ava` test runner.
43+
44+
~~~
45+
npm install
46+
npm build
47+
npm test
48+
~~~
49+
50+
Currently we only have simple unit tests verifying generated Cypher as translated from GraphQL queries and schema. More complete tests and CI integration coming soon.
51+
52+
## Examples
53+
54+
See [example/graphql-tools/movies.js](example/graphql-tools/movies.js)
55+
56+
## Features
57+
58+
- [x] translate basic GraphQL queries to Cypher by
59+
- [x] `first` and `offset` arguments for
60+
- [x] `@cypher` directive for mapping
61+
- [ ] Handle enumeration types
62+
- [ ] Handle interface types
63+
- [ ] Handle fragments
64+
- [ ] Handle named fragments
65+
- [ ] Ordering
66+
- [ ] Example with `graphql-js`
67+
- [ ] Example with `apollo-server`
68+
- [ ] Example with Apollo Launchpad

example/graphql-tools/movies.js

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
import { makeExecutableSchema } from 'graphql-tools';
2+
import {neo4jgraphql} from '../../src/index';
3+
import express from 'express';
4+
import { graphqlExpress, graphiqlExpress } from 'graphql-server-express';
5+
import bodyParser from 'body-parser';
6+
import {v1 as neo4j} from 'neo4j-driver';
7+
8+
9+
10+
// Simple Movie schema
11+
const typeDefs = `
12+
type Movie {
13+
movieId: ID!
14+
title: String
15+
year: Int
16+
plot: String
17+
poster: String
18+
imdbRating: Float
19+
genres: [String]
20+
similar(first: Int = 3, offset: Int = 0): [Movie] @cypher(statement: "WITH {this} AS this MATCH (this)--(:Genre)--(o:Movie) RETURN o")
21+
mostSimilar: Movie @cypher(statement: "WITH {this} AS this RETURN this")
22+
degree: Int @cypher(statement: "WITH {this} AS this RETURN SIZE((this)--())")
23+
actors(first: Int = 3, offset: Int = 0): [Actor] @relation(name: "ACTED_IN", direction:"IN")
24+
avgStars: Float
25+
}
26+
27+
interface Person {
28+
id: ID!
29+
name: String
30+
}
31+
32+
type Actor implements Person {
33+
id: ID!
34+
name: String
35+
movies: [Movie]
36+
}
37+
38+
type User implements Person {
39+
id: ID!
40+
name: String
41+
}
42+
43+
44+
type Query {
45+
Movie(id: ID, title: String, year: Int, plot: String, poster: String, imdbRating: Float, first: Int, offset: Int): [Movie]
46+
}
47+
`;
48+
49+
const resolvers = {
50+
// root entry point to GraphQL service
51+
Query: {
52+
// fetch movies by title substring
53+
Movie(object, params, ctx, resolveInfo) {
54+
return neo4jgraphql(object, params, ctx, resolveInfo);
55+
}
56+
}
57+
};
58+
59+
60+
const schema = makeExecutableSchema({
61+
typeDefs,
62+
resolvers,
63+
});
64+
65+
function context(headers, secrets) {
66+
let driver = neo4j.driver(secrets.NEO4J_URI || "bolt://localhost:7687", neo4j.auth.basic(secrets.NEO4J_USER || "neo4j", secrets.NEO4J_PASSWORD || "letmein"))
67+
return {driver};
68+
}
69+
70+
const rootValue = {};
71+
72+
73+
const PORT = 3000;
74+
const server = express();
75+
76+
server.use('/graphql', bodyParser.json(), graphqlExpress(request => ({
77+
schema,
78+
rootValue,
79+
context: context(request.headers, process.env),
80+
})));
81+
82+
server.use('/graphiql', graphiqlExpress({
83+
endpointURL: '/graphql',
84+
query: `{
85+
86+
}`,
87+
}));
88+
89+
server.listen(PORT, () => {
90+
console.log(`GraphQL Server is now running on http://localhost:${PORT}/graphql`);
91+
console.log(`View GraphiQL at http://localhost:${PORT}/graphiql`);
92+
});

package.json

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
{
2+
"name": "neo4j-graphql-js",
3+
"version": "0.1.0",
4+
"description": "A GraphQL to Cypher query execution layer for Neo4j. ",
5+
"main": "./dist/index.js",
6+
"scripts": {
7+
"start": "nodemon ./example/graphql-tools/movies.js --exec babel-node -e js",
8+
"build": "babel src --presets babel-preset-es2015 --out-dir dist",
9+
"prepublish": "npm run build",
10+
"test": "ava test/*.js"
11+
},
12+
"author": "William Lyon",
13+
"license": "Apache 2.0",
14+
"devDependencies": {
15+
"ava": "^0.22.0",
16+
"babel-cli": "^6.24.0",
17+
"babel-plugin-transform-async-generator-functions": "^6.24.1",
18+
"babel-plugin-transform-object-rest-spread": "^6.23.0",
19+
"babel-plugin-transform-runtime": "^6.23.0",
20+
"babel-preset-latest": "^6.23.0",
21+
"nodemon": "^1.11.0"
22+
},
23+
"dependencies": {
24+
"body-parser": "^1.17.1",
25+
"express": "^4.15.2",
26+
"graphql": "^0.11.3",
27+
"graphql-server-core": "^1.1.0",
28+
"graphql-server-express": "^0.6.0",
29+
"graphql-tools": "^1.2.2",
30+
"lodash": "^4.17.4",
31+
"neo4j-driver": "^1.4.0"
32+
}
33+
}

0 commit comments

Comments
 (0)