merge_params 0.1.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 +7 -0
- data/.gitignore +12 -0
- data/.rspec +3 -0
- data/.travis.yml +7 -0
- data/Gemfile +6 -0
- data/License +18 -0
- data/Rakefile +6 -0
- data/Readme.md +84 -0
- data/bin/console +14 -0
- data/bin/setup +8 -0
- data/lib/merge_params.rb +8 -0
- data/lib/merge_params/helpers.rb +97 -0
- data/lib/merge_params/version.rb +5 -0
- data/merge_params.gemspec +36 -0
- metadata +142 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: 3c1c70a888ac373aa3469a2f5c9363f40b201fdd
|
4
|
+
data.tar.gz: 465b22de3bebaef210a4ee8a5e94de389f38a099
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: e63a767704dd575d3376bff60cc895b182b43c88f8c549394b7dbeec777964d040b83a57dd8eee5326e6c534978859a70d44eac3bf4bca2fdcf234f711f6b80c
|
7
|
+
data.tar.gz: 3e57322d8275102875e4f313ad240dba635302f4fabb9d601d1c26d92fa59845f2039f7448e05819dd6ff01424b3af6dbfceba58790b3bebc514bf9a7ab24e20
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/.travis.yml
ADDED
data/Gemfile
ADDED
data/License
ADDED
@@ -0,0 +1,18 @@
|
|
1
|
+
Copyright (c) 2018 Tyler Rick
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
|
4
|
+
associated documentation files (the "Software"), to deal in the Software without restriction,
|
5
|
+
including without limitation the rights to use, copy, modify, merge, publish, distribute,
|
6
|
+
sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
|
7
|
+
furnished to do so, subject to the following conditions:
|
8
|
+
|
9
|
+
The above copyright notice and this permission notice shall be included in all copies or substantial
|
10
|
+
portions of the Software.
|
11
|
+
|
12
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
13
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
14
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
15
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
16
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
17
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
18
|
+
SOFTWARE.
|
data/Rakefile
ADDED
data/Readme.md
ADDED
@@ -0,0 +1,84 @@
|
|
1
|
+
# MergeParams
|
2
|
+
|
3
|
+
## Why do we need it?
|
4
|
+
|
5
|
+
Have you ever wanted to take the current route and change just one parameter in the route to
|
6
|
+
generate a new route?
|
7
|
+
|
8
|
+
For example, maybe you've tried to do something like this:
|
9
|
+
|
10
|
+
```ruby
|
11
|
+
redirect_to url_for(params.merge(thing_id: thing.id));
|
12
|
+
```
|
13
|
+
|
14
|
+
or this:
|
15
|
+
|
16
|
+
```ruby
|
17
|
+
link_to 'Download as CSV', params.merge(format: :csv)
|
18
|
+
```
|
19
|
+
|
20
|
+
If you have tried that, and you are on Rails 5.0 or later, then you have probably run into this
|
21
|
+
error:
|
22
|
+
|
23
|
+
Attempting to generate a URL from non-sanitized request parameters! An attacker can
|
24
|
+
inject malicious data into the generated URL, such as changing the host. Whitelist and sanitize
|
25
|
+
passed parameters to be secure.
|
26
|
+
|
27
|
+
(See also: https://github.com/rails/rails/issues/26289)
|
28
|
+
|
29
|
+
## How do I use it?
|
30
|
+
|
31
|
+
Anywhere you would be tempted to do `params.merge(hash)`, just replace with `merge_params(hash)` or `merge_url_for(hash)`. For example:
|
32
|
+
|
33
|
+
```ruby
|
34
|
+
link_to 'Download as CSV', merge_params(format: :csv)
|
35
|
+
```
|
36
|
+
|
37
|
+
```ruby
|
38
|
+
redirect_to merge_url_for(thing_id: thing.id);
|
39
|
+
```
|
40
|
+
|
41
|
+
## Is it guaranteed to be safe?
|
42
|
+
|
43
|
+
No. While a best effort has been made to ensure unsafe params are not used to generate a URL, we may
|
44
|
+
have overlooked something. Please review the code and the tests (coming soon) and open an issue if
|
45
|
+
you find any security holes in this approach.
|
46
|
+
|
47
|
+
## Other helpers
|
48
|
+
|
49
|
+
Unlike `url_for_merge`, which tries to generate a route from the given params, sometimes you just
|
50
|
+
want to add the given params to the "end" of the URL as part of the query string:
|
51
|
+
|
52
|
+
```ruby
|
53
|
+
add_params(key: 'value')
|
54
|
+
# => "/current_path?key=value
|
55
|
+
|
56
|
+
add_params({key: 'value'}, '/other_url')
|
57
|
+
# => "/other_url?key=value
|
58
|
+
```
|
59
|
+
|
60
|
+
## Installation
|
61
|
+
|
62
|
+
Add this line to your application's Gemfile:
|
63
|
+
|
64
|
+
```ruby
|
65
|
+
gem 'merge_params'
|
66
|
+
```
|
67
|
+
|
68
|
+
Add this line to your `ApplicationController` (or whichever controller you want to have the
|
69
|
+
helpers):
|
70
|
+
|
71
|
+
```ruby
|
72
|
+
include MergeParams::Helpers
|
73
|
+
```
|
74
|
+
|
75
|
+
The helpers will be also be added with `helper_method` so that they are available for use in view
|
76
|
+
templates as well.
|
77
|
+
|
78
|
+
## Similar projects
|
79
|
+
|
80
|
+
- [uri_query_merger](https://libraries.io/github/jordanmaguire/uri_query_merger)
|
81
|
+
|
82
|
+
## Contributing
|
83
|
+
|
84
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/TylerRick/merge_params.
|
data/bin/console
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
|
3
|
+
require "bundler/setup"
|
4
|
+
require "merge_params"
|
5
|
+
|
6
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
7
|
+
# with your gem easier. You can also use a different console, if you like.
|
8
|
+
|
9
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
10
|
+
# require "pry"
|
11
|
+
# Pry.start
|
12
|
+
|
13
|
+
require "irb"
|
14
|
+
IRB.start(__FILE__)
|
data/bin/setup
ADDED
data/lib/merge_params.rb
ADDED
@@ -0,0 +1,97 @@
|
|
1
|
+
require 'uri'
|
2
|
+
|
3
|
+
module MergeParams::Helpers
|
4
|
+
extend ActiveSupport::Concern
|
5
|
+
|
6
|
+
# request.parameters but with symbolized keys.
|
7
|
+
def request_params
|
8
|
+
request.parameters.symbolize_keys
|
9
|
+
end
|
10
|
+
|
11
|
+
# request.parameters (which also includes POST params) but with only those keys that would
|
12
|
+
# normally be passed in a query string (without :controller, :action, :format) and with symbolized
|
13
|
+
# keys.
|
14
|
+
def query_params_from_request_params
|
15
|
+
request.parameters.symbolize_keys.
|
16
|
+
except(*request.path_parameters.symbolize_keys.keys)
|
17
|
+
end
|
18
|
+
|
19
|
+
# Returns a hash of params from the query string (https://en.wikipedia.org/wiki/Query_string),
|
20
|
+
# with symbolized keys.
|
21
|
+
def query_params
|
22
|
+
request.query_parameters.symbolize_keys
|
23
|
+
end
|
24
|
+
|
25
|
+
# Params that can safely be passed to url_for to build a route. (Used by merge_url_for.)
|
26
|
+
#
|
27
|
+
# We exclude RESERVED_OPTIONS such as :host because such options should only come from your app
|
28
|
+
# code. Allowing :host to be set via query params, for example, means a bad actor could cause
|
29
|
+
# links that go to a different site entirely:
|
30
|
+
#
|
31
|
+
# # Request for /things?host=somehackingsite.ru
|
32
|
+
# url_for(params) => "http://somehackingsite.ru/things"
|
33
|
+
#
|
34
|
+
# Similarly, the :controller and :action keys of `params` *never* come from the query string, but
|
35
|
+
# from `path_parameters`. (TODO: So why not just use params.except(...)?)
|
36
|
+
#
|
37
|
+
# TODO: Why not allow :format from params? To force people to use .:format? But doesn't that also
|
38
|
+
# come through as params?
|
39
|
+
#
|
40
|
+
# (And we don't even need to pass the path_parameters on to url_for because url_for already
|
41
|
+
# includes those (from :_recall)
|
42
|
+
#
|
43
|
+
def params_for_url_for
|
44
|
+
params.to_unsafe_h.symbolize_keys.except(
|
45
|
+
*ActionDispatch::Routing::RouteSet::RESERVED_OPTIONS,
|
46
|
+
:controller,
|
47
|
+
:action,
|
48
|
+
:format
|
49
|
+
)
|
50
|
+
end
|
51
|
+
|
52
|
+
# Safely merges the given params with the params from the current request
|
53
|
+
def merge_params(new_params = {})
|
54
|
+
params_for_url_for.merge(new_params)
|
55
|
+
end
|
56
|
+
|
57
|
+
# Safely merges the given params with the params from the current request, then generates a route
|
58
|
+
# from the merged params.
|
59
|
+
def merge_url_for(new_params = {})
|
60
|
+
url = url_for(merge_params(new_params))
|
61
|
+
|
62
|
+
# Now pass along in the *query string* any params that we couldn't pass to url_for because they
|
63
|
+
# were reserved options.
|
64
|
+
query_params_already_added = parse_nested_query(URI(url).query || '')
|
65
|
+
query_params_to_add = query_params.except(*query_params_already_added.keys)
|
66
|
+
add_params(query_params_to_add, url)
|
67
|
+
end
|
68
|
+
|
69
|
+
# Adds params to the query string
|
70
|
+
# (Unlike url_for_merge, which tries to generate a route from the params.)
|
71
|
+
# TODO: Should URL be first like https://libraries.io/github/jordanmaguire/uri_query_merger ?
|
72
|
+
# UriQueryMerger.new("http://www.google.com?other=1", {jordan: "rules"}).merge
|
73
|
+
# Can we make it work that way when a URL is supplied buth otherwise let the params be the first
|
74
|
+
# and only argument (to optimize for that more common use case)?
|
75
|
+
def add_params(params = {}, url = request.fullpath)
|
76
|
+
uri = URI(url)
|
77
|
+
params = parse_nested_query(uri.query || '').merge(params)
|
78
|
+
uri.query = Rack::Utils.build_nested_query(params)
|
79
|
+
uri.to_s
|
80
|
+
end
|
81
|
+
|
82
|
+
included do
|
83
|
+
helper_method(
|
84
|
+
:request_params,
|
85
|
+
:params_for_url_for,
|
86
|
+
:merge_params,
|
87
|
+
:merge_url_for,
|
88
|
+
:add_params
|
89
|
+
) if respond_to?(:helper_method)
|
90
|
+
end
|
91
|
+
|
92
|
+
private
|
93
|
+
|
94
|
+
def parse_nested_query(query)
|
95
|
+
Rack::Utils.parse_nested_query(query || '').symbolize_keys
|
96
|
+
end
|
97
|
+
end
|
@@ -0,0 +1,36 @@
|
|
1
|
+
lib = File.expand_path("../lib", __FILE__)
|
2
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
3
|
+
require "merge_params/version"
|
4
|
+
|
5
|
+
Gem::Specification.new do |spec|
|
6
|
+
spec.name = "merge_params"
|
7
|
+
spec.version = MergeParams.version
|
8
|
+
spec.authors = ["Tyler Rick"]
|
9
|
+
spec.email = ["tyler@tylerrick.com"]
|
10
|
+
spec.license = "MIT"
|
11
|
+
|
12
|
+
spec.summary = %q{Safely merge params for use with url_for or for the query string}
|
13
|
+
spec.description = spec.summary
|
14
|
+
spec.homepage = "http://github.com/TylerRick/merge_params"
|
15
|
+
|
16
|
+
spec.metadata["homepage_uri"] = spec.homepage
|
17
|
+
spec.metadata["source_code_uri"] = spec.homepage
|
18
|
+
spec.metadata["changelog_uri"] = "TODO: Put your gem's Changelog.md URL here."
|
19
|
+
|
20
|
+
# Specify which files should be added to the gem when it is released.
|
21
|
+
# The `git ls-files -z` loads the files in the RubyGem that have been added into git.
|
22
|
+
spec.files = Dir.chdir(File.expand_path('..', __FILE__)) do
|
23
|
+
`git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
|
24
|
+
end
|
25
|
+
spec.bindir = "exe"
|
26
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
27
|
+
spec.require_paths = ["lib"]
|
28
|
+
|
29
|
+
spec.required_ruby_version = ">= 2.3.0"
|
30
|
+
spec.add_dependency "activesupport", [">= 4.2", "< 5.3"]
|
31
|
+
spec.add_dependency "actionpack", [">= 4.2", "< 5.3"]
|
32
|
+
|
33
|
+
spec.add_development_dependency "bundler", "~> 1.17"
|
34
|
+
spec.add_development_dependency "rake", "~> 10.0"
|
35
|
+
spec.add_development_dependency "rspec", "~> 3.0"
|
36
|
+
end
|
metadata
ADDED
@@ -0,0 +1,142 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: merge_params
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Tyler Rick
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2018-11-16 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: activesupport
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - ">="
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '4.2'
|
20
|
+
- - "<"
|
21
|
+
- !ruby/object:Gem::Version
|
22
|
+
version: '5.3'
|
23
|
+
type: :runtime
|
24
|
+
prerelease: false
|
25
|
+
version_requirements: !ruby/object:Gem::Requirement
|
26
|
+
requirements:
|
27
|
+
- - ">="
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: '4.2'
|
30
|
+
- - "<"
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '5.3'
|
33
|
+
- !ruby/object:Gem::Dependency
|
34
|
+
name: actionpack
|
35
|
+
requirement: !ruby/object:Gem::Requirement
|
36
|
+
requirements:
|
37
|
+
- - ">="
|
38
|
+
- !ruby/object:Gem::Version
|
39
|
+
version: '4.2'
|
40
|
+
- - "<"
|
41
|
+
- !ruby/object:Gem::Version
|
42
|
+
version: '5.3'
|
43
|
+
type: :runtime
|
44
|
+
prerelease: false
|
45
|
+
version_requirements: !ruby/object:Gem::Requirement
|
46
|
+
requirements:
|
47
|
+
- - ">="
|
48
|
+
- !ruby/object:Gem::Version
|
49
|
+
version: '4.2'
|
50
|
+
- - "<"
|
51
|
+
- !ruby/object:Gem::Version
|
52
|
+
version: '5.3'
|
53
|
+
- !ruby/object:Gem::Dependency
|
54
|
+
name: bundler
|
55
|
+
requirement: !ruby/object:Gem::Requirement
|
56
|
+
requirements:
|
57
|
+
- - "~>"
|
58
|
+
- !ruby/object:Gem::Version
|
59
|
+
version: '1.17'
|
60
|
+
type: :development
|
61
|
+
prerelease: false
|
62
|
+
version_requirements: !ruby/object:Gem::Requirement
|
63
|
+
requirements:
|
64
|
+
- - "~>"
|
65
|
+
- !ruby/object:Gem::Version
|
66
|
+
version: '1.17'
|
67
|
+
- !ruby/object:Gem::Dependency
|
68
|
+
name: rake
|
69
|
+
requirement: !ruby/object:Gem::Requirement
|
70
|
+
requirements:
|
71
|
+
- - "~>"
|
72
|
+
- !ruby/object:Gem::Version
|
73
|
+
version: '10.0'
|
74
|
+
type: :development
|
75
|
+
prerelease: false
|
76
|
+
version_requirements: !ruby/object:Gem::Requirement
|
77
|
+
requirements:
|
78
|
+
- - "~>"
|
79
|
+
- !ruby/object:Gem::Version
|
80
|
+
version: '10.0'
|
81
|
+
- !ruby/object:Gem::Dependency
|
82
|
+
name: rspec
|
83
|
+
requirement: !ruby/object:Gem::Requirement
|
84
|
+
requirements:
|
85
|
+
- - "~>"
|
86
|
+
- !ruby/object:Gem::Version
|
87
|
+
version: '3.0'
|
88
|
+
type: :development
|
89
|
+
prerelease: false
|
90
|
+
version_requirements: !ruby/object:Gem::Requirement
|
91
|
+
requirements:
|
92
|
+
- - "~>"
|
93
|
+
- !ruby/object:Gem::Version
|
94
|
+
version: '3.0'
|
95
|
+
description: Safely merge params for use with url_for or for the query string
|
96
|
+
email:
|
97
|
+
- tyler@tylerrick.com
|
98
|
+
executables: []
|
99
|
+
extensions: []
|
100
|
+
extra_rdoc_files: []
|
101
|
+
files:
|
102
|
+
- ".gitignore"
|
103
|
+
- ".rspec"
|
104
|
+
- ".travis.yml"
|
105
|
+
- Gemfile
|
106
|
+
- License
|
107
|
+
- Rakefile
|
108
|
+
- Readme.md
|
109
|
+
- bin/console
|
110
|
+
- bin/setup
|
111
|
+
- lib/merge_params.rb
|
112
|
+
- lib/merge_params/helpers.rb
|
113
|
+
- lib/merge_params/version.rb
|
114
|
+
- merge_params.gemspec
|
115
|
+
homepage: http://github.com/TylerRick/merge_params
|
116
|
+
licenses:
|
117
|
+
- MIT
|
118
|
+
metadata:
|
119
|
+
homepage_uri: http://github.com/TylerRick/merge_params
|
120
|
+
source_code_uri: http://github.com/TylerRick/merge_params
|
121
|
+
changelog_uri: 'TODO: Put your gem''s Changelog.md URL here.'
|
122
|
+
post_install_message:
|
123
|
+
rdoc_options: []
|
124
|
+
require_paths:
|
125
|
+
- lib
|
126
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
127
|
+
requirements:
|
128
|
+
- - ">="
|
129
|
+
- !ruby/object:Gem::Version
|
130
|
+
version: 2.3.0
|
131
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
132
|
+
requirements:
|
133
|
+
- - ">="
|
134
|
+
- !ruby/object:Gem::Version
|
135
|
+
version: '0'
|
136
|
+
requirements: []
|
137
|
+
rubyforge_project:
|
138
|
+
rubygems_version: 2.6.11
|
139
|
+
signing_key:
|
140
|
+
specification_version: 4
|
141
|
+
summary: Safely merge params for use with url_for or for the query string
|
142
|
+
test_files: []
|