divergence_meter 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 11401d367f7f29f91db7ec7b2a96629558f66b3e
4
+ data.tar.gz: 7e9c6d0a2dd47883133ad9656ce475026e85ff76
5
+ SHA512:
6
+ metadata.gz: e8e3400eb78ef692b6b69d12d38f1260507103a4ba2e7f0afe094eeac2cd061028dd87e869ce66cf37795b54240497ab6a773406eea0d6e08c56177ded3e04d3
7
+ data.tar.gz: f2051a1fae07242df719f07dff08445001c4f7841b93b42ef20f9b97a13432c97ddcbf058428836b05383d83af8e59544e89fe94461c56cfbe9a3a4fd9ff7529
data/.gitignore ADDED
@@ -0,0 +1,14 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /Gemfile.lock
4
+ /_yardoc/
5
+ /coverage/
6
+ /doc/
7
+ /pkg/
8
+ /spec/reports/
9
+ /tmp/
10
+ *.bundle
11
+ *.so
12
+ *.o
13
+ *.a
14
+ mkmf.log
data/.rspec ADDED
@@ -0,0 +1,2 @@
1
+ --format documentation
2
+ --color
data/.rubocop.yml ADDED
@@ -0,0 +1 @@
1
+ inherit_from: .rubocop_todo.yml
data/.rubocop_todo.yml ADDED
@@ -0,0 +1,29 @@
1
+ # This configuration was generated by `rubocop --auto-gen-config`
2
+ # on 2014-11-09 22:14:15 +0900 using RuboCop version 0.27.0.
3
+ # The point is for the user to remove these configuration records
4
+ # one by one as the offenses are removed from the code base.
5
+ # Note that changes in the inspected code, or installation of new
6
+ # versions of RuboCop, may require this file to be generated again.
7
+
8
+ # Offense count: 2
9
+ Metrics/AbcSize:
10
+ Max: 33
11
+
12
+ # Offense count: 2
13
+ # Configuration parameters: CountComments.
14
+ Metrics/MethodLength:
15
+ Max: 13
16
+
17
+ # Offense count: 4
18
+ Style/Documentation:
19
+ Enabled: false
20
+
21
+ # Offense count: 1
22
+ # Configuration parameters: MinBodyLength.
23
+ Style/GuardClause:
24
+ Enabled: false
25
+
26
+ # Offense count: 2
27
+ Style/RegexpLiteral:
28
+ MaxSlashes: 0
29
+
data/.travis.yml ADDED
@@ -0,0 +1,5 @@
1
+ language: ruby
2
+ rvm:
3
+ - 2.0
4
+ - 2.1
5
+ - 2.2
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in divergence_meter.gemspec
4
+ gemspec
data/LICENSE.txt ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2014 sugamasao
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,71 @@
1
+ # DivergenceMeter
2
+
3
+ [![Build Status](https://travis-ci.org/sugamasao/divergence_meter.svg)](https://travis-ci.org/sugamasao/divergence_meter)
4
+
5
+ DivergenceMeter is Levenshtein distance tool and Library.
6
+
7
+ ## Installation
8
+
9
+ Add this line to your application's Gemfile:
10
+
11
+ ```ruby
12
+ gem 'divergence_meter'
13
+ ```
14
+
15
+ And then execute:
16
+
17
+ $ bundle
18
+
19
+ Or install it yourself as:
20
+
21
+ $ gem install divergence_meter
22
+
23
+ ## Usage
24
+
25
+ ### use command line.
26
+
27
+ calculate Levenshtein distance.
28
+
29
+ ```sh
30
+ $ divergence_meter retire tire
31
+ 2
32
+ $ divergence_meter retire retare
33
+ 1
34
+ $ divergence_meter retire hoge
35
+ 5
36
+ ```
37
+
38
+ did you mean?
39
+
40
+ ```sh
41
+ $ divergence_meter retire tire retare hoge
42
+ retare
43
+ ```
44
+
45
+ ### use library
46
+
47
+ calculate Levenshtein distance.
48
+
49
+ ```ruby
50
+ require 'divergence_meter'
51
+
52
+ puts DivergenceMeter.distance('retire', 'tire')
53
+ # => 2
54
+ ```
55
+
56
+ did you mean?
57
+
58
+ ```ruby
59
+ require 'divergence_meter'
60
+
61
+ puts DivergenceMeter.did_you_mean('retire', %w(tire retare hoge))
62
+ # => hog1
63
+ ```
64
+
65
+ ## Contributing
66
+
67
+ 1. Fork it ( https://github.com/[my-github-username]/divergence_meter/fork )
68
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
69
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
70
+ 4. Push to the branch (`git push origin my-new-feature`)
71
+ 5. Create a new Pull Request
data/Rakefile ADDED
@@ -0,0 +1,6 @@
1
+ require 'bundler/gem_tasks'
2
+ require 'rspec/core/rake_task'
3
+
4
+ RSpec::Core::RakeTask.new(:spec)
5
+
6
+ task default: :spec
@@ -0,0 +1,12 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'divergence_meter'
4
+
5
+ DivergenceMeter::CLI.new(ARGV).parse
6
+
7
+ target = ARGV.shift
8
+ if ARGV.size == 1
9
+ puts DivergenceMeter.distance(target, ARGV.first)
10
+ else
11
+ puts DivergenceMeter.did_you_mean(target, ARGV)
12
+ end
@@ -0,0 +1,26 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'divergence_meter/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = 'divergence_meter'
8
+ spec.version = DivergenceMeter::VERSION
9
+ spec.authors = ['sugamasao']
10
+ spec.email = ['sugamasao@gmail.com']
11
+ spec.summary = 'Levenshtein distance tool and Library.'
12
+ spec.description = 'Levenshtein distance tool and Library. provide Levenshtein distance and did you mean keyword.'
13
+ spec.homepage = 'https://github.com/sugamasao/divergence_meter'
14
+ spec.license = 'MIT'
15
+
16
+ spec.files = `git ls-files -z`.split("\x0")
17
+ spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
18
+ spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
19
+ spec.require_paths = ['lib']
20
+
21
+ spec.add_development_dependency 'bundler', '~> 1.7'
22
+ spec.add_development_dependency 'rake', '~> 10.0'
23
+ spec.add_development_dependency 'rspec', '~> 3.1'
24
+ spec.add_development_dependency 'simplecov', '~> 0.9'
25
+ spec.add_development_dependency 'rubocop', '~> 0.27'
26
+ end
@@ -0,0 +1,43 @@
1
+ module DivergenceMeter
2
+ class CLI
3
+ def initialize(argv)
4
+ @argv = argv
5
+ end
6
+
7
+ def parse
8
+ if @argv.include?('-h') || @argv.include?('--help')
9
+ puts usage
10
+ exit
11
+ end
12
+
13
+ if @argv.include?('-v') || @argv.include?('--version')
14
+ puts version
15
+ exit
16
+ end
17
+
18
+ if @argv.size < 2
19
+ puts 'Invalid Argument'
20
+ puts usage
21
+ exit 1
22
+ end
23
+ end
24
+
25
+ private
26
+
27
+ def usage
28
+ <<-EOS
29
+ usage: #{ $PROGRAM_NAME } word1 word2
30
+ -> levenshtein distance
31
+
32
+ or
33
+
34
+ usage: #{ $PROGRAM_NAME } target word1 word2
35
+ -> target did you mean word1 or word2
36
+ EOS
37
+ end
38
+
39
+ def version
40
+ "Version: #{ VERSION }"
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,43 @@
1
+ module DivergenceMeter
2
+ class LevenshteinDistance
3
+ def self.run(word1, word2)
4
+ new.run(word1, word2)
5
+ end
6
+
7
+ def run(word1, word2)
8
+ list = create_list(word1.size, word2.size)
9
+
10
+ 1.upto(word1.size) do |n|
11
+ 1.upto(word2.size) do |m|
12
+ list[n][m] = [
13
+ list[n - 1][m] + 1,
14
+ list[n][m - 1] + 1,
15
+ list[n - 1][m - 1] + cost(word1[n - 1], word2[m - 1])
16
+ ].min
17
+ end
18
+ end
19
+
20
+ list[word1.size][word2.size]
21
+ end
22
+
23
+ private
24
+
25
+ def create_list(word1_size, word2_size)
26
+ list = Array.new(word1_size + 1) { Array.new(word2_size + 1, 0) }
27
+
28
+ list.size.times do |n|
29
+ list[n][0] = n
30
+ end
31
+
32
+ list[0].size.times do |n|
33
+ list[0][n] = n
34
+ end
35
+
36
+ list
37
+ end
38
+
39
+ def cost(char1, char2)
40
+ char1 == char2 ? 0 : 1
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,3 @@
1
+ module DivergenceMeter
2
+ VERSION = '1.0.0'
3
+ end
@@ -0,0 +1,27 @@
1
+ require 'divergence_meter/version'
2
+ require_relative 'divergence_meter/levenshtein_distance'
3
+ require_relative 'divergence_meter/cli'
4
+
5
+ module DivergenceMeter
6
+ class << self
7
+ def distance(word1, word2)
8
+ LevenshteinDistance.run(word1, word2)
9
+ end
10
+
11
+ def did_you_mean(target, words)
12
+ size = target.size
13
+
14
+ Array(words).map { |word|
15
+ { word: word, distance: distance(target, word) }
16
+ }.sort { |a, b|
17
+ if a[:distance] == b[:distance]
18
+ (size - a[:word].size).abs <=> (size - b[:word].size).abs
19
+ else
20
+ a[:distance] <=> b[:distance]
21
+ end
22
+ }.first[:word]
23
+ end
24
+
25
+ alias_method :もしかして, :did_you_mean
26
+ end
27
+ end
@@ -0,0 +1,33 @@
1
+ require 'spec_helper'
2
+
3
+ describe DivergenceMeter do
4
+ it 'has a version number' do
5
+ expect(DivergenceMeter::VERSION).not_to be nil
6
+ end
7
+
8
+ describe '.distance' do
9
+
10
+ it 'hoge vs hoga' do
11
+ expect(DivergenceMeter.distance('hoge', 'hoga')).to eq(1)
12
+ end
13
+
14
+ it 'apple vs play' do
15
+ expect(DivergenceMeter.distance('apple', 'play')).to eq(4)
16
+ end
17
+
18
+ it 'perl vs pearl' do
19
+ expect(DivergenceMeter.distance('perl', 'pearl')).to eq(1)
20
+ end
21
+ end
22
+
23
+ describe '.もしかして(did_you_mean)' do
24
+ it '__send vs [__send__, send]' do
25
+ expect(DivergenceMeter.もしかして(':__send', methods)).to eq(:__send__)
26
+ end
27
+ it 'tire, retire' do
28
+ expect(DivergenceMeter.もしかして('tire', %w(tire retire))).to eq('tire')
29
+ expect(DivergenceMeter.もしかして('rtire', %w(tire retire))).to eq('tire')
30
+ expect(DivergenceMeter.もしかして('abtire', %w(tire retire))).to eq('retire')
31
+ end
32
+ end
33
+ end
@@ -0,0 +1,4 @@
1
+ require 'simplecov'
2
+ SimpleCov.start
3
+ $LOAD_PATH.unshift File.expand_path('../../lib', __FILE__)
4
+ require 'divergence_meter'
metadata ADDED
@@ -0,0 +1,135 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: divergence_meter
3
+ version: !ruby/object:Gem::Version
4
+ version: 1.0.0
5
+ platform: ruby
6
+ authors:
7
+ - sugamasao
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-11-11 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: bundler
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '1.7'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '1.7'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '10.0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '10.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rspec
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '3.1'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '3.1'
55
+ - !ruby/object:Gem::Dependency
56
+ name: simplecov
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '0.9'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '0.9'
69
+ - !ruby/object:Gem::Dependency
70
+ name: rubocop
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '0.27'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '0.27'
83
+ description: Levenshtein distance tool and Library. provide Levenshtein distance and
84
+ did you mean keyword.
85
+ email:
86
+ - sugamasao@gmail.com
87
+ executables:
88
+ - divergence_meter
89
+ extensions: []
90
+ extra_rdoc_files: []
91
+ files:
92
+ - ".gitignore"
93
+ - ".rspec"
94
+ - ".rubocop.yml"
95
+ - ".rubocop_todo.yml"
96
+ - ".travis.yml"
97
+ - Gemfile
98
+ - LICENSE.txt
99
+ - README.md
100
+ - Rakefile
101
+ - bin/divergence_meter
102
+ - divergence_meter.gemspec
103
+ - lib/divergence_meter.rb
104
+ - lib/divergence_meter/cli.rb
105
+ - lib/divergence_meter/levenshtein_distance.rb
106
+ - lib/divergence_meter/version.rb
107
+ - spec/divergence_meter_spec.rb
108
+ - spec/spec_helper.rb
109
+ homepage: https://github.com/sugamasao/divergence_meter
110
+ licenses:
111
+ - MIT
112
+ metadata: {}
113
+ post_install_message:
114
+ rdoc_options: []
115
+ require_paths:
116
+ - lib
117
+ required_ruby_version: !ruby/object:Gem::Requirement
118
+ requirements:
119
+ - - ">="
120
+ - !ruby/object:Gem::Version
121
+ version: '0'
122
+ required_rubygems_version: !ruby/object:Gem::Requirement
123
+ requirements:
124
+ - - ">="
125
+ - !ruby/object:Gem::Version
126
+ version: '0'
127
+ requirements: []
128
+ rubyforge_project:
129
+ rubygems_version: 2.2.2
130
+ signing_key:
131
+ specification_version: 4
132
+ summary: Levenshtein distance tool and Library.
133
+ test_files:
134
+ - spec/divergence_meter_spec.rb
135
+ - spec/spec_helper.rb