hashtree 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
data/.gitignore ADDED
@@ -0,0 +1,18 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
18
+ .rbx
data/.rspec ADDED
@@ -0,0 +1,2 @@
1
+ --color
2
+ --format progress
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in hashtree.gemspec
4
+ gemspec
data/LICENSE.txt ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2012 Marcelo Wiermann
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,29 @@
1
+ # Hashtree
2
+
3
+ TODO: Write a gem description
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'hashtree'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install hashtree
18
+
19
+ ## Usage
20
+
21
+ TODO: Write usage instructions here
22
+
23
+ ## Contributing
24
+
25
+ 1. Fork it
26
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
27
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
28
+ 4. Push to the branch (`git push origin my-new-feature`)
29
+ 5. Create new Pull Request
data/Rakefile ADDED
@@ -0,0 +1 @@
1
+ require "bundler/gem_tasks"
@@ -0,0 +1,93 @@
1
+ require 'rubygems'
2
+ require 'benchmark'
3
+ require 'hashtree'
4
+ require 'securerandom'
5
+ require 'digest'
6
+
7
+ def gen_doc(i)
8
+ name = "John Doe #{i}"
9
+ email = "john.doe.#{i}@example.com"
10
+ key = "jdoe#{i}"
11
+ doc = { name: name, email: email, password: "secret#{i}" }
12
+ return key, doc
13
+ end
14
+
15
+ Benchmark.bm do |x|
16
+ puts "Creating a HashTree with 1.000 documents"
17
+ x.report do
18
+ @htree = HashTree.new
19
+ 1_000.times do |i|
20
+ key, doc = gen_doc(i)
21
+ @htree[key] = doc
22
+ end
23
+ end
24
+
25
+ puts "Creating a HashTree with 10.000 documents"
26
+ x.report do
27
+ @htree = HashTree.new
28
+ 10_000.times do |i|
29
+ key, doc = gen_doc(i)
30
+ @htree[key] = doc
31
+ end
32
+ end
33
+
34
+ puts "Creating a HashTree with 100.000 documents"
35
+ x.report do
36
+ @htree = HashTree.new
37
+ 100_000.times do |i|
38
+ key, doc = gen_doc(i)
39
+ @htree[key] = doc
40
+ end
41
+ end
42
+
43
+ puts "Reading 1.000 random documents"
44
+ x.report do
45
+ 1_000.times do |i|
46
+ key = "jdoe#{rand(100_000)}"
47
+ @htree[key]
48
+ end
49
+ end
50
+
51
+ puts "Reading 10.000 random documents"
52
+ x.report do
53
+ 10_000.times do |i|
54
+ key = "jdoe#{rand(100_000)}"
55
+ @htree[key]
56
+ end
57
+ end
58
+
59
+ puts "Reading 100.000 random documents"
60
+ x.report do
61
+ 100_000.times do |i|
62
+ key = "jdoe#{rand(100_000)}"
63
+ @htree[key]
64
+ end
65
+ end
66
+
67
+ puts "Deleting 1.000 random documents"
68
+ x.report do
69
+ lhtree = @htree
70
+ 1_000.times do |i|
71
+ key = "jdoe#{rand(100_000)}"
72
+ lhtree.delete(key)
73
+ end
74
+ end
75
+
76
+ puts "Deleting 10.000 random documents"
77
+ x.report do
78
+ lhtree = @htree
79
+ 10_000.times do |i|
80
+ key = "jdoe#{rand(100_000)}"
81
+ lhtree.delete(key)
82
+ end
83
+ end
84
+
85
+ puts "Deleting 100.000 random documents"
86
+ x.report do
87
+ lhtree = @htree
88
+ 100_000.times do |i|
89
+ key = "jdoe#{rand(100_000)}"
90
+ lhtree.delete(key)
91
+ end
92
+ end
93
+ end
data/hashtree.gemspec ADDED
@@ -0,0 +1,24 @@
1
+ # -*- encoding: utf-8 -*-
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'hashtree/version'
5
+
6
+ Gem::Specification.new do |gem|
7
+ gem.name = "hashtree"
8
+ gem.version = HashTree::VERSION
9
+ gem.authors = ["Marcelo Wiermann"]
10
+ gem.email = ["marcelo.wiermann@gmail.com"]
11
+ gem.description = %q{HashTree that has a Hash-like interface and stores data into blocks}
12
+ gem.summary = %q{HashTree}
13
+ gem.homepage = ""
14
+
15
+ gem.files = `git ls-files`.split($/)
16
+ gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
17
+ gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
18
+ gem.require_paths = ["lib"]
19
+
20
+ gem.add_development_dependency 'rspec'
21
+ gem.add_development_dependency 'guard'
22
+ gem.add_development_dependency 'factory_girl'
23
+
24
+ end
@@ -0,0 +1,3 @@
1
+ class HashTree
2
+ VERSION = "0.0.1"
3
+ end
data/lib/hashtree.rb ADDED
@@ -0,0 +1,130 @@
1
+ require 'rubygems'
2
+ require 'digest'
3
+ require 'securerandom'
4
+
5
+ require "hashtree/version"
6
+
7
+ class HashTree
8
+
9
+ attr_reader :index, :branches, :settings
10
+
11
+ def initialize(settings = {})
12
+ @settings = {
13
+ max_branch_size: 16
14
+ }.merge(settings)
15
+ @index = {}
16
+ @branches = {
17
+ free: {}
18
+ }
19
+ end
20
+
21
+ # Public API
22
+ def [](key)
23
+ begin
24
+ id = @index[key]
25
+ if id
26
+ read(id)
27
+ else
28
+ nil
29
+ end
30
+ rescue
31
+ nil
32
+ end
33
+ end
34
+
35
+ def []=(key, value)
36
+ begin
37
+ id = @index[key]
38
+ if id
39
+ update(id, value)
40
+ else
41
+ create(key, value)
42
+ end
43
+ rescue
44
+ nil
45
+ end
46
+ end
47
+
48
+ def delete(key)
49
+ begin
50
+ id = @index[key]
51
+ if id
52
+ doc = read(id)
53
+ destroy(id)
54
+ @index.delete(key)
55
+ return doc
56
+ else
57
+ nil
58
+ end
59
+ rescue
60
+ nil
61
+ end
62
+ end
63
+
64
+ def clear
65
+ @index = {}
66
+ @branches = {}
67
+ end
68
+
69
+ def count
70
+ @index.count
71
+ end
72
+
73
+ def branch(key)
74
+ branch_id = get_branch_id(key)
75
+ @branches[branch_id] if branch_id
76
+ end
77
+
78
+ private
79
+ def get_branch_id(key)
80
+ id = @index[key]
81
+ if id
82
+ branch_id = id[0..15]
83
+ else
84
+ nil
85
+ end
86
+ end
87
+
88
+ # Private API
89
+ def read(id)
90
+ branch_id, leaf_id = id[0..15], id[16..32]
91
+ @branches[branch_id][leaf_id]
92
+ end
93
+
94
+ def update(id, value)
95
+ branch_id, leaf_id = id[0..15], id[16..32]
96
+ @branches[branch_id][leaf_id] = value
97
+ return value
98
+ end
99
+
100
+ def create(key, value)
101
+ free_branch_id = @branches[:free].keys.sample
102
+ if free_branch_id
103
+ leaf_id = Digest::MD5.hexdigest(SecureRandom.uuid)[0..15]
104
+ @index[key] = free_branch_id + leaf_id
105
+ branch = @branches[free_branch_id]
106
+ branch[leaf_id] = value
107
+ @branches[:free].delete(branch_id) if branch.size >= @settings[:max_branch_size]
108
+ else
109
+ id = Digest::MD5.hexdigest(SecureRandom.uuid)
110
+ @index[key] = id
111
+ branch_id, leaf_id = id[0..15], id[16..32]
112
+ @branches[branch_id] = {}
113
+ @branches[branch_id][leaf_id] = value
114
+ end
115
+ return value
116
+ end
117
+
118
+ def destroy(id)
119
+ branch_id, leaf_id = id[0..15], id[16..32]
120
+ branch = @branches[branch_id]
121
+ destroyed_element = branch.delete(leaf_id)
122
+ if branch.size == 0
123
+ @branches.delete(branch_id)
124
+ @branches[:free].delete(branch_id)
125
+ elsif branch.size > 0 and branch.size < 16
126
+ @branches[:free][branch_id] = nil
127
+ end
128
+ end
129
+
130
+ end
@@ -0,0 +1,17 @@
1
+ # This file was generated by the `rspec --init` command. Conventionally, all
2
+ # specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`.
3
+ # Require this file using `require "spec_helper"` to ensure that it is only
4
+ # loaded once.
5
+ #
6
+ # See http://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration
7
+ RSpec.configure do |config|
8
+ config.treat_symbols_as_metadata_keys_with_true_values = true
9
+ config.run_all_when_everything_filtered = true
10
+ config.filter_run :focus
11
+
12
+ # Run specs in random order to surface order dependencies. If you find an
13
+ # order dependency and want to debug it, you can fix the order by providing
14
+ # the seed, which is printed after each run.
15
+ # --seed 1234
16
+ config.order = 'random'
17
+ end
metadata ADDED
@@ -0,0 +1,105 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: hashtree
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Marcelo Wiermann
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2012-09-29 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: rspec
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ! '>='
20
+ - !ruby/object:Gem::Version
21
+ version: '0'
22
+ type: :development
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ! '>='
28
+ - !ruby/object:Gem::Version
29
+ version: '0'
30
+ - !ruby/object:Gem::Dependency
31
+ name: guard
32
+ requirement: !ruby/object:Gem::Requirement
33
+ none: false
34
+ requirements:
35
+ - - ! '>='
36
+ - !ruby/object:Gem::Version
37
+ version: '0'
38
+ type: :development
39
+ prerelease: false
40
+ version_requirements: !ruby/object:Gem::Requirement
41
+ none: false
42
+ requirements:
43
+ - - ! '>='
44
+ - !ruby/object:Gem::Version
45
+ version: '0'
46
+ - !ruby/object:Gem::Dependency
47
+ name: factory_girl
48
+ requirement: !ruby/object:Gem::Requirement
49
+ none: false
50
+ requirements:
51
+ - - ! '>='
52
+ - !ruby/object:Gem::Version
53
+ version: '0'
54
+ type: :development
55
+ prerelease: false
56
+ version_requirements: !ruby/object:Gem::Requirement
57
+ none: false
58
+ requirements:
59
+ - - ! '>='
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ description: HashTree that has a Hash-like interface and stores data into blocks
63
+ email:
64
+ - marcelo.wiermann@gmail.com
65
+ executables: []
66
+ extensions: []
67
+ extra_rdoc_files: []
68
+ files:
69
+ - .gitignore
70
+ - .rspec
71
+ - Gemfile
72
+ - LICENSE.txt
73
+ - README.md
74
+ - Rakefile
75
+ - examples/benchmarks.rb
76
+ - hashtree.gemspec
77
+ - lib/hashtree.rb
78
+ - lib/hashtree/version.rb
79
+ - spec/spec_helper.rb
80
+ homepage: ''
81
+ licenses: []
82
+ post_install_message:
83
+ rdoc_options: []
84
+ require_paths:
85
+ - lib
86
+ required_ruby_version: !ruby/object:Gem::Requirement
87
+ none: false
88
+ requirements:
89
+ - - ! '>='
90
+ - !ruby/object:Gem::Version
91
+ version: '0'
92
+ required_rubygems_version: !ruby/object:Gem::Requirement
93
+ none: false
94
+ requirements:
95
+ - - ! '>='
96
+ - !ruby/object:Gem::Version
97
+ version: '0'
98
+ requirements: []
99
+ rubyforge_project:
100
+ rubygems_version: 1.8.24
101
+ signing_key:
102
+ specification_version: 3
103
+ summary: HashTree
104
+ test_files:
105
+ - spec/spec_helper.rb