hanami-utils 0.0.0 → 0.7.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 +4 -4
- data/CHANGELOG.md +183 -0
- data/LICENSE.md +22 -0
- data/README.md +106 -7
- data/hanami-utils.gemspec +15 -13
- data/lib/hanami-utils.rb +1 -0
- data/lib/hanami/interactor.rb +497 -0
- data/lib/hanami/logger.rb +141 -0
- data/lib/hanami/utils.rb +31 -2
- data/lib/hanami/utils/attributes.rb +132 -0
- data/lib/hanami/utils/basic_object.rb +53 -0
- data/lib/hanami/utils/callbacks.rb +286 -0
- data/lib/hanami/utils/class.rb +94 -0
- data/lib/hanami/utils/class_attribute.rb +95 -0
- data/lib/hanami/utils/deprecation.rb +70 -0
- data/lib/hanami/utils/duplicable.rb +80 -0
- data/lib/hanami/utils/escape.rb +577 -0
- data/lib/hanami/utils/hash.rb +299 -0
- data/lib/hanami/utils/inflector.rb +439 -0
- data/lib/hanami/utils/io.rb +37 -0
- data/lib/hanami/utils/kernel.rb +1031 -0
- data/lib/hanami/utils/load_paths.rb +167 -0
- data/lib/hanami/utils/path_prefix.rb +146 -0
- data/lib/hanami/utils/string.rb +419 -0
- data/lib/hanami/utils/version.rb +4 -1
- metadata +51 -16
- data/.gitignore +0 -9
- data/Gemfile +0 -4
- data/Rakefile +0 -2
- data/bin/console +0 -14
- data/bin/setup +0 -8
@@ -0,0 +1,94 @@
|
|
1
|
+
require 'hanami/utils/string'
|
2
|
+
|
3
|
+
module Hanami
|
4
|
+
module Utils
|
5
|
+
# Class utilities
|
6
|
+
# @since 0.1.0
|
7
|
+
class Class
|
8
|
+
# Loads a class for the given name.
|
9
|
+
#
|
10
|
+
# @param name [String, Class] the specific class name
|
11
|
+
# @param namespace [Class, Module] the Ruby namespace where we want to perform the lookup.
|
12
|
+
# @return [Class, Module] the found Ruby constant.
|
13
|
+
#
|
14
|
+
# @raise [NameError] if no constant can be found.
|
15
|
+
#
|
16
|
+
# @since 0.1.0
|
17
|
+
#
|
18
|
+
# @example
|
19
|
+
# require 'hanami/utils/class'
|
20
|
+
#
|
21
|
+
# module App
|
22
|
+
# module Service
|
23
|
+
# class Endpoint
|
24
|
+
# end
|
25
|
+
# end
|
26
|
+
#
|
27
|
+
# class ServiceEndpoint
|
28
|
+
# end
|
29
|
+
# end
|
30
|
+
#
|
31
|
+
# # basic usage
|
32
|
+
# Hanami::Utils::Class.load!('App::Service') # => App::Service
|
33
|
+
# Hanami::Utils::Class.load!(App::Service) # => App::Service
|
34
|
+
#
|
35
|
+
# # with explicit namespace
|
36
|
+
# Hanami::Utils::Class.load!('Service', App) # => App::Service
|
37
|
+
#
|
38
|
+
# # with missing constant
|
39
|
+
# Hanami::Utils::Class.load!('Unknown') # => raises NameError
|
40
|
+
def self.load!(name, namespace = Object)
|
41
|
+
namespace.const_get(name.to_s)
|
42
|
+
end
|
43
|
+
|
44
|
+
# Loads a class from the given pattern name and namespace
|
45
|
+
#
|
46
|
+
# @param pattern [String] the class name pattern
|
47
|
+
# @param namespace [Class, Module] the Ruby namespace where we want to perform the lookup.
|
48
|
+
# @return [Class, Module] the found Ruby constant.
|
49
|
+
#
|
50
|
+
# @raise [NameError] if no constant can be found.
|
51
|
+
#
|
52
|
+
# @since 0.3.1
|
53
|
+
#
|
54
|
+
# @see Hanami::Utils::String#tokenize
|
55
|
+
#
|
56
|
+
# @example
|
57
|
+
# require 'hanami/utils/class'
|
58
|
+
#
|
59
|
+
# module App
|
60
|
+
# module Service
|
61
|
+
# class Endpoint
|
62
|
+
# end
|
63
|
+
# end
|
64
|
+
#
|
65
|
+
# class ServiceEndpoint
|
66
|
+
# end
|
67
|
+
# end
|
68
|
+
#
|
69
|
+
# # basic usage
|
70
|
+
# Hanami::Utils::Class.load_from_pattern!('App::Service') # => App::Service
|
71
|
+
#
|
72
|
+
# # with explicit namespace
|
73
|
+
# Hanami::Utils::Class.load_from_pattern!('Service', App) # => App::Service
|
74
|
+
#
|
75
|
+
# # with pattern
|
76
|
+
# Hanami::Utils::Class.load_from_pattern!('App::Service(::Endpoint|Endpoint)') # => App::Service::Endpoint
|
77
|
+
# Hanami::Utils::Class.load_from_pattern!('App::Service(Endpoint|::Endpoint)') # => App::ServiceEndpoint
|
78
|
+
#
|
79
|
+
# # with missing constant
|
80
|
+
# Hanami::Utils::Class.load_from_pattern!('Unknown') # => raises NameError
|
81
|
+
def self.load_from_pattern!(pattern, namespace = Object)
|
82
|
+
String.new(pattern).tokenize do |token|
|
83
|
+
begin
|
84
|
+
return namespace.const_get(token)
|
85
|
+
rescue NameError
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
full_name = [ (namespace == Object ? nil : namespace), pattern ].compact.join('::')
|
90
|
+
raise NameError.new("uninitialized constant #{ full_name }")
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end
|
@@ -0,0 +1,95 @@
|
|
1
|
+
require 'set'
|
2
|
+
require 'hanami/utils/duplicable'
|
3
|
+
|
4
|
+
module Hanami
|
5
|
+
module Utils
|
6
|
+
# Inheritable class level variable accessors.
|
7
|
+
# @since 0.1.0
|
8
|
+
#
|
9
|
+
# @see Hanami::Utils::ClassAttribute::ClassMethods
|
10
|
+
module ClassAttribute
|
11
|
+
def self.included(base)
|
12
|
+
base.extend ClassMethods
|
13
|
+
end
|
14
|
+
|
15
|
+
module ClassMethods
|
16
|
+
# Defines a class level accessor for the given attribute(s).
|
17
|
+
#
|
18
|
+
# A value set for a superclass is automatically available by their
|
19
|
+
# subclasses, unless a different value is explicitely set within the
|
20
|
+
# inheritance chain.
|
21
|
+
#
|
22
|
+
# @param attributes [Array<Symbol>] a single or multiple attribute name(s)
|
23
|
+
#
|
24
|
+
# @return [void]
|
25
|
+
#
|
26
|
+
# @since 0.1.0
|
27
|
+
#
|
28
|
+
# @example
|
29
|
+
# require 'hanami/utils/class_attribute'
|
30
|
+
#
|
31
|
+
# class Vehicle
|
32
|
+
# include Hanami::Utils::ClassAttribute
|
33
|
+
# class_attribute :engines, :wheels
|
34
|
+
#
|
35
|
+
# self.engines = 0
|
36
|
+
# self.wheels = 0
|
37
|
+
# end
|
38
|
+
#
|
39
|
+
# class Car < Vehicle
|
40
|
+
# self.engines = 1
|
41
|
+
# self.wheels = 4
|
42
|
+
# end
|
43
|
+
#
|
44
|
+
# class Airplane < Vehicle
|
45
|
+
# self.engines = 4
|
46
|
+
# self.wheels = 16
|
47
|
+
# end
|
48
|
+
#
|
49
|
+
# class SmallAirplane < Airplane
|
50
|
+
# self.engines = 2
|
51
|
+
# self.wheels = 8
|
52
|
+
# end
|
53
|
+
#
|
54
|
+
# Vehicle.engines # => 0
|
55
|
+
# Vehicle.wheels # => 0
|
56
|
+
#
|
57
|
+
# Car.engines # => 1
|
58
|
+
# Car.wheels # => 4
|
59
|
+
#
|
60
|
+
# Airplane.engines # => 4
|
61
|
+
# Airplane.wheels # => 16
|
62
|
+
#
|
63
|
+
# SmallAirplane.engines # => 2
|
64
|
+
# SmallAirplane.wheels # => 8
|
65
|
+
def class_attribute(*attributes)
|
66
|
+
singleton_class.class_eval do
|
67
|
+
attr_accessor *attributes
|
68
|
+
end
|
69
|
+
|
70
|
+
class_attributes.merge(attributes)
|
71
|
+
end
|
72
|
+
|
73
|
+
protected
|
74
|
+
# @see Class#inherited
|
75
|
+
def inherited(subclass)
|
76
|
+
class_attributes.each do |attr|
|
77
|
+
value = send(attr)
|
78
|
+
value = Duplicable.dup(value)
|
79
|
+
subclass.class_attribute attr
|
80
|
+
subclass.send("#{attr}=", value)
|
81
|
+
end
|
82
|
+
|
83
|
+
super
|
84
|
+
end
|
85
|
+
|
86
|
+
private
|
87
|
+
# Class accessor for class attributes.
|
88
|
+
# @private
|
89
|
+
def class_attributes
|
90
|
+
@class_attributes ||= Set.new
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end
|
95
|
+
end
|
@@ -0,0 +1,70 @@
|
|
1
|
+
require 'hanami/utils'
|
2
|
+
|
3
|
+
module Hanami
|
4
|
+
module Utils
|
5
|
+
# Prints a deprecation warning when initialized
|
6
|
+
#
|
7
|
+
# @since 0.3.1
|
8
|
+
class Deprecation
|
9
|
+
# Initialize a deprecation message and prints it to standard error.
|
10
|
+
#
|
11
|
+
# @param message [#to_s] a deprecation message
|
12
|
+
#
|
13
|
+
# @since 0.3.1
|
14
|
+
#
|
15
|
+
# @example Direct usage
|
16
|
+
# require 'hanami/utils/deprecation'
|
17
|
+
#
|
18
|
+
# class Engine
|
19
|
+
# def old_method
|
20
|
+
# Hanami::Utils::Deprecation.new('old_method is deprecated, please use new_method')
|
21
|
+
# new_method
|
22
|
+
# end
|
23
|
+
#
|
24
|
+
# def new_method
|
25
|
+
# puts 'started'
|
26
|
+
# end
|
27
|
+
# end
|
28
|
+
#
|
29
|
+
# Engine.new.old_method
|
30
|
+
# # => old_method is deprecated, please use new_method - called from: test.rb:14:in `<main>'.
|
31
|
+
# # => started
|
32
|
+
#
|
33
|
+
# @example Indirect usage
|
34
|
+
# require 'hanami/utils/deprecation'
|
35
|
+
#
|
36
|
+
# class Engine
|
37
|
+
# def old_method
|
38
|
+
# Hanami::Utils::Deprecation.new('old_method is deprecated, please use new_method')
|
39
|
+
# new_method
|
40
|
+
# end
|
41
|
+
#
|
42
|
+
# def new_method
|
43
|
+
# puts 'started'
|
44
|
+
# end
|
45
|
+
# end
|
46
|
+
#
|
47
|
+
# class Car
|
48
|
+
# def initialize
|
49
|
+
# @engine = Engine.new
|
50
|
+
# end
|
51
|
+
#
|
52
|
+
# def start
|
53
|
+
# @engine.old_method
|
54
|
+
# end
|
55
|
+
# end
|
56
|
+
#
|
57
|
+
# Car.new.start
|
58
|
+
# # => old_method is deprecated, please use new_method - called from: test.rb:20:in `start'.
|
59
|
+
# # => started
|
60
|
+
def initialize(message)
|
61
|
+
::Kernel.warn("#{ message } - called from: #{ caller[caller_index] }.")
|
62
|
+
end
|
63
|
+
|
64
|
+
private
|
65
|
+
def caller_index
|
66
|
+
Utils.jruby? || Utils.rubinius? ? 1 : 2
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
70
|
+
end
|
@@ -0,0 +1,80 @@
|
|
1
|
+
module Hanami
|
2
|
+
module Utils
|
3
|
+
# Safe dup logic
|
4
|
+
#
|
5
|
+
# @since 0.6.0
|
6
|
+
module Duplicable
|
7
|
+
# Duplicates the given value.
|
8
|
+
#
|
9
|
+
# It accepts a block to customize the logic.
|
10
|
+
#
|
11
|
+
# The following types aren't duped:
|
12
|
+
#
|
13
|
+
# * <tt>NilClass</tt>
|
14
|
+
# * <tt>FalseClass</tt>
|
15
|
+
# * <tt>TrueClass</tt>
|
16
|
+
# * <tt>Symbol</tt>
|
17
|
+
# * <tt>Numeric</tt>
|
18
|
+
#
|
19
|
+
# All the other types are duped via <tt>#dup</tt>
|
20
|
+
#
|
21
|
+
# @param value [Object] the value to duplicate
|
22
|
+
# @param blk [Proc] the optional block to customize the logic
|
23
|
+
#
|
24
|
+
# @return [Object] the duped value
|
25
|
+
#
|
26
|
+
# @since 0.6.0
|
27
|
+
#
|
28
|
+
# @example Basic Usage With Types That Can't Be Duped
|
29
|
+
# require 'hanami/utils/duplicable'
|
30
|
+
#
|
31
|
+
# object = 23
|
32
|
+
# puts object.object_id # => 47
|
33
|
+
#
|
34
|
+
# result = Hanami::Utils::Duplicable.dup(object)
|
35
|
+
#
|
36
|
+
# puts result # => 23
|
37
|
+
# puts result.object_id # => 47 - Same object, because numbers can't be duped
|
38
|
+
#
|
39
|
+
# @example Basic Usage With Types That Can Be Duped
|
40
|
+
# require 'hanami/utils/duplicable'
|
41
|
+
#
|
42
|
+
# object = "hello"
|
43
|
+
# puts object.object_id # => 70172661782360
|
44
|
+
#
|
45
|
+
# result = Hanami::Utils::Duplicable.dup(object)
|
46
|
+
#
|
47
|
+
# puts result # => "hello"
|
48
|
+
# puts result.object_id # => 70172671467020 – Different object
|
49
|
+
#
|
50
|
+
# @example Custom Logic
|
51
|
+
# require 'hanami/utils/duplicable'
|
52
|
+
# require 'hanami/utils/hash'
|
53
|
+
#
|
54
|
+
# hash = { a: 1 }
|
55
|
+
# puts hash.object_id # => 70207105061680
|
56
|
+
#
|
57
|
+
# result = Hanami::Utils::Duplicable.dup(hash) do |value|
|
58
|
+
# case value
|
59
|
+
# when Hanami::Utils::Hash
|
60
|
+
# value.deep_dup
|
61
|
+
# when ::Hash
|
62
|
+
# Hanami::Utils::Hash.new(value).deep_dup.to_h
|
63
|
+
# end
|
64
|
+
# end
|
65
|
+
#
|
66
|
+
# puts result # => "{:a=>1}"
|
67
|
+
# puts result.object_id # => 70207105185500 – Different object
|
68
|
+
def self.dup(value, &blk)
|
69
|
+
case value
|
70
|
+
when NilClass, FalseClass, TrueClass, Symbol, Numeric
|
71
|
+
value
|
72
|
+
when v = blk && blk.call(value)
|
73
|
+
v
|
74
|
+
else
|
75
|
+
value.dup
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|
80
|
+
end
|
@@ -0,0 +1,577 @@
|
|
1
|
+
module Hanami
|
2
|
+
module Utils
|
3
|
+
# HTML escape utilities
|
4
|
+
#
|
5
|
+
# Based on OWASP research and OWASP ESAPI code
|
6
|
+
#
|
7
|
+
# @since 0.4.0
|
8
|
+
#
|
9
|
+
# @see https://www.owasp.org
|
10
|
+
# @see https://www.owasp.org/index.php/Cross-site_Scripting_%28XSS%29
|
11
|
+
# @see https://www.owasp.org/index.php/XSS_(Cross_Site_Scripting)_Prevention_Cheat_Sheet
|
12
|
+
# @see https://www.owasp.org/index.php/ESAPI
|
13
|
+
# @see https://github.com/ESAPI/esapi-java-legacy
|
14
|
+
module Escape
|
15
|
+
# Hex base for base 10 integer conversion
|
16
|
+
#
|
17
|
+
# @since 0.4.0
|
18
|
+
# @api private
|
19
|
+
#
|
20
|
+
# @see http://www.ruby-doc.org/core/Fixnum.html#method-i-to_s
|
21
|
+
HEX_BASE = 16
|
22
|
+
|
23
|
+
# Limit for non printable chars
|
24
|
+
#
|
25
|
+
# @since 0.4.0
|
26
|
+
# @api private
|
27
|
+
LOW_HEX_CODE_LIMIT = 0xff
|
28
|
+
|
29
|
+
# Replacement hex for non printable characters
|
30
|
+
#
|
31
|
+
# @since 0.4.0
|
32
|
+
# @api private
|
33
|
+
REPLACEMENT_HEX = "fffd".freeze
|
34
|
+
|
35
|
+
# Low hex codes lookup table
|
36
|
+
#
|
37
|
+
# @since 0.4.0
|
38
|
+
# @api private
|
39
|
+
HEX_CODES = (0..255).each_with_object({}) do |c, codes|
|
40
|
+
if (c >= 0x30 && c <= 0x39) || (c >= 0x41 && c <= 0x5A) || (c >= 0x61 && c <= 0x7A)
|
41
|
+
codes[c] = nil
|
42
|
+
else
|
43
|
+
codes[c] = c.to_s(HEX_BASE)
|
44
|
+
end
|
45
|
+
end.freeze
|
46
|
+
|
47
|
+
# Non printable chars
|
48
|
+
#
|
49
|
+
# This is a Hash instead of a Set, to make lookup faster.
|
50
|
+
#
|
51
|
+
# @since 0.4.0
|
52
|
+
# @api private
|
53
|
+
#
|
54
|
+
# @see https://gist.github.com/jodosha/ac5dd54416de744b9600
|
55
|
+
NON_PRINTABLE_CHARS = {
|
56
|
+
0x0 => true, 0x1 => true, 0x2 => true, 0x3 => true, 0x4 => true,
|
57
|
+
0x5 => true, 0x6 => true, 0x7 => true, 0x8 => true, 0x11 => true,
|
58
|
+
0x12 => true, 0x14 => true, 0x15 => true, 0x16 => true, 0x17 => true,
|
59
|
+
0x18 => true, 0x19 => true, 0x1a => true, 0x1b => true, 0x1c => true,
|
60
|
+
0x1d => true, 0x1e => true, 0x1f => true, 0x7f => true, 0x80 => true,
|
61
|
+
0x81 => true, 0x82 => true, 0x83 => true, 0x84 => true, 0x85 => true,
|
62
|
+
0x86 => true, 0x87 => true, 0x88 => true, 0x89 => true, 0x8a => true,
|
63
|
+
0x8b => true, 0x8c => true, 0x8d => true, 0x8e => true, 0x8f => true,
|
64
|
+
0x90 => true, 0x91 => true, 0x92 => true, 0x93 => true, 0x94 => true,
|
65
|
+
0x95 => true, 0x96 => true, 0x97 => true, 0x98 => true, 0x99 => true,
|
66
|
+
0x9a => true, 0x9b => true, 0x9c => true, 0x9d => true, 0x9e => true,
|
67
|
+
0x9f => true
|
68
|
+
}.freeze
|
69
|
+
|
70
|
+
# Lookup table for HTML escape
|
71
|
+
#
|
72
|
+
# @since 0.4.0
|
73
|
+
# @api private
|
74
|
+
#
|
75
|
+
# @see Hanami::Utils::Escape.html
|
76
|
+
HTML_CHARS = {
|
77
|
+
'&' => '&',
|
78
|
+
'<' => '<',
|
79
|
+
'>' => '>',
|
80
|
+
'"' => '"',
|
81
|
+
"'" => ''',
|
82
|
+
'/' => '/'
|
83
|
+
}.freeze
|
84
|
+
|
85
|
+
# Lookup table for safe chars for HTML attributes.
|
86
|
+
#
|
87
|
+
# This is a Hash instead of a Set, to make lookup faster.
|
88
|
+
#
|
89
|
+
# @since 0.4.0
|
90
|
+
# @api private
|
91
|
+
#
|
92
|
+
# @see Lookup::Utils::Escape.html_attribute
|
93
|
+
# @see https://gist.github.com/jodosha/ac5dd54416de744b9600
|
94
|
+
HTML_ATTRIBUTE_SAFE_CHARS = {
|
95
|
+
',' => true, '.' => true, '-' => true, '_' => true
|
96
|
+
}.freeze
|
97
|
+
|
98
|
+
# Lookup table for HTML attribute escape
|
99
|
+
#
|
100
|
+
# @since 0.4.0
|
101
|
+
# @api private
|
102
|
+
#
|
103
|
+
# @see Hanami::Utils::Escape.html_attribute
|
104
|
+
HTML_ENTITIES = {
|
105
|
+
34 => 'quot', # quotation mark
|
106
|
+
38 => 'amp', # ampersand
|
107
|
+
60 => 'lt', # less-than sign
|
108
|
+
62 => 'gt', # greater-than sign
|
109
|
+
160 => 'nbsp', # no-break space
|
110
|
+
161 => 'iexcl', # inverted exclamation mark
|
111
|
+
162 => 'cent', # cent sign
|
112
|
+
163 => 'pound', # pound sign
|
113
|
+
164 => 'curren', # currency sign
|
114
|
+
165 => 'yen', # yen sign
|
115
|
+
166 => 'brvbar', # broken bar
|
116
|
+
167 => 'sect', # section sign
|
117
|
+
168 => 'uml', # diaeresis
|
118
|
+
169 => 'copy', # copyright sign
|
119
|
+
170 => 'ordf', # feminine ordinal indicator
|
120
|
+
171 => 'laquo', # left-pointing double angle quotation mark
|
121
|
+
172 => 'not', # not sign
|
122
|
+
173 => 'shy', # soft hyphen
|
123
|
+
174 => 'reg', # registered sign
|
124
|
+
175 => 'macr', # macron
|
125
|
+
176 => 'deg', # degree sign
|
126
|
+
177 => 'plusmn', # plus-minus sign
|
127
|
+
178 => 'sup2', # superscript two
|
128
|
+
179 => 'sup3', # superscript three
|
129
|
+
180 => 'acute', # acute accent
|
130
|
+
181 => 'micro', # micro sign
|
131
|
+
182 => 'para', # pilcrow sign
|
132
|
+
183 => 'middot', # middle dot
|
133
|
+
184 => 'cedil', # cedilla
|
134
|
+
185 => 'sup1', # superscript one
|
135
|
+
186 => 'ordm', # masculine ordinal indicator
|
136
|
+
187 => 'raquo', # right-pointing double angle quotation mark
|
137
|
+
188 => 'frac14', # vulgar fraction one quarter
|
138
|
+
189 => 'frac12', # vulgar fraction one half
|
139
|
+
190 => 'frac34', # vulgar fraction three quarters
|
140
|
+
191 => 'iquest', # inverted question mark
|
141
|
+
192 => 'Agrave', # Latin capital letter a with grave
|
142
|
+
193 => 'Aacute', # Latin capital letter a with acute
|
143
|
+
194 => 'Acirc', # Latin capital letter a with circumflex
|
144
|
+
195 => 'Atilde', # Latin capital letter a with tilde
|
145
|
+
196 => 'Auml', # Latin capital letter a with diaeresis
|
146
|
+
197 => 'Aring', # Latin capital letter a with ring above
|
147
|
+
198 => 'AElig', # Latin capital letter ae
|
148
|
+
199 => 'Ccedil', # Latin capital letter c with cedilla
|
149
|
+
200 => 'Egrave', # Latin capital letter e with grave
|
150
|
+
201 => 'Eacute', # Latin capital letter e with acute
|
151
|
+
202 => 'Ecirc', # Latin capital letter e with circumflex
|
152
|
+
203 => 'Euml', # Latin capital letter e with diaeresis
|
153
|
+
204 => 'Igrave', # Latin capital letter i with grave
|
154
|
+
205 => 'Iacute', # Latin capital letter i with acute
|
155
|
+
206 => 'Icirc', # Latin capital letter i with circumflex
|
156
|
+
207 => 'Iuml', # Latin capital letter i with diaeresis
|
157
|
+
208 => 'ETH', # Latin capital letter eth
|
158
|
+
209 => 'Ntilde', # Latin capital letter n with tilde
|
159
|
+
210 => 'Ograve', # Latin capital letter o with grave
|
160
|
+
211 => 'Oacute', # Latin capital letter o with acute
|
161
|
+
212 => 'Ocirc', # Latin capital letter o with circumflex
|
162
|
+
213 => 'Otilde', # Latin capital letter o with tilde
|
163
|
+
214 => 'Ouml', # Latin capital letter o with diaeresis
|
164
|
+
215 => 'times', # multiplication sign
|
165
|
+
216 => 'Oslash', # Latin capital letter o with stroke
|
166
|
+
217 => 'Ugrave', # Latin capital letter u with grave
|
167
|
+
218 => 'Uacute', # Latin capital letter u with acute
|
168
|
+
219 => 'Ucirc', # Latin capital letter u with circumflex
|
169
|
+
220 => 'Uuml', # Latin capital letter u with diaeresis
|
170
|
+
221 => 'Yacute', # Latin capital letter y with acute
|
171
|
+
222 => 'THORN', # Latin capital letter thorn
|
172
|
+
223 => 'szlig', # Latin small letter sharp sXCOMMAX German Eszett
|
173
|
+
224 => 'agrave', # Latin small letter a with grave
|
174
|
+
225 => 'aacute', # Latin small letter a with acute
|
175
|
+
226 => 'acirc', # Latin small letter a with circumflex
|
176
|
+
227 => 'atilde', # Latin small letter a with tilde
|
177
|
+
228 => 'auml', # Latin small letter a with diaeresis
|
178
|
+
229 => 'aring', # Latin small letter a with ring above
|
179
|
+
230 => 'aelig', # Latin lowercase ligature ae
|
180
|
+
231 => 'ccedil', # Latin small letter c with cedilla
|
181
|
+
232 => 'egrave', # Latin small letter e with grave
|
182
|
+
233 => 'eacute', # Latin small letter e with acute
|
183
|
+
234 => 'ecirc', # Latin small letter e with circumflex
|
184
|
+
235 => 'euml', # Latin small letter e with diaeresis
|
185
|
+
236 => 'igrave', # Latin small letter i with grave
|
186
|
+
237 => 'iacute', # Latin small letter i with acute
|
187
|
+
238 => 'icirc', # Latin small letter i with circumflex
|
188
|
+
239 => 'iuml', # Latin small letter i with diaeresis
|
189
|
+
240 => 'eth', # Latin small letter eth
|
190
|
+
241 => 'ntilde', # Latin small letter n with tilde
|
191
|
+
242 => 'ograve', # Latin small letter o with grave
|
192
|
+
243 => 'oacute', # Latin small letter o with acute
|
193
|
+
244 => 'ocirc', # Latin small letter o with circumflex
|
194
|
+
245 => 'otilde', # Latin small letter o with tilde
|
195
|
+
246 => 'ouml', # Latin small letter o with diaeresis
|
196
|
+
247 => 'divide', # division sign
|
197
|
+
248 => 'oslash', # Latin small letter o with stroke
|
198
|
+
249 => 'ugrave', # Latin small letter u with grave
|
199
|
+
250 => 'uacute', # Latin small letter u with acute
|
200
|
+
251 => 'ucirc', # Latin small letter u with circumflex
|
201
|
+
252 => 'uuml', # Latin small letter u with diaeresis
|
202
|
+
253 => 'yacute', # Latin small letter y with acute
|
203
|
+
254 => 'thorn', # Latin small letter thorn
|
204
|
+
255 => 'yuml', # Latin small letter y with diaeresis
|
205
|
+
338 => 'OElig', # Latin capital ligature oe
|
206
|
+
339 => 'oelig', # Latin small ligature oe
|
207
|
+
352 => 'Scaron', # Latin capital letter s with caron
|
208
|
+
353 => 'scaron', # Latin small letter s with caron
|
209
|
+
376 => 'Yuml', # Latin capital letter y with diaeresis
|
210
|
+
402 => 'fnof', # Latin small letter f with hook
|
211
|
+
710 => 'circ', # modifier letter circumflex accent
|
212
|
+
732 => 'tilde', # small tilde
|
213
|
+
913 => 'Alpha', # Greek capital letter alpha
|
214
|
+
914 => 'Beta', # Greek capital letter beta
|
215
|
+
915 => 'Gamma', # Greek capital letter gamma
|
216
|
+
916 => 'Delta', # Greek capital letter delta
|
217
|
+
917 => 'Epsilon', # Greek capital letter epsilon
|
218
|
+
918 => 'Zeta', # Greek capital letter zeta
|
219
|
+
919 => 'Eta', # Greek capital letter eta
|
220
|
+
920 => 'Theta', # Greek capital letter theta
|
221
|
+
921 => 'Iota', # Greek capital letter iota
|
222
|
+
922 => 'Kappa', # Greek capital letter kappa
|
223
|
+
923 => 'Lambda', # Greek capital letter lambda
|
224
|
+
924 => 'Mu', # Greek capital letter mu
|
225
|
+
925 => 'Nu', # Greek capital letter nu
|
226
|
+
926 => 'Xi', # Greek capital letter xi
|
227
|
+
927 => 'Omicron', # Greek capital letter omicron
|
228
|
+
928 => 'Pi', # Greek capital letter pi
|
229
|
+
929 => 'Rho', # Greek capital letter rho
|
230
|
+
931 => 'Sigma', # Greek capital letter sigma
|
231
|
+
932 => 'Tau', # Greek capital letter tau
|
232
|
+
933 => 'Upsilon', # Greek capital letter upsilon
|
233
|
+
934 => 'Phi', # Greek capital letter phi
|
234
|
+
935 => 'Chi', # Greek capital letter chi
|
235
|
+
936 => 'Psi', # Greek capital letter psi
|
236
|
+
937 => 'Omega', # Greek capital letter omega
|
237
|
+
945 => 'alpha', # Greek small letter alpha
|
238
|
+
946 => 'beta', # Greek small letter beta
|
239
|
+
947 => 'gamma', # Greek small letter gamma
|
240
|
+
948 => 'delta', # Greek small letter delta
|
241
|
+
949 => 'epsilon', # Greek small letter epsilon
|
242
|
+
950 => 'zeta', # Greek small letter zeta
|
243
|
+
951 => 'eta', # Greek small letter eta
|
244
|
+
952 => 'theta', # Greek small letter theta
|
245
|
+
953 => 'iota', # Greek small letter iota
|
246
|
+
954 => 'kappa', # Greek small letter kappa
|
247
|
+
955 => 'lambda', # Greek small letter lambda
|
248
|
+
956 => 'mu', # Greek small letter mu
|
249
|
+
957 => 'nu', # Greek small letter nu
|
250
|
+
958 => 'xi', # Greek small letter xi
|
251
|
+
959 => 'omicron', # Greek small letter omicron
|
252
|
+
960 => 'pi', # Greek small letter pi
|
253
|
+
961 => 'rho', # Greek small letter rho
|
254
|
+
962 => 'sigmaf', # Greek small letter final sigma
|
255
|
+
963 => 'sigma', # Greek small letter sigma
|
256
|
+
964 => 'tau', # Greek small letter tau
|
257
|
+
965 => 'upsilon', # Greek small letter upsilon
|
258
|
+
966 => 'phi', # Greek small letter phi
|
259
|
+
967 => 'chi', # Greek small letter chi
|
260
|
+
968 => 'psi', # Greek small letter psi
|
261
|
+
969 => 'omega', # Greek small letter omega
|
262
|
+
977 => 'thetasym', # Greek theta symbol
|
263
|
+
978 => 'upsih', # Greek upsilon with hook symbol
|
264
|
+
982 => 'piv', # Greek pi symbol
|
265
|
+
8194 => 'ensp', # en space
|
266
|
+
8195 => 'emsp', # em space
|
267
|
+
8201 => 'thinsp', # thin space
|
268
|
+
8204 => 'zwnj', # zero width non-joiner
|
269
|
+
8205 => 'zwj', # zero width joiner
|
270
|
+
8206 => 'lrm', # left-to-right mark
|
271
|
+
8207 => 'rlm', # right-to-left mark
|
272
|
+
8211 => 'ndash', # en dash
|
273
|
+
8212 => 'mdash', # em dash
|
274
|
+
8216 => 'lsquo', # left single quotation mark
|
275
|
+
8217 => 'rsquo', # right single quotation mark
|
276
|
+
8218 => 'sbquo', # single low-9 quotation mark
|
277
|
+
8220 => 'ldquo', # left double quotation mark
|
278
|
+
8221 => 'rdquo', # right double quotation mark
|
279
|
+
8222 => 'bdquo', # double low-9 quotation mark
|
280
|
+
8224 => 'dagger', # dagger
|
281
|
+
8225 => 'Dagger', # double dagger
|
282
|
+
8226 => 'bull', # bullet
|
283
|
+
8230 => 'hellip', # horizontal ellipsis
|
284
|
+
8240 => 'permil', # per mille sign
|
285
|
+
8242 => 'prime', # prime
|
286
|
+
8243 => 'Prime', # double prime
|
287
|
+
8249 => 'lsaquo', # single left-pointing angle quotation mark
|
288
|
+
8250 => 'rsaquo', # single right-pointing angle quotation mark
|
289
|
+
8254 => 'oline', # overline
|
290
|
+
8260 => 'frasl', # fraction slash
|
291
|
+
8364 => 'euro', # euro sign
|
292
|
+
8465 => 'image', # black-letter capital i
|
293
|
+
8472 => 'weierp', # script capital pXCOMMAX Weierstrass p
|
294
|
+
8476 => 'real', # black-letter capital r
|
295
|
+
8482 => 'trade', # trademark sign
|
296
|
+
8501 => 'alefsym', # alef symbol
|
297
|
+
8592 => 'larr', # leftwards arrow
|
298
|
+
8593 => 'uarr', # upwards arrow
|
299
|
+
8594 => 'rarr', # rightwards arrow
|
300
|
+
8595 => 'darr', # downwards arrow
|
301
|
+
8596 => 'harr', # left right arrow
|
302
|
+
8629 => 'crarr', # downwards arrow with corner leftwards
|
303
|
+
8656 => 'lArr', # leftwards double arrow
|
304
|
+
8657 => 'uArr', # upwards double arrow
|
305
|
+
8658 => 'rArr', # rightwards double arrow
|
306
|
+
8659 => 'dArr', # downwards double arrow
|
307
|
+
8660 => 'hArr', # left right double arrow
|
308
|
+
8704 => 'forall', # for all
|
309
|
+
8706 => 'part', # partial differential
|
310
|
+
8707 => 'exist', # there exists
|
311
|
+
8709 => 'empty', # empty set
|
312
|
+
8711 => 'nabla', # nabla
|
313
|
+
8712 => 'isin', # element of
|
314
|
+
8713 => 'notin', # not an element of
|
315
|
+
8715 => 'ni', # contains as member
|
316
|
+
8719 => 'prod', # n-ary product
|
317
|
+
8721 => 'sum', # n-ary summation
|
318
|
+
8722 => 'minus', # minus sign
|
319
|
+
8727 => 'lowast', # asterisk operator
|
320
|
+
8730 => 'radic', # square root
|
321
|
+
8733 => 'prop', # proportional to
|
322
|
+
8734 => 'infin', # infinity
|
323
|
+
8736 => 'ang', # angle
|
324
|
+
8743 => 'and', # logical and
|
325
|
+
8744 => 'or', # logical or
|
326
|
+
8745 => 'cap', # intersection
|
327
|
+
8746 => 'cup', # union
|
328
|
+
8747 => 'int', # integral
|
329
|
+
8756 => 'there4', # therefore
|
330
|
+
8764 => 'sim', # tilde operator
|
331
|
+
8773 => 'cong', # congruent to
|
332
|
+
8776 => 'asymp', # almost equal to
|
333
|
+
8800 => 'ne', # not equal to
|
334
|
+
8801 => 'equiv', # identical toXCOMMAX equivalent to
|
335
|
+
8804 => 'le', # less-than or equal to
|
336
|
+
8805 => 'ge', # greater-than or equal to
|
337
|
+
8834 => 'sub', # subset of
|
338
|
+
8835 => 'sup', # superset of
|
339
|
+
8836 => 'nsub', # not a subset of
|
340
|
+
8838 => 'sube', # subset of or equal to
|
341
|
+
8839 => 'supe', # superset of or equal to
|
342
|
+
8853 => 'oplus', # circled plus
|
343
|
+
8855 => 'otimes', # circled times
|
344
|
+
8869 => 'perp', # up tack
|
345
|
+
8901 => 'sdot', # dot operator
|
346
|
+
8968 => 'lceil', # left ceiling
|
347
|
+
8969 => 'rceil', # right ceiling
|
348
|
+
8970 => 'lfloor', # left floor
|
349
|
+
8971 => 'rfloor', # right floor
|
350
|
+
9001 => 'lang', # left-pointing angle bracket
|
351
|
+
9002 => 'rang', # right-pointing angle bracket
|
352
|
+
9674 => 'loz', # lozenge
|
353
|
+
9824 => 'spades', # black spade suit
|
354
|
+
9827 => 'clubs', # black club suit
|
355
|
+
9829 => 'hearts', # black heart suit
|
356
|
+
9830 => 'diams', # black diamond suit
|
357
|
+
}.freeze
|
358
|
+
|
359
|
+
# Allowed URL schemes
|
360
|
+
#
|
361
|
+
# @since 0.4.0
|
362
|
+
# @api private
|
363
|
+
#
|
364
|
+
# @see Hanami::Utils::Escape.url
|
365
|
+
DEFAULT_URL_SCHEMES = ['http', 'https', 'mailto'].freeze
|
366
|
+
|
367
|
+
# The output of an escape.
|
368
|
+
#
|
369
|
+
# It's marked with this special class for two reasons:
|
370
|
+
#
|
371
|
+
# * Don't double escape the same string (this is for `Hanami::Helpers` compatibility)
|
372
|
+
# * Leave open the possibility to developers to mark a string as safe with an higher API (eg. `#raw` in `Hanami::View` or `Hanami::Helpers`)
|
373
|
+
#
|
374
|
+
# @since 0.4.0
|
375
|
+
# @api private
|
376
|
+
class SafeString < ::String
|
377
|
+
# @return [SafeString] the duped string
|
378
|
+
#
|
379
|
+
# @since 0.4.0
|
380
|
+
# @api private
|
381
|
+
#
|
382
|
+
# @see http://www.ruby-doc.org/core/String.html#method-i-to_s
|
383
|
+
def to_s
|
384
|
+
dup
|
385
|
+
end
|
386
|
+
|
387
|
+
# Encode the string the given encoding
|
388
|
+
#
|
389
|
+
# @return [SafeString] an encoded SafeString
|
390
|
+
#
|
391
|
+
# @since 0.4.0
|
392
|
+
# @api private
|
393
|
+
#
|
394
|
+
# @see http://www.ruby-doc.org/core/String.html#method-i-encode
|
395
|
+
def encode(*args)
|
396
|
+
self.class.new super
|
397
|
+
end
|
398
|
+
end
|
399
|
+
|
400
|
+
# Escape HTML contents
|
401
|
+
#
|
402
|
+
# This MUST be used only for tag contents.
|
403
|
+
# Please use `html_attribute` for escaping HTML attributes.
|
404
|
+
#
|
405
|
+
# @param input [String] the input
|
406
|
+
#
|
407
|
+
# @return [String] the escaped string
|
408
|
+
#
|
409
|
+
# @since 0.4.0
|
410
|
+
#
|
411
|
+
# @see https://www.owasp.org/index.php/XSS_%28Cross_Site_Scripting%29_Prevention_Cheat_Sheet OWASP XSS Cheat Sheet Rule #1
|
412
|
+
#
|
413
|
+
# @example Good practice
|
414
|
+
# <div><%= Hanami::Utils::Escape.html('<script>alert(1);</script>') %></div>
|
415
|
+
# <div><script>alert(1);</script></div>
|
416
|
+
#
|
417
|
+
# @example Bad practice
|
418
|
+
# # WRONG Use Escape.html_attribute
|
419
|
+
# <a title="<%= Hanami::Utils::Escape.html('...') %>">link</a>
|
420
|
+
def self.html(input)
|
421
|
+
input = encode(input)
|
422
|
+
return input if input.is_a?(SafeString)
|
423
|
+
|
424
|
+
result = SafeString.new
|
425
|
+
|
426
|
+
input.chars do |chr|
|
427
|
+
result << HTML_CHARS.fetch(chr, chr)
|
428
|
+
end
|
429
|
+
|
430
|
+
result
|
431
|
+
end
|
432
|
+
|
433
|
+
# Escape HTML attributes
|
434
|
+
#
|
435
|
+
# This can be used both for HTML attributes and contents.
|
436
|
+
# Please note that this is more computational expensive.
|
437
|
+
# If you need to escape only HTML contents, please use `.html`.
|
438
|
+
#
|
439
|
+
# @param input [String] the input
|
440
|
+
#
|
441
|
+
# @return [String] the escaped string
|
442
|
+
#
|
443
|
+
# @since 0.4.0
|
444
|
+
#
|
445
|
+
# @see https://www.owasp.org/index.php/XSS_%28Cross_Site_Scripting%29_Prevention_Cheat_Sheet OWASP XSS Cheat Sheet Rule #2
|
446
|
+
#
|
447
|
+
# @example Good practice
|
448
|
+
# <a title="<%= Hanami::Utils::Escape.html_attribute('...') %>">link</a>
|
449
|
+
#
|
450
|
+
# @example Good but expensive practice
|
451
|
+
# # Alternatively you can use Escape.html
|
452
|
+
# <p><%= Hanami::Utils::Escape.html_attribute('...') %></p>
|
453
|
+
def self.html_attribute(input)
|
454
|
+
input = encode(input)
|
455
|
+
return input if input.is_a?(SafeString)
|
456
|
+
|
457
|
+
result = SafeString.new
|
458
|
+
|
459
|
+
input.chars do |chr|
|
460
|
+
result << encode_char(chr, HTML_ATTRIBUTE_SAFE_CHARS)
|
461
|
+
end
|
462
|
+
|
463
|
+
result
|
464
|
+
end
|
465
|
+
|
466
|
+
# Escape URL for HTML attributes (href, src, etc..).
|
467
|
+
#
|
468
|
+
# It extracts from the given input the first valid URL that matches the
|
469
|
+
# whitelisted schemes (default: http, https and mailto).
|
470
|
+
#
|
471
|
+
# It's possible to pass a second optional argument to specify different
|
472
|
+
# schemes.
|
473
|
+
#
|
474
|
+
# @param input [String] the input
|
475
|
+
# @param schemes [Array<String>] an array of whitelisted schemes
|
476
|
+
#
|
477
|
+
# @return [String] the escaped string
|
478
|
+
#
|
479
|
+
# @since 0.4.0
|
480
|
+
#
|
481
|
+
# @see Hanami::Utils::Escape::DEFAULT_URL_SCHEMES
|
482
|
+
# @see http://www.ruby-doc.org/stdlib/libdoc/uri/rdoc/URI.html#method-c-extract
|
483
|
+
#
|
484
|
+
# @example Basic usage
|
485
|
+
# <%
|
486
|
+
# good_input = "http://hanamirb.org"
|
487
|
+
# evil_input = "javascript:alert('xss')"
|
488
|
+
#
|
489
|
+
# escaped_good_input = Hanami::Utils::Escape.url(good_input) # => "http://hanamirb.org"
|
490
|
+
# escaped_evil_input = Hanami::Utils::Escape.url(evil_input) # => ""
|
491
|
+
# %>
|
492
|
+
#
|
493
|
+
# <a href="<%= escaped_good_input %>">personal website</a>
|
494
|
+
# <a href="<%= escaped_evil_input %>">personal website</a>
|
495
|
+
#
|
496
|
+
# @example Custom scheme
|
497
|
+
# <%
|
498
|
+
# schemes = ['ftp', 'ftps']
|
499
|
+
#
|
500
|
+
# accepted = "ftps://ftp.example.org"
|
501
|
+
# rejected = "http://www.example.org"
|
502
|
+
#
|
503
|
+
# escaped_accepted = Hanami::Utils::Escape.url(accepted) # => "ftps://ftp.example.org"
|
504
|
+
# escaped_rejected = Hanami::Utils::Escape.url(rejected) # => ""
|
505
|
+
# %>
|
506
|
+
#
|
507
|
+
# <a href="<%= escaped_accepted %>">FTP</a>
|
508
|
+
# <a href="<%= escaped_rejected %>">FTP</a>
|
509
|
+
def self.url(input, schemes = DEFAULT_URL_SCHEMES)
|
510
|
+
input = encode(input)
|
511
|
+
return input if input.is_a?(SafeString)
|
512
|
+
|
513
|
+
SafeString.new(
|
514
|
+
URI.extract(
|
515
|
+
URI.decode(input),
|
516
|
+
schemes
|
517
|
+
).first.to_s
|
518
|
+
)
|
519
|
+
end
|
520
|
+
|
521
|
+
private
|
522
|
+
# Encode the given string into UTF-8
|
523
|
+
#
|
524
|
+
# @param input [String] the input
|
525
|
+
#
|
526
|
+
# @return [String] an UTF-8 encoded string
|
527
|
+
#
|
528
|
+
# @since 0.4.0
|
529
|
+
# @api private
|
530
|
+
def self.encode(input)
|
531
|
+
return '' if input.nil?
|
532
|
+
input.encode(Encoding::UTF_8)
|
533
|
+
rescue Encoding::UndefinedConversionError
|
534
|
+
input.dup.force_encoding(Encoding::UTF_8)
|
535
|
+
end
|
536
|
+
|
537
|
+
# Encode the given UTF-8 char.
|
538
|
+
#
|
539
|
+
# @param char [String] an UTF-8 char
|
540
|
+
# @param safe_chars [Hash] a table of safe chars
|
541
|
+
#
|
542
|
+
# @return [String] an HTML encoded string
|
543
|
+
#
|
544
|
+
# @since 0.4.0
|
545
|
+
# @api private
|
546
|
+
def self.encode_char(char, safe_chars = {})
|
547
|
+
return char if safe_chars[char]
|
548
|
+
|
549
|
+
code = char.ord
|
550
|
+
hex = hex_for_non_alphanumeric_code(code)
|
551
|
+
return char if hex.nil?
|
552
|
+
|
553
|
+
if NON_PRINTABLE_CHARS[code]
|
554
|
+
hex = REPLACEMENT_HEX
|
555
|
+
end
|
556
|
+
|
557
|
+
if entity = HTML_ENTITIES[code]
|
558
|
+
"&#{ entity };"
|
559
|
+
else
|
560
|
+
"&#x#{ hex };"
|
561
|
+
end
|
562
|
+
end
|
563
|
+
|
564
|
+
# Transforms the given char code
|
565
|
+
#
|
566
|
+
# @since 0.4.0
|
567
|
+
# @api private
|
568
|
+
def self.hex_for_non_alphanumeric_code(input)
|
569
|
+
if input < LOW_HEX_CODE_LIMIT
|
570
|
+
HEX_CODES[input]
|
571
|
+
else
|
572
|
+
input.to_s(HEX_BASE)
|
573
|
+
end
|
574
|
+
end
|
575
|
+
end
|
576
|
+
end
|
577
|
+
end
|