ruby_auth_metamask 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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: '082c3d67f770e78978eda75d5d4e6094a00264a258a8fc882447bd890d0fe8f9'
4
+ data.tar.gz: fc4f10499d4006701e0ee14e6707ed010df7dc45e6942df2859d367dc55d7869
5
+ SHA512:
6
+ metadata.gz: 1f06a0fa4cf7bc7a62c5449bdc04e97ad4713c61db89c6ece9dcf924dd63c3d22990664110171a9c8e316622d82487ff6cefe377a8dc69aa236a3cde995a0872
7
+ data.tar.gz: b32bb59e75509d5ed13bfdd30de498915c636c2289eb8af06f5223633ec9cd940d35975de53c0a623cbc4bb55a7857c18c06c0c9d5499a1817493514f05d340b
data/MIT-LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright Ethan Zhang
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,28 @@
1
+ # RubyAuthMetamask
2
+ Short description and motivation.
3
+
4
+ ## Usage
5
+ How to use my plugin.
6
+
7
+ ## Installation
8
+ Add this line to your application's Gemfile:
9
+
10
+ ```ruby
11
+ gem "ruby_auth_metamask"
12
+ ```
13
+
14
+ And then execute:
15
+ ```bash
16
+ $ bundle
17
+ ```
18
+
19
+ Or install it yourself as:
20
+ ```bash
21
+ $ gem install ruby_auth_metamask
22
+ ```
23
+
24
+ ## Contributing
25
+ Contribution directions go here.
26
+
27
+ ## License
28
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,8 @@
1
+ require "bundler/setup"
2
+
3
+ APP_RAKEFILE = File.expand_path("test/dummy/Rakefile", __dir__)
4
+ load "rails/tasks/engine.rake"
5
+
6
+ load "rails/tasks/statistics.rake"
7
+
8
+ require "bundler/gem_tasks"
@@ -0,0 +1 @@
1
+ //= link_directory ../stylesheets/ruby_auth_metamask .css
@@ -0,0 +1,15 @@
1
+ /*
2
+ * This is a manifest file that'll be compiled into application.css, which will include all the files
3
+ * listed below.
4
+ *
5
+ * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets,
6
+ * or any plugin's vendor/assets/stylesheets directory can be referenced here using a relative path.
7
+ *
8
+ * You're free to add application-wide styles to this file and they'll appear at the bottom of the
9
+ * compiled file so the styles you add here take precedence over styles defined in any other CSS/SCSS
10
+ * files in this directory. Styles in this file should be added after the last require_* statement.
11
+ * It is generally better to create a new file per style scope.
12
+ *
13
+ *= require_tree .
14
+ *= require_self
15
+ */
@@ -0,0 +1,4 @@
1
+ module RubyAuthMetamask
2
+ class ApplicationController < ActionController::Base
3
+ end
4
+ end
@@ -0,0 +1,141 @@
1
+ require 'ecdsa'
2
+ require 'digest/keccak'
3
+
4
+ module RubyAuthMetamask
5
+ # a simple wrapper to carry the v value of signture
6
+ class MySignature
7
+ attr_reader :sig_obj, :v
8
+
9
+ def initialize(sig_obj, v_value)
10
+ @sig_obj = sig_obj
11
+ @v = v_value
12
+ end
13
+ end
14
+
15
+ class UsersController < ApplicationController
16
+ def signin
17
+ @message = "ruby_auth_metamask:#{SecureRandom.hex}"
18
+ session[:message] = @message
19
+ end
20
+
21
+ def verify
22
+ address = sanitize_and_return_address
23
+ my_signature = sanitize_and_return_my_signature
24
+ if address.nil? || my_signature.nil?
25
+ redirect_to main_app.root_path, notice: 'Address or signature is invalid'
26
+ return
27
+ end
28
+
29
+ hash = metamask_digest(session[:message])
30
+ if hash.nil?
31
+ redirect_to main_app.root_path, notice: 'User authentication failed'
32
+ return
33
+ end
34
+
35
+ public_key = recover_public_key(hash, my_signature)
36
+ if public_key.nil?
37
+ redirect_to main_app.root_path, notice: 'User authentication failed'
38
+ return
39
+ end
40
+
41
+ unless verify_public_key_and_address(public_key, address)
42
+ redirect_to main_app.root_path, notice: 'User address does not match public key'
43
+ return
44
+ end
45
+
46
+ valid = ECDSA.valid_signature?(public_key, hash, my_signature.sig_obj) rescue false
47
+ if valid
48
+ redirect_to main_app.root_path, notice: 'User authenticated successfully'
49
+ else
50
+ redirect_to main_app.root_path, notice: 'User authentication failed'
51
+ end
52
+ end
53
+
54
+ private
55
+
56
+ def sanitize_and_return_address
57
+ return nil if params[:address].nil?
58
+
59
+ params[:address].downcase
60
+ end
61
+
62
+ def sanitize_and_return_my_signature
63
+ signature = params[:signature] # 0x0123456789abcdef
64
+ if signature.nil?
65
+ return nil
66
+ elsif signature.size < 3
67
+ return nil
68
+ else
69
+ signature = signature[2..]
70
+ end
71
+
72
+ my_signature_from_hex(signature)
73
+ end
74
+
75
+ def public_key_to_address(public_key)
76
+ # public_key_string_short = ECDSA::Format::PointOctetString.encode(public_key, compression: true)
77
+ # pub_key_hex = public_key_string_short.unpack1("H*")
78
+ # puts "pub key hex: #{pub_key_hex}"
79
+
80
+ public_key_string = ECDSA::Format::PointOctetString.encode(public_key, compression: false)
81
+ calculated_address = Digest::Keccak.digest(public_key_string[1..], 256)
82
+ calculated_address[-20..].unpack1('H*')
83
+ end
84
+
85
+ def verify_public_key_and_address(public_key, address)
86
+ "0x#{public_key_to_address(public_key)}".downcase == address.downcase
87
+ end
88
+
89
+ METAMASK_HASH_PREFIX = "\x19Ethereum Signed Message:\n".freeze
90
+
91
+ def message_for_hashing(raw_message)
92
+ "#{METAMASK_HASH_PREFIX}#{raw_message.size}#{raw_message}"
93
+ end
94
+
95
+ def metamask_digest(message)
96
+ return nil if message.nil?
97
+
98
+ Digest::Keccak.digest(message_for_hashing(message), 256)
99
+ end
100
+
101
+ def recovery_id_from_v(v_value)
102
+ raise 'v needs to be an integerr' unless v_value.is_a? Integer
103
+
104
+ rec_id = v_value - 27
105
+ raise "Invalid recovery id: #{rec_id}" if rec_id < 0 || rec_id > 3
106
+
107
+ rec_id
108
+ end
109
+
110
+ def my_signature_from_hex(signature)
111
+ signature_byte_string = signature
112
+
113
+ case signature.size
114
+ when 130
115
+ signature_byte_string = [signature].pack('H*')
116
+ when 65
117
+ # do nothing
118
+ else
119
+ return nil
120
+ end
121
+
122
+ signature_bytes = signature_byte_string.unpack('C*')
123
+ r = signature_bytes[0, 32].pack('C*').unpack1('H*').to_i(16)
124
+ s = signature_bytes[32, 32].pack('C*').unpack1('H*').to_i(16)
125
+ v = signature_bytes[64].to_i
126
+
127
+ sig_obj = ECDSA::Signature.new(r, s)
128
+ MySignature.new(sig_obj, v)
129
+ end
130
+
131
+ def recover_public_key(hash, my_signature)
132
+ keys = ECDSA.recover_public_key(ECDSA::Group::Secp256k1, hash, my_signature.sig_obj).map do |p|
133
+ ECDSA::Format::PointOctetString.encode(p, compression: true)
134
+ end
135
+ key = keys[recovery_id_from_v(my_signature.v)]
136
+ return nil if key.nil?
137
+
138
+ ECDSA::Format::PointOctetString.decode(key, ECDSA::Group::Secp256k1)
139
+ end
140
+ end
141
+ end
@@ -0,0 +1,4 @@
1
+ module RubyAuthMetamask
2
+ module ApplicationHelper
3
+ end
4
+ end
@@ -0,0 +1,4 @@
1
+ module RubyAuthMetamask
2
+ module UsersHelper
3
+ end
4
+ end
@@ -0,0 +1,4 @@
1
+ module RubyAuthMetamask
2
+ class ApplicationJob < ActiveJob::Base
3
+ end
4
+ end
@@ -0,0 +1,6 @@
1
+ module RubyAuthMetamask
2
+ class ApplicationMailer < ActionMailer::Base
3
+ default from: "from@example.com"
4
+ layout "mailer"
5
+ end
6
+ end
@@ -0,0 +1,5 @@
1
+ module RubyAuthMetamask
2
+ class ApplicationRecord < ActiveRecord::Base
3
+ self.abstract_class = true
4
+ end
5
+ end
@@ -0,0 +1,4 @@
1
+ module RubyAuthMetamask
2
+ class User < ApplicationRecord
3
+ end
4
+ end
@@ -0,0 +1,15 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Ruby auth metamask</title>
5
+ <%= csrf_meta_tags %>
6
+ <%= csp_meta_tag %>
7
+
8
+ <%= stylesheet_link_tag "ruby_auth_metamask/application", media: "all" %>
9
+ </head>
10
+ <body>
11
+
12
+ <%= yield %>
13
+
14
+ </body>
15
+ </html>
@@ -0,0 +1,68 @@
1
+ <div class="h-screen flex justify-center items-center">
2
+ <div class="w-1/2 sm:text-base lg:text-xl bg-gray-200 m-4 px-4 py-10 rounded-xl ">
3
+ <%= form_with url: verify_path, method: :post do |form| %>
4
+ <div class="flex flex-col justfy-center mt-2 p-4">
5
+ <div class="w-full bg-orange-200 p-2 rounded-lg ">
6
+ <%= form.label :address, "Address:", class: "w-2/3" %>
7
+ </div>
8
+ <div class="w-full rounded-lg mt-1">
9
+ <%= form.text_field :address, class: "border-2 border-blue-200 w-full p-2 rounded-lg text-base" %>
10
+ </div>
11
+ <div class="w-full mt-2 bg-orange-200 p-2 rounded-lg ">
12
+ <%= form.label :signature, "Signature:", class: "w-2/3" %>
13
+ </div>
14
+ <div class="w-full rounded-lg mt-1">
15
+ <%= form.text_area :signature, rows: 4, class: "border-2 border-blue-200 w-full p-2 rounded-lg text-base" %>
16
+ </div>
17
+ <div class="flex justfy-center mt-10">
18
+ <hr class="h-px my-2 bg-gray-200 border-0 dark:bg-gray-700" />
19
+ <%= form.submit "Submit", class: "w-1/2 mx-auto rounded-lg p-3 bg-orange-300" %>
20
+ </div>
21
+ </div>
22
+ <% end %>
23
+ </div>
24
+ </div>
25
+
26
+
27
+ <script src="https://cdn.tailwindcss.com"></script>
28
+ <script src="https://cdn.jsdelivr.net/npm/web3@1.3.6/dist/web3.min.js"></script>
29
+
30
+ <script>
31
+
32
+ function myHandler() {
33
+ if (! window.ethereum) {
34
+ alert("MetaMask is not directed. Please install MetaMask extension first.");
35
+ return
36
+ }
37
+
38
+ const web3 = new Web3(window.ethereum);
39
+ window.ethereum.request({ method: 'eth_requestAccounts' })
40
+ .then(accounts => {
41
+ const accountAddress = accounts[0];
42
+ console.log('Account Address:', accountAddress);
43
+
44
+ const addrInput = document.getElementById("address");
45
+ addrInput.value = accountAddress;
46
+
47
+ const message = "<%= @message %>";
48
+
49
+ web3.eth.personal.sign(message, accountAddress, (error, signature) => {
50
+ if (!error) {
51
+ console.log("Signature: ", signature);
52
+ const sigInput = document.getElementById("signature");
53
+ sigInput.value = signature;
54
+ } else {
55
+ alert("Error signing message: " + error);
56
+ return;
57
+ }
58
+ });
59
+ })
60
+ .catch(error => {
61
+ alert("Error requesting accounts: " + error);
62
+ return;
63
+ });
64
+ };
65
+
66
+ myHandler();
67
+
68
+ </script>
data/config/routes.rb ADDED
@@ -0,0 +1,6 @@
1
+ RubyAuthMetamask::Engine.routes.draw do
2
+ get 'signin', to: 'users#signin', as: 'signin'
3
+ post 'verify', to: 'users#verify', as: 'verify'
4
+
5
+ root to: 'users#signin'
6
+ end
@@ -0,0 +1,9 @@
1
+ class CreateRubyAuthMetamaskUsers < ActiveRecord::Migration[7.1]
2
+ def change
3
+ create_table :ruby_auth_metamask_users do |t|
4
+ t.string :address
5
+
6
+ t.timestamps
7
+ end
8
+ end
9
+ end
@@ -0,0 +1,5 @@
1
+ module RubyAuthMetamask
2
+ class Engine < ::Rails::Engine
3
+ isolate_namespace RubyAuthMetamask
4
+ end
5
+ end
@@ -0,0 +1,3 @@
1
+ module RubyAuthMetamask
2
+ VERSION = "0.1.0"
3
+ end
@@ -0,0 +1,6 @@
1
+ require "ruby_auth_metamask/version"
2
+ require "ruby_auth_metamask/engine"
3
+
4
+ module RubyAuthMetamask
5
+ # Your code goes here...
6
+ end
@@ -0,0 +1,4 @@
1
+ # desc "Explaining what the task does"
2
+ # task :ruby_auth_metamask do
3
+ # # Task goes here
4
+ # end
metadata ADDED
@@ -0,0 +1,110 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: ruby_auth_metamask
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Ethan Zhang
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2023-12-17 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rails
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: 7.1.2
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: 7.1.2
27
+ - !ruby/object:Gem::Dependency
28
+ name: ecdsa
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: keccak
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ description: This is a Ruby on Rails Engine. It authenticates users against their
56
+ metamask signature.
57
+ email:
58
+ - yzhang.wa@gmail.com
59
+ executables: []
60
+ extensions: []
61
+ extra_rdoc_files: []
62
+ files:
63
+ - MIT-LICENSE
64
+ - README.md
65
+ - Rakefile
66
+ - app/assets/config/ruby_auth_metamask_manifest.js
67
+ - app/assets/stylesheets/ruby_auth_metamask/application.css
68
+ - app/controllers/ruby_auth_metamask/application_controller.rb
69
+ - app/controllers/ruby_auth_metamask/users_controller.rb
70
+ - app/helpers/ruby_auth_metamask/application_helper.rb
71
+ - app/helpers/ruby_auth_metamask/users_helper.rb
72
+ - app/jobs/ruby_auth_metamask/application_job.rb
73
+ - app/mailers/ruby_auth_metamask/application_mailer.rb
74
+ - app/models/ruby_auth_metamask/application_record.rb
75
+ - app/models/ruby_auth_metamask/user.rb
76
+ - app/views/layouts/ruby_auth_metamask/application.html.erb
77
+ - app/views/ruby_auth_metamask/users/signin.html.erb
78
+ - config/routes.rb
79
+ - db/migrate/20231213214539_create_ruby_auth_metamask_users.rb
80
+ - lib/ruby_auth_metamask.rb
81
+ - lib/ruby_auth_metamask/engine.rb
82
+ - lib/ruby_auth_metamask/version.rb
83
+ - lib/tasks/ruby_auth_metamask_tasks.rake
84
+ homepage: https://github.com/yzhanginwa/ruby_auth_metamask
85
+ licenses:
86
+ - MIT
87
+ metadata:
88
+ homepage_uri: https://github.com/yzhanginwa/ruby_auth_metamask
89
+ source_code_uri: https://github.com/yzhanginwa/ruby_auth_metamask
90
+ changelog_uri: https://github.com/yzhanginwa/ruby_auth_metamask
91
+ post_install_message:
92
+ rdoc_options: []
93
+ require_paths:
94
+ - lib
95
+ required_ruby_version: !ruby/object:Gem::Requirement
96
+ requirements:
97
+ - - ">="
98
+ - !ruby/object:Gem::Version
99
+ version: '0'
100
+ required_rubygems_version: !ruby/object:Gem::Requirement
101
+ requirements:
102
+ - - ">="
103
+ - !ruby/object:Gem::Version
104
+ version: '0'
105
+ requirements: []
106
+ rubygems_version: 3.3.26
107
+ signing_key:
108
+ specification_version: 4
109
+ summary: A Ruby on Rails Engine to authenticate metamask users
110
+ test_files: []