-
-
Notifications
You must be signed in to change notification settings - Fork 88
Home
This project was jQuery-GraphQL formerly. Now it's fully independent. Originally inspired by Robert Mosolgo's blog post
- Nothing depended, plain vanilla JavaScript.
- Plug & Play.
- Isomorphic.
- Runs on most of the browsers.
- You don't need to install Node.js ecosystem into your computer.
GraphQL based on a very simple HTTP transaction which sends a request to an endpoint
with query
and variables
.
Many libraries requires complex stacks to make that simple request. In any project you don't use React, Relay you'll need a simpler client which manages your query and makes a simple request.
// Connect...
var graph = graphql("/graphql")
// Prepare...
var allUsers = graph(`query { allUsers {id, name} }`)
// Run...
allUsers().then(function (users) {
console.log(users)
})
You can download graphql.js
directly, or you can use Bower or NPM.
bower install graphql.js --save
npm install graphql.js --save
You can use GraphQL.js with Rails Asset Pipeline using graphqljs-rails.
GraphQL.js is isomorphic. You can use it in both browser and Node.js.
<script src="/path/to/graphql.js"></script>
var graphql = require('graphql.js')
Create a simple connection to your GraphQL.
var graph = graphql("http://localhost:3000/graphql", {
method: "POST", // POST by default.
headers: {
// headers
"Access-Token": "some-access-token"
},
fragments: {
// fragments, you don't need to say `fragment name`.
auth: "on User { token }",
error: "on Error { messages }"
}
})
graph
will be a simple function that accepts query
and variables
as parameters.
graph(`query ($email: String!, $password: String!) {
auth(email: $email, password: $password) {
... auth # if you use any fragment, it will be added to the query.
... error
}
}`, {
email: "[email protected]",
password: "my-super-password"
}).then(function (response) {
// response is originally response.data of query result
console.log(response)
}).catch(function (error) {
// response is originally response.errors of query result
console.log(error)
})
You can prepare queries for lazy execution. It will allow you to reuse your queries with different variables without any hassle.
var login = graph(`query ($email: String!, $password: String!) {
auth(email: $email, password: $password) {
... on User {
token
}
}
}`)
// Call it later...
login({
email: "[email protected]",
password: "my-super-password"
})
If your query doesn't need any variable, it will generate a lazy execution query by default. If you want to run your query immediately, you have three following options:
// 1st option. create and run function.
graph(`...`)()
graph.query(`...`)()
graph.mutate(`...`)()
//...
// 2nd option. create and run function with `run` method.
graph.run(`...`)
graph.query.run(`...`)
graph.mutate.run(`...`)
// 3rd option. create and run function with template tag.
graph`...`
graph.query`...`
graph.mutate`...`
I don't recommend using this. Using it too much may break your DRY. Use lazy execution as much as possible.
You can prefix your queries by simply calling helper methods: .query
, .mutate
or .subscribe
var login = graph.query(`($email: String!, $password: String!) {
auth(email: $email, password: $password) {
... on User {
token
}
}
}`)
var increment = graph.mutate`increment { state }`
var onIncrement = graph.subscribe`onIncrement { state }`
Declaring simple-typed (String
, Int
, Boolean
) variables in query were a
little bothering to me. That's why I added an @autodeclare
keyword or {declare: true}
setting to the processor.
It detects types from the variables and declares them in query automatically.
var login = graph.query(`(@autodeclare) {
auth(email: $email, password: $password) {
... on User {
token
}
}
}`)
login({
email: "[email protected]", // It's String! obviously.
password: "my-super-password" // It is, too.
})
This will create following query:
query ($email: String!, $password: String!) {
auth(email: $email, password: $password) {
... on User {
token
}
}
}
You can also pass {declare: true}
option to the .query
, .mutate
and .subscribe
helper:
var login = graph.query(`auth(email: $email, password: $password) {
... on User {
token
}
}`, {declare: true})
This will also create the same query above.
Beside you can pass {declare: true}
to helpers:
graph.query("auth(email: $email, password: $password) { token }", {declare: true})
Also you can enable auto declaration to run by default using alwaysAutodeclare
setting.
var graph = graphql("http://localhost:3000/graphql", {
alwaysAutodeclare: true
})
After you enable alwaysAutodeclare
option, your methods will try to detect types of variables and declare them.
// When alwaysAutodeclare is true, you don't have to pass {declare: true} option.
graph.query("auth(email: $email, password: $password) { token }")
You can define custom types when defining variables by using a simple "variable!Type"
notation.
It will help you to make more complex variables:
var register = graph.mutate(`(@autodeclare) {
userRegister(input: $input) { ... }
}`)
register({
// variable name and type.
"input!UserRegisterInput": { ... }
})
This will generate following query:
mutation ($input: UserRegisterInput!) {
userRegister(input: $input) { ... }
}
Fragments make your GraphQL more DRY and improves reusability. With .fragment
method, you'll
manage your fragments easily.
While constructing your endpoint, you can predefine all of your fragments.
var graph = graphql("/graphql", {
fragments: {
userInfo: `on User { id, name, surname, avatar }`
}
})
And you can use your fragments in your queries. The query will pick your fragments and will add them to the bottom of your query.
graph.query(`{ allUsers { ...userInfo } }`)
You can nest your fragments to keep them organized/namespaced.
var graph = graphql("/graphql", {
fragments: {
user: {
info: `on User { id, name, surname, avatar }`
}
}
})
Accessing them is also intuitive:
graph.query(`{ allUsers { ...user.info } }`)
You can reuse fragments in your fragments.
graph.fragment({
user: "on User {name, surname}",
login: {
auth: "on User {token, ...user}"
}
})
You can also add fragments lazily. So you can use your fragments more modular.
// Adds a profile fragment
graph.fragment({
profile: `on User {
id
name(full: true)
avatar
}`
})
var allUsers = graph.query(`{
allUsers {
... profile
}
}`)
allUsers().then(...)
Also you can add nested fragments lazily, too:
graph.fragment({
login: {
error: `on LoginError {
reason
}`
}
})
graph.fragment({
something: {
error: `on SomeError {
messages
}`
}
})
graph.query(`{ login {... login.error } }`)
graph.query(`{ something {... something.error } }`)
You can call fragment string by using .fragment
method. You have to pass path string to get the fragment.
graph.fragment('login.error')
This will give you the matching fragment code:
fragment login_error on LoginError {
reason
}
You can use fragments lazily using ES6 template tag queries.
var userProfileToShow = graph.fragment('user.profile')
graph`query { ... ${userProfileToShow} }`
You can create queries using .ql
ES6 template tag.
// Add some fragments...
graph.fragment({
username: {
user: `on User {
username
}`,
admin: `on AdminUser {
username,
administrationLevel
}`
}
})
// Get any fragment with its path...
var admin = graph.fragment('username.admin')
// Build your query with using fragment paths or dynamic template variables.
var query = graph.ql`query {
...username.user
...${admin}
}`
// Use query anywhere...
$.post("/graphql", {query: query}, function (response) { ... })
graph.ql
will generate this query string:
query {
... username_user
... username_admin
}
fragment username_user on User {
username
}
fragment username_admin on AdminUser {
username,
administrationLevel
}
A CRUD ToDo app example code to show how to use GraphQL.js. An implementation can be found at f/graphql.js-demo
var graph = graphql("/graphql", {
alwaysAutodeclare: true,
fragments: {
todo: `on Todo {
id
text
isCompleted
}`
}
})
function getTodos() {
return graph.query.run(`allTodos {
...todo
}`)
}
function addTodo(text) {
return graph.mutate(`todoAdd(text: $text) {
...todo
}`)({
text: text
})
}
function setTodo(id, isCompleted) {
return graph.mutate(`todoComplete(
id: $id,
status: $isCompleted
) {
...todo
}`)({
"id!ID": id,
isCompleted: isCompleted
})
}
function removeTodo(id) {
return graph.mutate(`todoRemove(
id: $id
) {
...todo
}`)({
"id!ID": id
})
}
MIT License
Copyright (c) 2017 Fatih Kadir Akın
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.