json_errors 0.1.1 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 26354ae40f5933c7f0c361ff1ec64f238a6d27571315192ba030dc534c22066f
4
- data.tar.gz: 206463accc495b565c19e66b1a8b1d9e0dc1df4fd21b22f50b6e5d6a36954c1e
3
+ metadata.gz: '09b6a5462beee5879c2112dcf5058e6a7a2641eb4268fc11f222fe854fe921e4'
4
+ data.tar.gz: 0c8542bd919023acadb2ec9ed83bd6204c9546e64785b7f62180ce067270837d
5
5
  SHA512:
6
- metadata.gz: 1426aa90ee3cb74f7dd1c3264d60cc5c5747bd170314ba6f19a4ac45d4200c44115e590648b8d298fa58253edf104c4a39aa97d40e893aeec70ec94451e3226f
7
- data.tar.gz: c7662a61fcdb269b637914ed33712f4b54271304a7c85bb11e6aecb90322494d03c04cf861c9b64f9513ebadcdf369483bd7c066560e25b2324ebbb1a16a0e3c
6
+ metadata.gz: 687d743cb610a6b2276781b9d57285035cf75095330ae7828161d4941b0146fdb8198f1b76a4b6cfab1e8cc827ff411539a3d1fce599112382c23511fd84f4d8
7
+ data.tar.gz: c12c26908c735334d3d8589449c6af052ecf1766028e0a6e79d57d1169c4626889a507d88da1f2b265facd4243f1b71502c0b53c71bc98d809603f1e78a87bac
data/CHANGELOG.md CHANGED
@@ -1,5 +1,10 @@
1
1
  ## [Unreleased]
2
2
 
3
+ ## [0.2.0] - 2021-08-03
4
+
5
+ - ActiveRecord validation errors handling added
6
+ - Custom payload added
7
+
3
8
  ## [0.1.1] - 2021-07-27
4
9
 
5
10
  - Missing configuration error added
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- json_errors (0.1.1)
4
+ json_errors (0.2.0)
5
5
  activesupport (>= 4.2.0)
6
6
 
7
7
  GEM
@@ -73,11 +73,11 @@ GEM
73
73
  crass (1.0.6)
74
74
  diff-lcs (1.4.4)
75
75
  erubi (1.10.0)
76
- globalid (0.4.2)
77
- activesupport (>= 4.2.0)
76
+ globalid (0.5.2)
77
+ activesupport (>= 5.0)
78
78
  i18n (1.8.10)
79
79
  concurrent-ruby (~> 1.0)
80
- loofah (2.10.0)
80
+ loofah (2.11.0)
81
81
  crass (~> 1.0.2)
82
82
  nokogiri (>= 1.5.9)
83
83
  mail (2.7.1)
@@ -87,7 +87,7 @@ GEM
87
87
  mini_mime (1.1.0)
88
88
  minitest (5.14.4)
89
89
  nio4r (2.5.7)
90
- nokogiri (1.11.7-x86_64-darwin)
90
+ nokogiri (1.12.0-x86_64-darwin)
91
91
  racc (~> 1.4)
92
92
  parallel (1.20.1)
93
93
  parser (3.0.2.0)
data/README.md CHANGED
@@ -21,113 +21,9 @@ Or install it yourself as:
21
21
 
22
22
  ___
23
23
  ## Usage
24
- ### Basic usage
25
24
 
26
- First, run the `json_errors:install` generator:
27
- ```
28
- > rails g json_errors:install
29
- create config/initializers/json_errors.rb
30
- ```
31
-
32
- It creates the initializer file with config. Check out the file and uncomment what you need.
33
-
34
- Then include `JsonErrors::Rescuer` to your JSON API controller and start raising errors.
35
- ```ruby
36
- class UsersController < ApplicationController
37
- include JsonErrors::Rescuer
38
-
39
- def create
40
- user = User.create!(params.require(:name))
41
-
42
- head :created
43
- end
44
- end
45
- ```
46
- Be sure to use error throwing methods everywhere, like `create!`, `save!`, `find_by!`, etc. The gem will rescue from any exception and respond with a properly formatted JSON body with a custom error code.
47
-
48
- No need for `rescue_from` or rendering the JSON error body. Every error is rescued, logged, formatted and rendered. The client application will receive a JSON body like this:
49
-
50
- ```json
51
- {
52
- "code":1010,
53
- "message":"The parameter 'name' is missing or value is empty",
54
- "payload":[]
55
- }
56
- ```
57
-
58
- ### Custom codes
59
-
60
- The initializer file defines custom error codes and error dictionary:
61
- ```ruby
62
- JsonErrors.configure do |config|
63
- config.custom_codes = {
64
- general_error: { code: 1001, http_status: 500 },
65
- not_found: { code: 1002, http_status: 404 },
66
- database_error: { code: 1003, http_status: 500 },
67
- parameter_missing: { code: 1010, http_status: 400 },
68
- validation_failed: { code: 1020, http_status: 422 },
69
- internal_server_error: { code: 1000, http_status: 500 }
70
- }
71
-
72
- config.error_dictionary = {
73
- ActiveRecord::RecordInvalid => :validation_failed,
74
- ActionController::ParameterMissing => :parameter_missing,
75
- ActiveRecord::RecordNotFound => :not_found,
76
- ActiveRecord::ActiveRecordError => :database_error,
77
- StandardError => :internal_server_error
78
- }
79
- end
80
- ```
81
-
82
- The `custom_codes` hash assigns the error label to the error code and HTTP status code of the response.
83
- So, the error labelled with `:general_error` will be considered as `1001` code under the HTTP `500 Internal Server Error` response.
84
- Codes are customizable. They can be either numeric or string.
85
-
86
- The `error_dictionary` hash defines what error classes are connected to which error codes.
87
-
88
- You can create different custom codes and connect them to your error classes.
89
- Lets say you have authentication and you want to respond with HTTP `403 Forbidden` whenever someone is unauthenticated. You can do it 2 ways:
90
-
91
- ### **1. By handling custom error class**
25
+ Check out the [manual](https://github.com/nomtek/JsonErrors/wiki)
92
26
 
93
- Add new label to the `custom_codes`
94
- ```ruby
95
- unauthenticated: { code: 2001, http_status: 403 }
96
- ```
97
- and your error class to the `error_dictionary`:
98
- ```ruby
99
- Unauthenticated => :unauthenticated
100
- ```
101
-
102
- Then just raise the error while checking the authentication.
103
-
104
- ```ruby
105
- Unauthenticated = Class.new(StandardError)
106
-
107
- raise Unauthenticated, 'Authentication needed'
108
- ```
109
-
110
-
111
- ### **2. By raising error by label**
112
- Add new label to the `custom_codes`
113
- ```ruby
114
- unauthenticated: { code: 2001, http_status: 403 }
115
- ```
116
- Then just raise the error while checking the authentication.
117
-
118
- ```ruby
119
- raise JsonErrors::ApplicationError.unauthenticated('Authentication needed')
120
- ```
121
-
122
- Both cases you will get the HTTP `403 Forbidden` response with JSON formatted body:
123
-
124
- ```json
125
- {
126
- "code":2001,
127
- "message":"Authentication needed",
128
- "payload":[]
129
- }
130
- ```
131
27
  ___
132
28
 
133
29
  ## Development
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ JsonErrors.configure do |config|
4
+ config.custom_codes = {
5
+ # general_error: { code: 1001, http_status: 500 },
6
+ # not_found: { code: 1002, http_status: 404 },
7
+ # database_error: { code: 1003, http_status: 500 },
8
+ # parameter_missing: { code: 1010, http_status: 400 },
9
+ # validation_failed: { code: 1020, http_status: 422 },
10
+ internal_server_error: { code: 1000, http_status: 500 }
11
+ }
12
+
13
+ config.error_dictionary = {
14
+ # Uncomment the following lines according to needs.
15
+ # They are caught and rescued in a cascade manned top to bottom.
16
+
17
+ # ActiveRecord::RecordInvalid => :validation_failed,
18
+ # ActionController::ParameterMissing => :parameter_missing,
19
+ # ActiveRecord::RecordNotFound => :not_found,
20
+ # ActiveRecord::ActiveRecordError => :database_error,
21
+ StandardError => :internal_server_error
22
+ }
23
+ end
data/json_errors.gemspec CHANGED
@@ -16,7 +16,7 @@ Gem::Specification.new do |spec|
16
16
 
17
17
  spec.metadata['homepage_uri'] = spec.homepage
18
18
  spec.metadata['source_code_uri'] = 'https://github.com/nomtek/JsonErrors'
19
- spec.metadata['changelog_uri'] = 'https://github.com/nomtek/JsonErrors/CHANGELOG.md'
19
+ spec.metadata['changelog_uri'] = 'https://github.com/nomtek/JsonErrors/blob/main/CHANGELOG.md'
20
20
 
21
21
  # Specify which files should be added to the gem when it is released.
22
22
  # The `git ls-files -z` loads the files in the RubyGem that have been added into git.
@@ -3,21 +3,21 @@
3
3
  JsonErrors.configure do |config|
4
4
  config.custom_codes = {
5
5
  # general_error: { code: 1001, http_status: 500 },
6
- # not_found: { code: 1002, http_status: 404 },
7
- # database_error: { code: 1003, http_status: 500 },
8
- # parameter_missing: { code: 1010, http_status: 400 },
9
- # validation_failed: { code: 1020, http_status: 422 },
6
+ not_found: { code: 1002, http_status: 404 },
7
+ database_error: { code: 1003, http_status: 500 },
8
+ parameter_missing: { code: 1010, http_status: 400 },
9
+ validation_failed: { code: 1020, http_status: 422, validation_errors: :active_record },
10
10
  internal_server_error: { code: 1000, http_status: 500 }
11
11
  }
12
12
 
13
13
  config.error_dictionary = {
14
- # Uncomment the following lines according to needs.
15
- # They are caught and rescued in a cascade manned top to bottom.
14
+ # Uncomment or comment the following lines according to needs.
15
+ # They are caught and rescued in a cascade manner top to bottom.
16
16
 
17
- # ActiveRecord::RecordInvalid => :validation_failed,
18
- # ActionController::ParameterMissing => :parameter_missing,
19
- # ActiveRecord::RecordNotFound => :not_found,
20
- # ActiveRecord::ActiveRecordError => :database_error,
17
+ ActiveRecord::RecordInvalid => :validation_failed,
18
+ ActionController::ParameterMissing => :parameter_missing,
19
+ ActiveRecord::RecordNotFound => :not_found,
20
+ ActiveRecord::ActiveRecordError => :database_error,
21
21
  StandardError => :internal_server_error
22
22
  }
23
23
  end
data/lib/json_errors.rb CHANGED
@@ -2,7 +2,10 @@
2
2
 
3
3
  require_relative 'json_errors/version'
4
4
  require_relative 'json_errors/config'
5
- require_relative 'json_errors/application_error'
5
+ require_relative 'json_errors/error'
6
+ require_relative 'json_errors/error/basic_error'
7
+ require_relative 'json_errors/error/custom_payload_error'
8
+ require_relative 'json_errors/error/validation_error'
6
9
  require_relative 'json_errors/rescuer'
7
10
 
8
11
  # Main module
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ module JsonErrors
4
+ # Error facade
5
+ class Error
6
+ def self.method_missing(name, *args)
7
+ message, payload = args
8
+ return super unless name.in?(codes.keys)
9
+
10
+ return BasicError.new(message, name) if payload.nil?
11
+ return ValidationError.new(message, name, payload&.record) if codes[name][:validation_errors] == :active_record
12
+
13
+ CustomPayloadError.new(message, name, payload)
14
+ end
15
+
16
+ def self.respond_to_missing?(name, _respond_to_private = false)
17
+ name.in?(codes.keys) || super
18
+ end
19
+
20
+ def self.codes
21
+ JsonErrors.config.custom_codes
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,39 @@
1
+ # frozen_string_literal: true
2
+
3
+ module JsonErrors
4
+ # Main error class to be rescued from
5
+ class BasicError < StandardError
6
+ attr_reader :code
7
+
8
+ def initialize(msg, name)
9
+ raise 'Wrong name' unless name.in?(codes.keys)
10
+
11
+ @code = codes[name][:code]
12
+ @name = name
13
+ super(msg)
14
+ end
15
+
16
+ def self.codes
17
+ JsonErrors.config.custom_codes
18
+ end
19
+
20
+ def to_json(_options = nil)
21
+ {
22
+ code: code,
23
+ message: message
24
+ }.to_json
25
+ end
26
+
27
+ def http_status
28
+ codes[name][:http_status]
29
+ end
30
+
31
+ private
32
+
33
+ attr_reader :name
34
+
35
+ def codes
36
+ self.class.codes
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ module JsonErrors
4
+ # Error class for custom payload errors
5
+ class CustomPayloadError < BasicError
6
+ attr_accessor :payload
7
+
8
+ def initialize(msg, name, payload)
9
+ super(msg, name)
10
+ @payload = payload
11
+ end
12
+
13
+ def to_json(_options = nil)
14
+ {
15
+ code: code,
16
+ message: message,
17
+ payload: payload
18
+ }.to_json
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,34 @@
1
+ # frozen_string_literal: true
2
+
3
+ module JsonErrors
4
+ # Error class for custom payload errors
5
+ class ValidationError < BasicError
6
+ def initialize(msg, name, record)
7
+ super(msg, name)
8
+ raise 'Wrong record' unless record.respond_to?(:errors)
9
+
10
+ @record = record
11
+ end
12
+
13
+ def to_json(_options = nil)
14
+ {
15
+ code: code,
16
+ message: message,
17
+ payload: payload
18
+ }.to_json
19
+ end
20
+
21
+ private
22
+
23
+ attr_reader :record
24
+
25
+ def payload
26
+ validation_payload = []
27
+ record.errors.each do |key, messages|
28
+ validation_payload << { key => messages }
29
+ end
30
+
31
+ { record.class.to_s => validation_payload }
32
+ end
33
+ end
34
+ end
@@ -15,11 +15,11 @@ module JsonErrors
15
15
  error_dictionary.keys.reverse.each do |error_class|
16
16
  rescue_from error_class do |error|
17
17
  log_error(error)
18
- render_error ApplicationError.send(error_dictionary[error_class], error)
18
+ render_error JsonErrors::Error.send(error_dictionary[error_class], error)
19
19
  end
20
20
  end
21
21
 
22
- rescue_from ApplicationError do |error|
22
+ rescue_from JsonErrors::BasicError do |error|
23
23
  log_error(error)
24
24
  render_error(error)
25
25
  end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module JsonErrors
4
- VERSION = '0.1.1'
4
+ VERSION = '0.2.0'
5
5
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: json_errors
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.1
4
+ version: 0.2.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Łukasz Pająk
@@ -9,7 +9,7 @@ authors:
9
9
  autorequire:
10
10
  bindir: exe
11
11
  cert_chain: []
12
- date: 2021-07-27 00:00:00.000000000 Z
12
+ date: 2021-08-03 00:00:00.000000000 Z
13
13
  dependencies:
14
14
  - !ruby/object:Gem::Dependency
15
15
  name: activesupport
@@ -45,13 +45,17 @@ files:
45
45
  - Rakefile
46
46
  - bin/console
47
47
  - bin/setup
48
+ - config/initializers/json_errors.rb
48
49
  - json_errors.gemspec
49
50
  - lib/generators/json_errors/install/USAGE
50
51
  - lib/generators/json_errors/install/install_generator.rb
51
52
  - lib/generators/json_errors/install/templates/json_errors.rb
52
53
  - lib/json_errors.rb
53
- - lib/json_errors/application_error.rb
54
54
  - lib/json_errors/config.rb
55
+ - lib/json_errors/error.rb
56
+ - lib/json_errors/error/basic_error.rb
57
+ - lib/json_errors/error/custom_payload_error.rb
58
+ - lib/json_errors/error/validation_error.rb
55
59
  - lib/json_errors/rescuer.rb
56
60
  - lib/json_errors/version.rb
57
61
  homepage: https://github.com/nomtek/JsonErrors
@@ -60,7 +64,7 @@ licenses:
60
64
  metadata:
61
65
  homepage_uri: https://github.com/nomtek/JsonErrors
62
66
  source_code_uri: https://github.com/nomtek/JsonErrors
63
- changelog_uri: https://github.com/nomtek/JsonErrors/CHANGELOG.md
67
+ changelog_uri: https://github.com/nomtek/JsonErrors/blob/main/CHANGELOG.md
64
68
  post_install_message:
65
69
  rdoc_options: []
66
70
  require_paths:
@@ -1,58 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- module JsonErrors
4
- # Main error class to be rescued from
5
- class ApplicationError < StandardError
6
- attr_reader :code, :payload
7
-
8
- def initialize(msg, name, payload = [])
9
- raise 'Wrong name' unless name.in?(codes.keys)
10
-
11
- @code = codes[name][:code]
12
- @name = name
13
- @payload = payload
14
- super(msg)
15
- end
16
-
17
- def self.method_missing(name, *args)
18
- error = args.first
19
- return super if error.nil?
20
- return super unless name.in?(codes.keys)
21
- return new(error.to_s, name) unless error.respond_to?(:record)
22
-
23
- validation_payload = []
24
- error.record.errors.each do |key, messages|
25
- validation_payload << { key => messages }
26
- end
27
- new(error.to_s, name, validation_payload)
28
- end
29
-
30
- def self.respond_to_missing?(name, _respond_to_private = false)
31
- name.in?(codes.keys) || super
32
- end
33
-
34
- def self.codes
35
- JsonErrors.config.custom_codes
36
- end
37
-
38
- def to_json(_options = nil)
39
- {
40
- code: code,
41
- message: message,
42
- payload: payload
43
- }.to_json
44
- end
45
-
46
- def http_status
47
- codes[name][:http_status]
48
- end
49
-
50
- private
51
-
52
- attr_reader :name
53
-
54
- def codes
55
- self.class.codes
56
- end
57
- end
58
- end