Initial commit
This commit is contained in:
		
						commit
						2a69e4f15c
					
				
					 16 changed files with 301 additions and 0 deletions
				
			
		
							
								
								
									
										7
									
								
								.editorconfig
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.editorconfig
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | |||
| [*.cr] | ||||
| charset = utf-8 | ||||
| end_of_line = lf | ||||
| insert_final_newline = true | ||||
| indent_style = space | ||||
| indent_size = 2 | ||||
| trim_trailing_whitespace = true | ||||
							
								
								
									
										8
									
								
								.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.gitignore
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | |||
| /doc/ | ||||
| /lib/ | ||||
| /bin/ | ||||
| /.shards/ | ||||
| 
 | ||||
| # Libraries don't need dependency lock | ||||
| # Dependencies will be locked in application that uses them | ||||
| /shard.lock | ||||
							
								
								
									
										1
									
								
								.travis.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.travis.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | |||
| language: crystal | ||||
							
								
								
									
										21
									
								
								LICENSE
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								LICENSE
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,21 @@ | |||
| The MIT License (MIT) | ||||
| 
 | ||||
| Copyright (c) 2017 Andrew Hamon | ||||
| 
 | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
| 
 | ||||
| The above copyright notice and this permission notice shall be included in | ||||
| all copies or substantial portions of the Software. | ||||
| 
 | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||
| THE SOFTWARE. | ||||
							
								
								
									
										52
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								README.md
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,52 @@ | |||
| # cox | ||||
| 
 | ||||
| Crystal bindings for the [libsodium box API](https://download.libsodium.org/doc/public-key_cryptography/authenticated_encryption.html) | ||||
| 
 | ||||
| Given a recipients public key, you can encrypt and sign a message for them. Upon | ||||
| receipt, they can decrypt and authenticate the message as having come from you. | ||||
| 
 | ||||
| ## Installation | ||||
| 
 | ||||
| Add this to your application's `shard.yml`: | ||||
| 
 | ||||
| ```yaml | ||||
| dependencies: | ||||
|   cox: | ||||
|     github: andrewhamon/cox | ||||
| ``` | ||||
| 
 | ||||
| ## Usage | ||||
| 
 | ||||
| ```crystal | ||||
| require "cox" | ||||
| 
 | ||||
| data = "Hello World!" | ||||
| 
 | ||||
| # Alice is the sender | ||||
| alice = Cox::KeyPair.new | ||||
| 
 | ||||
| # Bob is the recipient | ||||
| bob = Cox::KeyPair.new | ||||
| 
 | ||||
| # Encrypt a message for Bob using his public key, signing it with Alice's | ||||
| # secret key | ||||
| nonce, encrypted = Cox.encrypt(data, bob.public, alice.secret) | ||||
| 
 | ||||
| # Decrypt the message using Bob's secret key, and verify its signature against | ||||
| # Alice's public key | ||||
| decrypted = Cox.decrypt(encrypted, nonce, alice.public, bob.secret) | ||||
| 
 | ||||
| String.new(decrypted) # => "Hello World!" | ||||
| ``` | ||||
| 
 | ||||
| ## Contributing | ||||
| 
 | ||||
| 1. Fork it ( https://github.com/andrewhamon/cox/fork ) | ||||
| 2. Create your feature branch (git checkout -b my-new-feature) | ||||
| 3. Commit your changes (git commit -am 'Add some feature') | ||||
| 4. Push to the branch (git push origin my-new-feature) | ||||
| 5. Create a new Pull Request | ||||
| 
 | ||||
| ## Contributors | ||||
| 
 | ||||
| - [andrewhamon](https://github.com/andrewhamon) Andrew Hamon - creator, maintainer | ||||
							
								
								
									
										9
									
								
								shard.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								shard.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,9 @@ | |||
| name: cox | ||||
| version: 0.1.0 | ||||
| 
 | ||||
| authors: | ||||
|   - Andrew Hamon <andrew@hamon.cc> | ||||
| 
 | ||||
| crystal: 0.23.0 | ||||
| 
 | ||||
| license: MIT | ||||
							
								
								
									
										25
									
								
								spec/cox_spec.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								spec/cox_spec.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | |||
| require "./spec_helper" | ||||
| 
 | ||||
| describe Cox do | ||||
|   # TODO: Write tests | ||||
| 
 | ||||
|   it "works" do | ||||
|     data = "Hello World!" | ||||
| 
 | ||||
|     # Alice is the sender | ||||
|     alice = Cox::KeyPair.new | ||||
| 
 | ||||
|     # Bob is the recipient | ||||
|     bob = Cox::KeyPair.new | ||||
| 
 | ||||
|     # Encrypt a message for Bob using his public key, signing it with Alice's | ||||
|     # secret key | ||||
|     nonce, encrypted = Cox.encrypt(data, bob.public, alice.secret) | ||||
| 
 | ||||
|     # Decrypt the message using Bob's secret key, and verify its signature against | ||||
|     # Alice's public key | ||||
|     decrypted = Cox.decrypt(encrypted, nonce, alice.public, bob.secret) | ||||
| 
 | ||||
|     String.new(decrypted).should eq(data) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										2
									
								
								spec/spec_helper.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								spec/spec_helper.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | |||
| require "spec" | ||||
| require "../src/cox" | ||||
							
								
								
									
										30
									
								
								src/cox.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/cox.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,30 @@ | |||
| require "./cox/*" | ||||
| require "secure_random" | ||||
| 
 | ||||
| module Cox | ||||
|   def self.encrypt(data, nonce : Nonce, recipient_public_key : PublicKey, sender_secret_key : SecretKey) | ||||
|     data_buffer = data.to_slice | ||||
|     data_size = data_buffer.bytesize | ||||
|     output_buffer = Bytes.new(data_buffer.bytesize + LibSodium::MAC_BYTES) | ||||
|     LibSodium.crypto_box_easy(output_buffer.to_unsafe, data_buffer, data_size, nonce.pointer, recipient_public_key.pointer, sender_secret_key.pointer) | ||||
|     output_buffer | ||||
|   end | ||||
| 
 | ||||
|   def self.encrypt(data, recipient_public_key : PublicKey, sender_secret_key : SecretKey) | ||||
|     nonce = Nonce.new | ||||
|     {nonce, encrypt(data, nonce, recipient_public_key, sender_secret_key)} | ||||
|   end | ||||
| 
 | ||||
|   def self.decrypt(data, nonce : Nonce, sender_public_key : PublicKey, recipient_secret_key : SecretKey) | ||||
|     data_buffer = data.to_slice | ||||
|     data_size = data_buffer.bytesize | ||||
|     output_buffer = Bytes.new(data_buffer.bytesize - LibSodium::MAC_BYTES) | ||||
|     LibSodium.crypto_box_open_easy(output_buffer.to_unsafe, data_buffer.to_unsafe, data_size, nonce.pointer, sender_public_key.pointer, recipient_secret_key.pointer) | ||||
|     output_buffer | ||||
|   end | ||||
| end | ||||
| 
 | ||||
| if Cox::LibSodium.sodium_init() == -1 | ||||
|   STDERR.puts("Failed to init libsodium") | ||||
|   exit(1) | ||||
| end | ||||
							
								
								
									
										21
									
								
								src/cox/key.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/cox/key.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,21 @@ | |||
| module Cox | ||||
|   abstract class Key | ||||
|     abstract def bytes | ||||
| 
 | ||||
|     def pointer | ||||
|       bytes.to_unsafe | ||||
|     end | ||||
| 
 | ||||
|     def pointer(size) | ||||
|       bytes.pointer(size) | ||||
|     end | ||||
| 
 | ||||
|     def to_base64 | ||||
|       Base64.encode(bytes) | ||||
|     end | ||||
| 
 | ||||
|     def self.from_base64(encoded_key) | ||||
|       new(Base64.decode(encoded_key)) | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										25
									
								
								src/cox/key_pair.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/cox/key_pair.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | |||
| require "./lib_sodium" | ||||
| 
 | ||||
| 
 | ||||
| module Cox | ||||
|   class KeyPair | ||||
|     property public : PublicKey | ||||
|     property secret : SecretKey | ||||
| 
 | ||||
|     def initialize(@public, @secret) | ||||
|     end | ||||
| 
 | ||||
|     def self.new(pub : Bytes, sec : Bytes) | ||||
|       new(PublicKey.new(pub), SecretKey.new(sec)) | ||||
|     end | ||||
| 
 | ||||
|     def self.new | ||||
|       public_key = Bytes.new(PublicKey::KEY_LENGTH) | ||||
|       secret_key = Bytes.new(SecretKey::KEY_LENGTH) | ||||
| 
 | ||||
|       LibSodium.crypto_box_keypair(public_key.to_unsafe, secret_key.to_unsafe) | ||||
| 
 | ||||
|       new(public_key, secret_key) | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										39
									
								
								src/cox/lib_sodium.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/cox/lib_sodium.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,39 @@ | |||
| module Cox | ||||
|   @[Link("sodium")] | ||||
|   lib LibSodium | ||||
|     fun sodium_init() : LibC::Int | ||||
| 
 | ||||
|     fun crypto_box_publickeybytes() : LibC::SizeT | ||||
|     fun crypto_box_secretkeybytes() : LibC::SizeT | ||||
|     fun crypto_box_noncebytes()     : LibC::SizeT | ||||
|     fun crypto_box_macbytes()       : LibC::SizeT | ||||
| 
 | ||||
|     PUBLIC_KEY_BYTES = crypto_box_publickeybytes() | ||||
|     SECRET_KEY_BYTES = crypto_box_secretkeybytes() | ||||
|     NONCE_BYTES      = crypto_box_macbytes() | ||||
|     MAC_BYTES        = crypto_box_macbytes() | ||||
| 
 | ||||
|     fun crypto_box_keypair( | ||||
|       public_key_output : Pointer(LibC::UChar), | ||||
|       secret_key_output : Pointer(LibC::UChar) | ||||
|     ) | ||||
| 
 | ||||
|     fun crypto_box_easy( | ||||
|       output               : Pointer(LibC::UChar), | ||||
|       data                 : Pointer(LibC::UChar), | ||||
|       data_size            : LibC::ULongLong, | ||||
|       nonce                : Pointer(LibC::UChar), | ||||
|       recipient_public_key : Pointer(LibC::UChar), | ||||
|       sender_secret_key    : Pointer(LibC::UChar) | ||||
|     ) : LibC::Int | ||||
| 
 | ||||
|     fun crypto_box_open_easy( | ||||
|       output               : Pointer(LibC::UChar), | ||||
|       data                 : Pointer(LibC::UChar), | ||||
|       data_size            : LibC::ULongLong, | ||||
|       nonce                : Pointer(LibC::UChar), | ||||
|       sender_public_key    : Pointer(LibC::UChar), | ||||
|       recipient_secret_key : Pointer(LibC::UChar) | ||||
|     ) : LibC::Int | ||||
|   end | ||||
| end | ||||
							
								
								
									
										28
									
								
								src/cox/nonce.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/cox/nonce.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | |||
| require "./lib_sodium" | ||||
| require "secure_random" | ||||
| 
 | ||||
| module Cox | ||||
|   class Nonce | ||||
|     property bytes : Bytes | ||||
| 
 | ||||
|     NONCE_LENGTH = LibSodium::NONCE_BYTES | ||||
| 
 | ||||
|     def initialize(@bytes : Bytes) | ||||
|       if bytes.bytesize != NONCE_LENGTH | ||||
|         raise ArgumentError.new("Nonce must be #{NONCE_LENGTH} bytes, got #{bytes.bytesize}") | ||||
|       end | ||||
|     end | ||||
| 
 | ||||
|     def self.new | ||||
|       new(SecureRandom.random_bytes(NONCE_LENGTH)) | ||||
|     end | ||||
| 
 | ||||
|     def pointer | ||||
|       bytes.to_unsafe | ||||
|     end | ||||
| 
 | ||||
|     def pointer(size) | ||||
|       bytes.pointer(size) | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										15
									
								
								src/cox/public_key.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/cox/public_key.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,15 @@ | |||
| require "./lib_sodium" | ||||
| 
 | ||||
| module Cox | ||||
|   class PublicKey < Key | ||||
|     property bytes : Bytes | ||||
| 
 | ||||
|     KEY_LENGTH = LibSodium::PUBLIC_KEY_BYTES | ||||
| 
 | ||||
|     def initialize(@bytes : Bytes) | ||||
|       if bytes.bytesize != KEY_LENGTH | ||||
|         raise ArgumentError.new("Public key must be #{KEY_LENGTH} bytes, got #{bytes.bytesize}") | ||||
|       end | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										15
									
								
								src/cox/secret_key.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/cox/secret_key.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,15 @@ | |||
| require "./lib_sodium" | ||||
| 
 | ||||
| module Cox | ||||
|   class SecretKey < Key | ||||
|     property bytes : Bytes | ||||
| 
 | ||||
|     KEY_LENGTH = LibSodium::SECRET_KEY_BYTES | ||||
| 
 | ||||
|     def initialize(@bytes : Bytes) | ||||
|       if bytes.bytesize != KEY_LENGTH | ||||
|         raise ArgumentError.new("Secret key must be #{KEY_LENGTH} bytes, got #{bytes.bytesize}") | ||||
|       end | ||||
|     end | ||||
|   end | ||||
| end | ||||
							
								
								
									
										3
									
								
								src/cox/version.cr
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/cox/version.cr
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | |||
| module Cox | ||||
|   VERSION = "0.1.0" | ||||
| end | ||||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 Andrew Hamon
						Andrew Hamon