summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAzul <azul@riseup.net>2017-09-22 15:30:40 +0200
committerAzul <azul@riseup.net>2017-09-22 15:30:47 +0200
commit3f777b4cbc7eaa4d127ec31e1d625bba65a40c0d (patch)
treec352dbcf7c493afad8fad1ee343734563f311301
parentbc48ac30c232f1e3fa5f5ad455f14d5fec17abeb (diff)
wkd: implement basic lookup of keys through wkd
wkd is the web key directory. See the Readme.md in /lib/nickserver/wkd
-rw-r--r--lib/nickserver/email_address.rb4
-rw-r--r--lib/nickserver/wkd/Readme.md8
-rw-r--r--lib/nickserver/wkd/source.rb16
-rw-r--r--lib/nickserver/wkd/url.rb31
-rw-r--r--lib/zbase32.rb19
-rw-r--r--test/unit/email_address_test.rb5
-rw-r--r--test/unit/wkd/url_test.rb26
-rw-r--r--test/unit/zbase_test.rb35
8 files changed, 144 insertions, 0 deletions
diff --git a/lib/nickserver/email_address.rb b/lib/nickserver/email_address.rb
index c5d5df7..20f642c 100644
--- a/lib/nickserver/email_address.rb
+++ b/lib/nickserver/email_address.rb
@@ -37,6 +37,10 @@ module Nickserver
address.split('@')[1]
end
+ def local_part
+ address.split('@')[0]
+ end
+
def to_s
address
end
diff --git a/lib/nickserver/wkd/Readme.md b/lib/nickserver/wkd/Readme.md
new file mode 100644
index 0000000..c93e08c
--- /dev/null
+++ b/lib/nickserver/wkd/Readme.md
@@ -0,0 +1,8 @@
+Allow querying keys from web key directories offered by the users
+provider.
+
+Summary is here:
+ https://wiki.gnupg.org/WKD
+
+Specs are here:
+ https://tools.ietf.org/html/draft-koch-openpgp-webkey-service-00
diff --git a/lib/nickserver/wkd/source.rb b/lib/nickserver/wkd/source.rb
new file mode 100644
index 0000000..01f376e
--- /dev/null
+++ b/lib/nickserver/wkd/source.rb
@@ -0,0 +1,16 @@
+require 'nickserver/source'
+require 'nickserver/response'
+
+module Nickserver
+ module Wkd
+ class Source < Nickserver::Source
+
+ def query(email)
+ url = Url.new(email)
+ status, body = adapter.get url
+ return Nickserver::Response.new(status, body)
+ end
+
+ end
+ end
+end
diff --git a/lib/nickserver/wkd/url.rb b/lib/nickserver/wkd/url.rb
new file mode 100644
index 0000000..965e7ec
--- /dev/null
+++ b/lib/nickserver/wkd/url.rb
@@ -0,0 +1,31 @@
+require 'digest/sha1'
+require 'zbase32'
+
+module Nickserver
+ module Wkd
+ class Url
+
+ def initialize(email)
+ @domain = email.domain.downcase
+ @local_part = email.local_part.downcase
+ end
+
+ def to_s
+ "https://#{domain}/.well-known/openpgpkey" +
+ "/hu/#{domain}/#{encoded_digest}"
+ end
+
+ protected
+
+ attr_reader :domain, :local_part
+
+ def encoded_digest
+ ZBase32.encode32(digest.to_i(16).to_s(2))
+ end
+
+ def digest
+ Digest::SHA1.hexdigest local_part
+ end
+ end
+ end
+end
diff --git a/lib/zbase32.rb b/lib/zbase32.rb
new file mode 100644
index 0000000..754213d
--- /dev/null
+++ b/lib/zbase32.rb
@@ -0,0 +1,19 @@
+module ZBase32
+
+ ALPHABET = 'ybndrfg8ejkmcpqxot1uwisza345h769'.split('').freeze
+
+ def self.encode32(bin_string)
+ bin_string.scan(/[01]{1,5}/).map do |bits|
+ ALPHABET[bits.ljust(5, '0').to_i(2)]
+ end.join
+ end
+
+ def self.decode32(enc)
+ bin = enc.split('').map do |char|
+ ALPHABET.index(char).to_s(2).rjust(5, '0')
+ end.join
+ bin[0, (8 * (bin.length / 8))]
+ # .sub /10*$/ ,'1'
+ end
+
+end
diff --git a/test/unit/email_address_test.rb b/test/unit/email_address_test.rb
index 6aecef7..2fea65d 100644
--- a/test/unit/email_address_test.rb
+++ b/test/unit/email_address_test.rb
@@ -10,6 +10,11 @@ class EmailAddressTest < Minitest::Test
assert !nick.domain?('est.me')
end
+ def test_local_part
+ nick = Nickserver::EmailAddress.new 'nick@test.me'
+ assert_equal 'nick', nick.local_part
+ end
+
def test_valid
nick = Nickserver::EmailAddress.new 'nick@remote.domain'
assert nick.valid?
diff --git a/test/unit/wkd/url_test.rb b/test/unit/wkd/url_test.rb
new file mode 100644
index 0000000..9bf8f64
--- /dev/null
+++ b/test/unit/wkd/url_test.rb
@@ -0,0 +1,26 @@
+require 'test_helper'
+require 'nickserver/wkd/url'
+require 'nickserver/email_address'
+
+module Nickserver::Wkd
+ class UrlTest < Minitest::Test
+ # TODO: test utf8 behavior
+
+ # https://tools.ietf.org/html/draft-koch-openpgp-webkey-service-00#section-3.1
+ def test_sample_from_draft
+ url = Url.new sample_email
+ assert_equal sample_url, url.to_s
+ end
+
+ protected
+
+ def sample_email
+ Nickserver::EmailAddress.new 'Joe.Doe@Example.ORG'
+ end
+
+ def sample_url
+ 'https://example.org/.well-known/openpgpkey/' +
+ 'hu/example.org/iy9q119eutrkn8s1mk4r39qejnbu3n5q'
+ end
+ end
+end
diff --git a/test/unit/zbase_test.rb b/test/unit/zbase_test.rb
new file mode 100644
index 0000000..97f0fc5
--- /dev/null
+++ b/test/unit/zbase_test.rb
@@ -0,0 +1,35 @@
+$LOAD_PATH.unshift File.expand_path(File.dirname(__FILE__) + '/../lib')
+require 'minitest/autorun'
+require 'zbase32'
+
+class Zbase32Test < Minitest::Test
+
+ def test_samples
+ samples.each do |k,v|
+ assert_equal k, decode(v)
+ assert_equal v, encode(k)
+ end
+ end
+
+ protected
+
+ def samples
+ {
+ '111100001011111111000111' => '6n9hq',
+ '110101000111101000000100' => '4t7ye',
+ wkd_sample => 'iy9q119eutrkn8s1mk4r39qejnbu3n5q'
+ }
+ end
+
+ def wkd_sample
+ 'a83ee94be89c48a11ed25ab44cfdc848833c8b6e'.to_i(16).to_s(2)
+ end
+
+ def encode(string)
+ ZBase32.encode32 string
+ end
+
+ def decode(enc)
+ ZBase32.decode32 enc
+ end
+end