Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/Garllon/graphql_connector

An easy connector to call your graphql server
https://github.com/Garllon/graphql_connector

client connector graphql ruby

Last synced: about 1 month ago
JSON representation

An easy connector to call your graphql server

Awesome Lists containing this project

README

        

# GraphqlConnector

[![Gem
Version](https://badge.fury.io/rb/graphql_connector.svg)](https://badge.fury.io/rb/graphql_connector)
[![CI](https://github.com/Garllon/graphql_connector/workflows/CI/badge.svg)](https://github.com/Garllon/graphql_connector/actions?query=workflow%3ACI)
[![Maintainability](https://api.codeclimate.com/v1/badges/548db3cf0d078b379c84/maintainability)](https://codeclimate.com/github/Garllon/graphql_connector/maintainability)
[![Test Coverage](https://api.codeclimate.com/v1/badges/548db3cf0d078b379c84/test_coverage)](https://codeclimate.com/github/Garllon/graphql_connector/test_coverage)

An easy connector to call your `graphql` server. Currently there is no schema
check in the code, but i will add it.

## Installation

Add this line to your application's Gemfile:

```ruby
gem 'graphql_connector'
```

And then execute:

$ bundle

Or install it yourself as:

$ gem install graphql_connector

## Usage

You need to configure the `graphql_connector` first:
``` ruby
GraphqlConnector.configure do |config|
config.add_server(name: 'Foo', uri: 'http://foo.com/api/graphql', headers: {}, connector: {}, httparty_adapter_options: {})
end
```

* `name` (**mandatory**): Add a namespace under which the API is reachable. In the above example `'Foo'` means that `GraphqlConnector::Foo` provides API functionality for the configured `add_server`

* `uri` (**mandatory**): Add uri of grapqhl API server that accepts **POST** requests.

* `connector` (**optionally**): Add a **Authentication Token class** in the following format:
```ruby
connector: { base: AuthTokenAgent.instance, method: 'get_authorization_header' }
```
`base` is an instance of Authentication Token class and `method` represents the function where it gets the token.
```ruby
class AuthTokenAgent
[...]
def get_authorization_header
[...]
{ 'Authorization' => 'Token HERE' }
end
end
```
:warning: The function under `method` must be a public one in your connector class.

:warning: When you set a connector, it will override the setting in the headers for Authorization.

* `httparty_adapter_options` (**optionally**): Add any [`connection_adapter`](https://github.com/jnunemaker/httparty/blob/master/lib/httparty/connection_adapter.rb) options that `httparty` supports in a hash format e.g. `{ timeout: 4 }`

For each graphql server you wish to query use `add_server`.

Afterwards you will have the following options to fetch and/or mutate data with
one or many graphql servers:

* `raw_query` --> [Examples](examples/raw_query_examples.rb)
* `query` --> [Examples](examples/query_examples.rb)
* `mutation` --> [Examples](examples/mutation_examples.rb)
* `service class inclusion` --> [Examples](examples/departments_service_class_examples.rb)

See the following sub sections for details

### raw_query

You can call your graphql_endpoint via:
```ruby
GraphqlConnector::.raw_query('query { products { id name } }')

GraphqlConnector::.raw_query('query { products($id: [ID!]) { products(id: $id) { id name } } }', variables: { id: 1 })

GraphqlConnector::.raw_query('mutation { createProduct(name: "Coca-Cola") { id name } }')
```

Or if you want to **override**/**set** any `httparty_adapter_options` it is also possible to pass them in:
```ruby
GraphqlConnector::.raw_query('query { products { id name } }', httparty_adapter_options: { timeout: 3, verify: true })

GraphqlConnector::.raw_query('query { products($id: [ID!]) { products(id: $id) { id name } } }', variables: { id: 1 }, httparty_adapter_options: { timeout: 3, verify: true })

GraphqlConnector::.raw_query('mutation { createProduct(name: "Coca-Cola") { id name } }', httparty_adapter_options: { timeout: 3, verify: true })
```

Note that `` has to be replaced by any of the ones added via `add_server`

See also [here](examples/raw_query_examples.rb) for example usage

---
### query

You can also use the more comfortable `query`:
```ruby
GraphqlConnector::.query('products', { id: [1,2] } , ['id','name'])
```

Or if you want to **override**/**set** any `httparty` adapter_options, it is also possible
to pass them in:

```ruby
GraphqlConnector::.query('products', { id: [1,2] } , ['id','name'], httparty_adapter_options: { timeout: 3 })
```

| Variable | Required | DataType | Example |
|--------------------------|-------------------------------|-------------------------|-----------------------------------------|
| model | Yes | String | 'product' |
| condition | Yes (but can be empty) | Hash(key, value) | { id: 1 } |
| selected_fields | Yes | Array of Strings/Hashes | ['id', 'name', productCategory: ['id']] |
| httparty_adapter_options | No | Hash | { timeout: 3 } |

See also [here](examples/query_examples.rb) for example usage

#### selected_fields

The syntax for the associations looks like the following:
```
['', : ['']]
```

Example:
```ruby
['id', 'name', productCategory: ['id', 'name']]
```

---

### mutation

Works in the same way as [query](#query)

See also [here](examples/mutation_examples.rb) for example usage

### Service class inclusion

This approach can be used to `graphqlize` **any** kind of ruby (service) class
so that it has re-usable graphql `query` and `mutation` **class methods**.

* First add `extend GraphqlConnector::::Query` in the the class(es) that should be `graphqlized`

* Then you can aliases as many graphql server types via `add_query` and/or `add_raw_query` and/or `add_mutation`:

```ruby
add_query products_by: :products, params: [:id], returns: [:id, :name]

add_raw_query products_raw_by: 'query { products(($id: [ID!]) products { id name } }', params: [:id]

add_mutation create: :createProduct, params: [:name], returns: [:id, :name]
```
* :grey_exclamation: If not needed omit `params`

Or if you want to **override**/**set** any `httparty` adapter_options, it is also possible
to pass them in:

```ruby
add_query products_by: :products, params: [:id], returns: [:id, :name], httparty_adapter_options: { timeout: 3 }

add_raw_query products_raw_by: 'query { products(($id: [ID!]) products { id name } }', params: [:id], httparty_adapter_options: { verify: true }

add_mutation create: :createProduct, params: [:name], returns: [:id, :name], httparty_adapter_options: { timeout: 5, verify: false }
```

See also [here](examples/departments_service_class_examples.rb) and also here for complete example usage:

```ruby
GraphqlConnector.configure do |config|
config.add_server(name: 'Foo', uri: 'http://foo.com/api/graphql', headers: {})
end

# product.rb
class Product
extend GraphqlConnector::Foo::Query

add_query all: :products,
returns: [:id, :name]

add_query by_id: :products,
params: :id,
returns: [:name, product_category: [:id, :name]],
httparty_adapter_options: { timeout: 3 }

add_query by_names: :products,
params: :names,
returns: [:id, :name, product_category: [:id, :name]]

add_query by: :products,
params: [:id, :name],
returns: [:name]

add_query by_category_id: :products,
params: :product_category,
returns: [product_category: [:id, :name]]

add_mutation create: :createProduct,
params: [:name, :catgetoryId],
returns: [:id, :name],
httparty_adapter_options: { verify: false }
end

Product.all
=> [OpenStruct [OpenStruct>]

Product.by_names(names: ['Demo Product', 'Non Demo Product'])
=> [OpenStruct>, Product OpenStruct

Product.by_category_id(product_category: { id: 10})
=> OpenStruct>

Product.create(name: 'Another Product', catgetoryId: 10)
=> OpenStruct
```

Also custom **class methods** can used to call any kind of `query` and do further selection instead:

```ruby
class Product
extend GraphqlConnector::Foo::Query

add_query all: :products, returns: [:name]

def self.by_id(id:)
all.select { |products| products.id == id }.first
end
end

Product.by_id(id: 1)
=> OpenStruct>
```

:warning: Ensure that your custom **class method** never has the **same name** as an `` of `add_query`, `add_raw_query` or `add_mutation`. Otherwise the associated grapqhl query will not be performed because of [Ruby Open Class principle](https://ruby-lang.co/ruby-open-class/)

Example for `raw_query`:

```ruby
class Product
extend GraphqlConnector::Foo::Query

add_raw_query all: ' query { products { id name } } '
add_raw_query by: ' query products($id: !ID, $name: !String) '\
'{ products(id: $id, name: $name) { id name } }',
params: [:id, :name]

end

Product.all
=> [ { id: '1', name: 'Demo Product' }, ...]

Product.by(id: '1', name: 'Demo Product')
=> { id: '1', name: 'Demo Product' }

```

:exclamation: There is no `add_raw_mutation` since `add_raw_query` does already cover such a case

## Development

After checking out the repo, run
```shell
bundle install
```

Then, run
```shell
bundle exec rspec spec
```
to run the tests.
You can also run `bin/console` for an interactive prompt that will allow you to experiment.

## Contributing

Bug reports and pull requests are welcome on GitHub at [https://github.com/garllon/graphql_connector](https://github.com/garllon/graphql_connector). This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.

## License

The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).

## Code of Conduct

Everyone interacting in the GraphqlConnector project’s codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](https://github.com/Garllon/graphql_connector/blob/master/CODE_OF_CONDUCT.md).