Skip to content

Commit f736439

Browse files
committed
Initial commit.
0 parents  commit f736439

13 files changed

+938
-0
lines changed

.gitignore

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
*.gem
2+
*.rbc
3+
/.config
4+
/coverage/
5+
/InstalledFiles
6+
/pkg/
7+
/spec/reports/
8+
/spec/examples.txt
9+
/test/tmp/
10+
/test/version_tmp/
11+
/tmp/
12+
13+
## Specific to RubyMotion:
14+
.dat*
15+
.repl_history
16+
build/
17+
18+
## Documentation cache and generated files:
19+
/.yardoc/
20+
/_yardoc/
21+
/doc/
22+
/rdoc/
23+
24+
## Environment normalization:
25+
/.bundle/
26+
/vendor/bundle
27+
/lib/bundler/man/
28+
29+
# for a library or gem, you might want to ignore these files since the code is
30+
# intended to run in multiple environments; otherwise, check them in:
31+
Gemfile.lock
32+
.ruby-version
33+
.ruby-gemset
34+
35+
# unless supporting rvm < 1.11.0 or doing something fancy, ignore this:
36+
.rvmrc

.travis.yml

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
language: ruby
2+
sudo: false
3+
before_install:
4+
- bundle update
5+
rvm:
6+
- 2.1
7+
- 2.2
8+
- 2.3.0
9+
- ruby-head
10+
matrix:
11+
allow_failures:
12+
- rvm: ruby-head

Gemfile

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
source 'https://rubygems.org'
2+
3+
gemspec

README.md

+96
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
# jsonapi-renderer
2+
Ruby gem for rendering [JSON API](http://jsonapi.org) documents.
3+
4+
## Status
5+
6+
[![Gem Version](https://badge.fury.io/rb/jsonapi-renderer.svg)](https://badge.fury.io/rb/jsonapi-renderer)
7+
[![Build Status](https://secure.travis-ci.org/jsonapi-rb/renderer.svg?branch=master)](http://travis-ci.org/jsonapi-rb/renderer?branch=master)
8+
9+
## Installation
10+
```ruby
11+
# In Gemfile
12+
gem 'jsonapi-renderer'
13+
```
14+
then
15+
```
16+
$ bundle
17+
```
18+
or manually via
19+
```
20+
$ gem install jsonapi-renderer
21+
```
22+
23+
## Usage
24+
25+
First, require the gem:
26+
```ruby
27+
require 'jsonapi/renderer'
28+
```
29+
30+
### Rendering resources
31+
32+
A resource here is any class that implements the following interface:
33+
```ruby
34+
class ResourceInterface
35+
# Returns the type of the resource.
36+
# @return [String]
37+
def jsonapi_type; end
38+
39+
# Returns the id of the resource.
40+
# @return [String]
41+
def jsonapi_id; end
42+
43+
# Returns a hash containing, for each included relationship, the resource(s)
44+
# to be included from that one.
45+
# @param [Array<Symbol>] included_relationships The keys of the relationships
46+
# to be included.
47+
# @return [Hash{Symbol => #ResourceInterface, Array<#ResourceInterface>}]
48+
def jsonapi_related(included_relationships); end
49+
50+
# Returns a JSON API-compliant representation of the resource as a hash.
51+
# @param [Hash] options
52+
# @option [Array<Symbol>, Nil] fields The requested fields, or nil.
53+
# @option [Array<Symbol>, Nil] included The requested included
54+
# relationships, or nil.
55+
# @return [Hash]
56+
def as_jsonapi(options = {}); end
57+
```
58+
59+
#### Rendering a single resource
60+
```ruby
61+
JSONAPI.render(resource,
62+
include: include_string,
63+
fields: fields_hash,
64+
meta: meta_hash,
65+
links: links_hash)
66+
```
67+
68+
This returns a JSON API compliant hash representing the described document.
69+
70+
#### Rendering a collection of resources
71+
```ruby
72+
JSONAPI.render(resources,
73+
include: include_string,
74+
fields: fields_hash,
75+
meta: meta_hash,
76+
links: links_hash)
77+
```
78+
79+
This returns a JSON API compliant hash representing the described document.
80+
81+
### Rendering errors
82+
83+
```ruby
84+
JSONAPI.render_errors(errors,
85+
meta: meta_hash,
86+
links: links_hash)
87+
```
88+
89+
where `errors` is an array of objects implementing the `as_jsonapi` method, that
90+
returns a JSON API-compliant representation of the error.
91+
92+
This returns a JSON API compliant hash representing the described document.
93+
94+
## License
95+
96+
jsonapi-renderer is released under the [MIT License](http://www.opensource.org/licenses/MIT).

Rakefile

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
require 'bundler/gem_tasks'
2+
require 'rspec/core/rake_task'
3+
4+
RSpec::Core::RakeTask.new(:spec) do |t|
5+
t.pattern = Dir.glob('spec/**/*_spec.rb')
6+
end
7+
8+
task default: :test
9+
task test: :spec

VERSION

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
0.1.1.beta2

jsonapi-renderer.gemspec

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
version = File.read(File.expand_path('../VERSION', __FILE__)).strip
2+
3+
Gem::Specification.new do |spec|
4+
spec.name = 'jsonapi-renderer'
5+
spec.version = version
6+
spec.author = 'Lucas Hosseini'
7+
spec.email = '[email protected]'
8+
spec.summary = 'Render JSONAPI documents.'
9+
spec.description = 'Efficiently render JSON API documents.'
10+
spec.homepage = 'https://github.com/jsonapi-rb/renderer'
11+
spec.license = 'MIT'
12+
13+
spec.files = Dir['README.md', 'lib/**/*']
14+
spec.require_path = 'lib'
15+
end

lib/jsonapi/include_directive.rb

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
require 'jsonapi/include_directive/parser'
2+
3+
module JSONAPI
4+
# Represent a recursive set of include directives
5+
# (c.f. http://jsonapi.org/format/#fetching-includes)
6+
#
7+
# Addition to the spec: two wildcards, namely '*' and '**'.
8+
# The former stands for any one level of relationship, and the latter stands
9+
# for any number of levels of relationships.
10+
# @example 'posts.*' # => Include related posts, and all the included posts'
11+
# related resources.
12+
# @example 'posts.**' # => Include related posts, and all the included
13+
# posts' related resources, and their related resources, recursively.
14+
class IncludeDirective
15+
# @param include_args (see Parser.parse_include_args)
16+
def initialize(include_args, options = {})
17+
include_hash = Parser.parse_include_args(include_args)
18+
@hash = include_hash.each_with_object({}) do |(key, value), hash|
19+
hash[key] = self.class.new(value, options)
20+
end
21+
@options = options
22+
end
23+
24+
# @param key [Symbol, String]
25+
def key?(key)
26+
@hash.key?(key.to_sym) ||
27+
(@options[:allow_wildcard] && (@hash.key?(:*) || @hash.key?(:**)))
28+
end
29+
30+
# @return [Array<Symbol>]
31+
def keys
32+
@hash.keys
33+
end
34+
35+
# @param key [Symbol, String]
36+
# @return [IncludeDirective, nil]
37+
def [](key)
38+
case
39+
when @hash.key?(key.to_sym)
40+
@hash[key.to_sym]
41+
when @options[:allow_wildcard] && @hash.key?(:**)
42+
self.class.new({ :** => {} }, @options)
43+
when @options[:allow_wildcard] && @hash.key?(:*)
44+
@hash[:*]
45+
end
46+
end
47+
48+
# @return [Hash{Symbol => Hash}]
49+
def to_hash
50+
@hash.each_with_object({}) do |(key, value), hash|
51+
hash[key] = value.to_hash
52+
end
53+
end
54+
55+
# @return [String]
56+
def to_string
57+
string_array = @hash.map do |(key, value)|
58+
string_value = value.to_string
59+
if string_value == ''
60+
key.to_s
61+
else
62+
string_value
63+
.split(',')
64+
.map { |x| key.to_s + '.' + x }
65+
.join(',')
66+
end
67+
end
68+
69+
string_array.join(',')
70+
end
71+
end
72+
end
+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
module JSONAPI
2+
class IncludeDirective
3+
# Utilities to create an IncludeDirective hash from various types of
4+
# inputs.
5+
module Parser
6+
module_function
7+
8+
# @api private
9+
def parse_include_args(include_args)
10+
case include_args
11+
when Symbol
12+
{ include_args => {} }
13+
when Hash
14+
parse_hash(include_args)
15+
when Array
16+
parse_array(include_args)
17+
when String
18+
parse_string(include_args)
19+
else
20+
{}
21+
end
22+
end
23+
24+
# @api private
25+
def parse_string(include_string)
26+
include_string.split(',')
27+
.each_with_object({}) do |path, hash|
28+
deep_merge!(hash, parse_path_string(path))
29+
end
30+
end
31+
32+
# @api private
33+
def parse_path_string(include_path)
34+
include_path.split('.')
35+
.reverse
36+
.reduce({}) { |a, e| { e.to_sym => a } }
37+
end
38+
39+
# @api private
40+
def parse_hash(include_hash)
41+
include_hash.each_with_object({}) do |(key, value), hash|
42+
hash[key.to_sym] = parse_include_args(value)
43+
end
44+
end
45+
46+
# @api private
47+
def parse_array(include_array)
48+
include_array.each_with_object({}) do |x, hash|
49+
deep_merge!(hash, parse_include_args(x))
50+
end
51+
end
52+
53+
# @api private
54+
def deep_merge!(src, ext)
55+
ext.each do |k, v|
56+
if src[k].is_a?(Hash) && v.is_a?(Hash)
57+
deep_merge!(src[k], v)
58+
else
59+
src[k] = v
60+
end
61+
end
62+
end
63+
end
64+
end
65+
end

0 commit comments

Comments
 (0)