lita-reviewme 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +18 -0
- data/.travis.yml +9 -0
- data/Gemfile +3 -0
- data/LICENSE +19 -0
- data/README.md +63 -0
- data/Rakefile +6 -0
- data/lib/lita-reviewme.rb +7 -0
- data/lib/lita/handlers/reviewme.rb +130 -0
- data/lita-reviewme.gemspec +24 -0
- data/locales/en.yml +4 -0
- data/spec/lita/handlers/reviewme_spec.rb +98 -0
- data/spec/spec_helper.rb +4 -0
- metadata +143 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: 1e8fff96ac7d5fbaceabb93d02c1f59fae42d91b
|
4
|
+
data.tar.gz: b1ede2cbc19443b492720d950eeb8db4911c9087
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: d8a4740bf51e05899730caa0aad8fe7d3d5da6ee02ac353d795a7824ebe9973038292704aaf2b77629b40ef68a57bedc05a075cc1ef1f6890c2a441992d83d86
|
7
|
+
data.tar.gz: cd51dd3f83e9f96ff2c11baa6dbd33747ca85a5b191226abca74d9b6c74cbffd9b9684116c2277670ec03ab6bcaf4e0dcd4e351a51bcce7c882868bed540212f
|
data/.gitignore
ADDED
data/.travis.yml
ADDED
data/Gemfile
ADDED
data/LICENSE
ADDED
@@ -0,0 +1,19 @@
|
|
1
|
+
Copyright (c) 2015 Jay Hayes
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
4
|
+
of this software and associated documentation files (the "Software"), to deal
|
5
|
+
in the Software without restriction, including without limitation the rights
|
6
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
7
|
+
copies of the Software, and to permit persons to whom the Software is
|
8
|
+
furnished to do so, subject to the following conditions:
|
9
|
+
|
10
|
+
The above copyright notice and this permission notice shall be included in
|
11
|
+
all copies or substantial portions of the Software.
|
12
|
+
|
13
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
14
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
15
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
16
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
17
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
18
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
19
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,63 @@
|
|
1
|
+
# lita-reviewme
|
2
|
+
[![Build Status](https://travis-ci.org/iamvery/lita-reviewme.svg?branch=master)](https://travis-ci.org/iamvery/lita-reviewme)
|
3
|
+
|
4
|
+
A [lita](https://www.lita.io/) handler that helps with [code review](http://en.wikipedia.org/wiki/Code_review)
|
5
|
+
without getting in the way.
|
6
|
+
|
7
|
+
The handler rotates, in order, through a list of names to provider a "reviewer"
|
8
|
+
for some unit of work.
|
9
|
+
|
10
|
+
## Installation
|
11
|
+
|
12
|
+
Add lita-reviewme to your Lita instance's Gemfile:
|
13
|
+
|
14
|
+
``` ruby
|
15
|
+
gem "lita-reviewme", github: "iamvery/lita-reviewme"
|
16
|
+
```
|
17
|
+
|
18
|
+
## Configuration
|
19
|
+
|
20
|
+
Environment variable needed for Github integration:
|
21
|
+
|
22
|
+
```
|
23
|
+
ENV["GITHUB_WOLFBRAIN_ACCESS_TOKEN"]
|
24
|
+
```
|
25
|
+
|
26
|
+
## Usage
|
27
|
+
|
28
|
+
### See who is in the review rotation.
|
29
|
+
|
30
|
+
> **Jay H.** Nerdbot: reviewers
|
31
|
+
>
|
32
|
+
> **Nerdbot** iamvery, zacstewart, ...
|
33
|
+
|
34
|
+
### Add a name to the review rotation
|
35
|
+
|
36
|
+
> **Jay H.** Nerdbot: add kyfast to reviews
|
37
|
+
>
|
38
|
+
> **Nerdbot** added kyfast to reviews
|
39
|
+
|
40
|
+
### Remove a name from the review rotation
|
41
|
+
|
42
|
+
> **Jay H.** Nerdbot: remove kyfast from reviews
|
43
|
+
>
|
44
|
+
> **Nerdbot** removed kyfast from reviews
|
45
|
+
|
46
|
+
### Fetch the next reviewer
|
47
|
+
|
48
|
+
> **Jay H.** Nerdbot: review me
|
49
|
+
>
|
50
|
+
> **Nerdbot** iamvery
|
51
|
+
|
52
|
+
### Comment on a Github pull request or issue
|
53
|
+
This will post a comment mentioning the next reviewer on the referenced Github
|
54
|
+
pull request or issue. In order for this to work, @wolfbrain must have access
|
55
|
+
to the repository.
|
56
|
+
|
57
|
+
> **Jay H.** Nerdbot: review https://github.com/iamvery/lita-reviewme/issues/7
|
58
|
+
>
|
59
|
+
> **Nerdbot** iamvery should be on it...
|
60
|
+
|
61
|
+
## License
|
62
|
+
|
63
|
+
[MIT](http://opensource.org/licenses/MIT)
|
data/Rakefile
ADDED
@@ -0,0 +1,130 @@
|
|
1
|
+
require 'octokit'
|
2
|
+
|
3
|
+
module Lita
|
4
|
+
module Handlers
|
5
|
+
class Reviewme < Handler
|
6
|
+
REDIS_LIST = "reviewers"
|
7
|
+
|
8
|
+
route(
|
9
|
+
/add (.+) to reviews/i,
|
10
|
+
:add_reviewer,
|
11
|
+
command: true,
|
12
|
+
)
|
13
|
+
|
14
|
+
route(
|
15
|
+
/add reviewer (.+)/i,
|
16
|
+
:add_reviewer,
|
17
|
+
command: true,
|
18
|
+
help: { "add reviewer iamvery" => "adds iamvery to the reviewer rotation" },
|
19
|
+
)
|
20
|
+
|
21
|
+
route(
|
22
|
+
/remove (.+) from reviews/i,
|
23
|
+
:remove_reviewer,
|
24
|
+
command: true,
|
25
|
+
)
|
26
|
+
|
27
|
+
route(
|
28
|
+
/remove reviewer (.+)/i,
|
29
|
+
:remove_reviewer,
|
30
|
+
command: true,
|
31
|
+
help: { "remove reviewer iamvery" => "removes iamvery from the reviewer rotation" },
|
32
|
+
)
|
33
|
+
|
34
|
+
route(
|
35
|
+
/reviewers/i,
|
36
|
+
:display_reviewers,
|
37
|
+
command: true,
|
38
|
+
help: { "reviewers" => "display list of reviewers" },
|
39
|
+
)
|
40
|
+
|
41
|
+
route(
|
42
|
+
/review me/i,
|
43
|
+
:generate_assignment,
|
44
|
+
command: true,
|
45
|
+
help: { "review me" => "responds with the next reviewer" },
|
46
|
+
)
|
47
|
+
|
48
|
+
route(
|
49
|
+
%r{review <?(?<url>(https://)?github.com/(?<repo>.+)/(pull|issues)/(?<id>\d+))>?}i,
|
50
|
+
:comment_on_github,
|
51
|
+
command: true,
|
52
|
+
help: { "review https://github.com/user/repo/pull/123" => "adds comment to GH issue requesting review" },
|
53
|
+
)
|
54
|
+
|
55
|
+
route(
|
56
|
+
%r{review <?(https?://(?!github.com).*)>?}i,
|
57
|
+
:mention_reviewer,
|
58
|
+
command: true,
|
59
|
+
help: { "review http://some-non-github-url.com" => "requests review of the given URL in chat" }
|
60
|
+
)
|
61
|
+
|
62
|
+
def add_reviewer(response)
|
63
|
+
reviewer = response.matches.flatten.first
|
64
|
+
redis.lpush(REDIS_LIST, reviewer)
|
65
|
+
response.reply("added #{reviewer} to reviews")
|
66
|
+
end
|
67
|
+
|
68
|
+
def remove_reviewer(response)
|
69
|
+
reviewer = response.matches.flatten.first
|
70
|
+
redis.lrem(REDIS_LIST, 0, reviewer)
|
71
|
+
response.reply("removed #{reviewer} from reviews")
|
72
|
+
end
|
73
|
+
|
74
|
+
def display_reviewers(response)
|
75
|
+
reviewers = redis.lrange(REDIS_LIST, 0, -1)
|
76
|
+
response.reply_privately(reviewers.join(', '))
|
77
|
+
end
|
78
|
+
|
79
|
+
def generate_assignment(response)
|
80
|
+
reviewer = next_reviewer
|
81
|
+
response.reply(reviewer.to_s)
|
82
|
+
end
|
83
|
+
|
84
|
+
def comment_on_github(response)
|
85
|
+
repo = response.match_data[:repo]
|
86
|
+
id = response.match_data[:id]
|
87
|
+
reviewer = next_reviewer
|
88
|
+
comment = github_comment(reviewer)
|
89
|
+
|
90
|
+
begin
|
91
|
+
github_client.add_comment(repo, id, comment)
|
92
|
+
response.reply("#{reviewer} should be on it...")
|
93
|
+
rescue Octokit::Error
|
94
|
+
url = response.match_data[:url]
|
95
|
+
response.reply("I couldn't post a comment. (Are the permissions right?) #{chat_mention(reviewer, url)}")
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
def mention_reviewer(response)
|
100
|
+
url = response.matches.flatten.first
|
101
|
+
reviewer = next_reviewer
|
102
|
+
response.reply(chat_mention(reviewer, url))
|
103
|
+
end
|
104
|
+
|
105
|
+
private
|
106
|
+
|
107
|
+
def next_reviewer
|
108
|
+
redis.rpoplpush(REDIS_LIST, REDIS_LIST)
|
109
|
+
end
|
110
|
+
|
111
|
+
def github_comment(reviewer)
|
112
|
+
":eyes: @#{reviewer}"
|
113
|
+
end
|
114
|
+
|
115
|
+
def github_client
|
116
|
+
@github_client ||= Octokit::Client.new(access_token: github_access_token)
|
117
|
+
end
|
118
|
+
|
119
|
+
def github_access_token
|
120
|
+
ENV['GITHUB_WOLFBRAIN_ACCESS_TOKEN']
|
121
|
+
end
|
122
|
+
|
123
|
+
def chat_mention(reviewer, url)
|
124
|
+
"#{reviewer}: :eyes: #{url}"
|
125
|
+
end
|
126
|
+
end
|
127
|
+
|
128
|
+
Lita.register_handler(Reviewme)
|
129
|
+
end
|
130
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
Gem::Specification.new do |spec|
|
2
|
+
spec.name = "lita-reviewme"
|
3
|
+
spec.version = "0.0.1"
|
4
|
+
spec.authors = ["Jay Hayes"]
|
5
|
+
spec.email = ["ur@iamvery.com"]
|
6
|
+
spec.description = %q{A lita handler that helps with code review}
|
7
|
+
spec.summary = %q{A lita handler that helps with code review}
|
8
|
+
spec.homepage = "https://github.com/iamvery/lita-reviewme"
|
9
|
+
spec.license = "MIT"
|
10
|
+
spec.metadata = { "lita_plugin_type" => "handler" }
|
11
|
+
|
12
|
+
spec.files = `git ls-files`.split($/)
|
13
|
+
spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
|
14
|
+
spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
|
15
|
+
spec.require_paths = ["lib"]
|
16
|
+
|
17
|
+
spec.add_runtime_dependency "lita", ">= 3.1"
|
18
|
+
spec.add_runtime_dependency "octokit", "~> 3.7"
|
19
|
+
|
20
|
+
spec.add_development_dependency "bundler", "~> 1.3"
|
21
|
+
spec.add_development_dependency "fakeredis"
|
22
|
+
spec.add_development_dependency "rake"
|
23
|
+
spec.add_development_dependency "rspec", ">= 3.0.0"
|
24
|
+
end
|
data/locales/en.yml
ADDED
@@ -0,0 +1,98 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
describe Lita::Handlers::Reviewme, lita_handler: true do
|
4
|
+
it { is_expected.to route_command("add iamvery to reviews").to :add_reviewer }
|
5
|
+
it { is_expected.to route_command("add reviewer iamvery").to :add_reviewer }
|
6
|
+
it { is_expected.to route_command("remove iamvery from reviews").to :remove_reviewer }
|
7
|
+
it { is_expected.to route_command("remove reviewer iamvery").to :remove_reviewer }
|
8
|
+
it { is_expected.to route_command("reviewers").to :display_reviewers }
|
9
|
+
it { is_expected.to route_command("review me").to :generate_assignment }
|
10
|
+
it { is_expected.to route_command("review https://github.com/user/repo/pull/123").to :comment_on_github }
|
11
|
+
it { is_expected.to route_command("review <https://github.com/user/repo/pull/123>").to :comment_on_github }
|
12
|
+
it { is_expected.to route_command("review https://github.com/user/repo/issues/123").to :comment_on_github }
|
13
|
+
it { is_expected.to route_command("review https://bitbucket.org/user/repo/pull-requests/123").to :mention_reviewer }
|
14
|
+
it { is_expected.to route_command("review <https://bitbucket.org/user/repo/pull-requests/123>").to :mention_reviewer }
|
15
|
+
|
16
|
+
let(:reply) { replies.last }
|
17
|
+
|
18
|
+
describe "#add_reviewer" do
|
19
|
+
it "adds a name to the list" do
|
20
|
+
send_command("add iamvery to reviews")
|
21
|
+
|
22
|
+
expect(reply).to eq("added iamvery to reviews")
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
describe "#remove_reviewer" do
|
27
|
+
it "removes a name from the list" do
|
28
|
+
send_command("remove iamvery from reviews")
|
29
|
+
|
30
|
+
expect(reply).to eq("removed iamvery from reviews")
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
describe "#generate_assignment" do
|
35
|
+
it "responds with the next reviewer's name" do
|
36
|
+
send_command("add iamvery to reviews")
|
37
|
+
send_command("review me")
|
38
|
+
|
39
|
+
expect(reply).to eq("iamvery")
|
40
|
+
end
|
41
|
+
|
42
|
+
it "rotates the response each time" do
|
43
|
+
send_command("add iamvery to reviews")
|
44
|
+
send_command("add zacstewart to reviews")
|
45
|
+
|
46
|
+
send_command("review me")
|
47
|
+
expect(replies.last).to eq("iamvery")
|
48
|
+
|
49
|
+
send_command("review me")
|
50
|
+
expect(replies.last).to eq("zacstewart")
|
51
|
+
end
|
52
|
+
end
|
53
|
+
|
54
|
+
describe "#comment_on_github" do
|
55
|
+
it "posts comment on github" do
|
56
|
+
repo = "gh_user/repo"
|
57
|
+
id = "123"
|
58
|
+
|
59
|
+
expect_any_instance_of(Octokit::Client).to receive(:add_comment)
|
60
|
+
.with(repo, id, ":eyes: @iamvery")
|
61
|
+
|
62
|
+
send_command("add iamvery to reviews")
|
63
|
+
send_command("review https://github.com/#{repo}/pull/#{id}")
|
64
|
+
|
65
|
+
expect(replies.last).to eq("iamvery should be on it...")
|
66
|
+
end
|
67
|
+
|
68
|
+
it "handles errors gracefully" do
|
69
|
+
expect_any_instance_of(Octokit::Client).to receive(:add_comment)
|
70
|
+
.and_raise(Octokit::Error)
|
71
|
+
|
72
|
+
url = "https://github.com/iamvery/lita-reviewme/pull/5"
|
73
|
+
|
74
|
+
send_command("add iamvery to reviews")
|
75
|
+
send_command("review #{url}")
|
76
|
+
|
77
|
+
expect(replies.last).to eq("I couldn't post a comment. (Are the permissions right?) iamvery: :eyes: #{url}")
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
81
|
+
describe "#display_reviewers" do
|
82
|
+
it "responds with list of reviewers" do
|
83
|
+
send_command("add iamvery to reviews")
|
84
|
+
send_command("add zacstewart to reviews")
|
85
|
+
send_command("reviewers")
|
86
|
+
|
87
|
+
expect(reply).to eq("zacstewart, iamvery")
|
88
|
+
end
|
89
|
+
end
|
90
|
+
|
91
|
+
describe "#mention_reviewer" do
|
92
|
+
it "mentions a reviewer in chat with the given URL" do
|
93
|
+
send_command("add iamvery to reviews")
|
94
|
+
send_command("review https://bitbucket.org/user/repo/pull-requests/123")
|
95
|
+
expect(replies.last).to eq("iamvery: :eyes: https://bitbucket.org/user/repo/pull-requests/123")
|
96
|
+
end
|
97
|
+
end
|
98
|
+
end
|
data/spec/spec_helper.rb
ADDED
metadata
ADDED
@@ -0,0 +1,143 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: lita-reviewme
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Jay Hayes
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
date: 2016-01-12 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: lita
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - ">="
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '3.1'
|
20
|
+
type: :runtime
|
21
|
+
prerelease: false
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - ">="
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: '3.1'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: octokit
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - "~>"
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '3.7'
|
34
|
+
type: :runtime
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '3.7'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: bundler
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - "~>"
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: '1.3'
|
48
|
+
type: :development
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '1.3'
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
name: fakeredis
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - ">="
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '0'
|
62
|
+
type: :development
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - ">="
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '0'
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: rake
|
71
|
+
requirement: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - ">="
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '0'
|
76
|
+
type: :development
|
77
|
+
prerelease: false
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - ">="
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '0'
|
83
|
+
- !ruby/object:Gem::Dependency
|
84
|
+
name: rspec
|
85
|
+
requirement: !ruby/object:Gem::Requirement
|
86
|
+
requirements:
|
87
|
+
- - ">="
|
88
|
+
- !ruby/object:Gem::Version
|
89
|
+
version: 3.0.0
|
90
|
+
type: :development
|
91
|
+
prerelease: false
|
92
|
+
version_requirements: !ruby/object:Gem::Requirement
|
93
|
+
requirements:
|
94
|
+
- - ">="
|
95
|
+
- !ruby/object:Gem::Version
|
96
|
+
version: 3.0.0
|
97
|
+
description: A lita handler that helps with code review
|
98
|
+
email:
|
99
|
+
- ur@iamvery.com
|
100
|
+
executables: []
|
101
|
+
extensions: []
|
102
|
+
extra_rdoc_files: []
|
103
|
+
files:
|
104
|
+
- ".gitignore"
|
105
|
+
- ".travis.yml"
|
106
|
+
- Gemfile
|
107
|
+
- LICENSE
|
108
|
+
- README.md
|
109
|
+
- Rakefile
|
110
|
+
- lib/lita-reviewme.rb
|
111
|
+
- lib/lita/handlers/reviewme.rb
|
112
|
+
- lita-reviewme.gemspec
|
113
|
+
- locales/en.yml
|
114
|
+
- spec/lita/handlers/reviewme_spec.rb
|
115
|
+
- spec/spec_helper.rb
|
116
|
+
homepage: https://github.com/iamvery/lita-reviewme
|
117
|
+
licenses:
|
118
|
+
- MIT
|
119
|
+
metadata:
|
120
|
+
lita_plugin_type: handler
|
121
|
+
post_install_message:
|
122
|
+
rdoc_options: []
|
123
|
+
require_paths:
|
124
|
+
- lib
|
125
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
126
|
+
requirements:
|
127
|
+
- - ">="
|
128
|
+
- !ruby/object:Gem::Version
|
129
|
+
version: '0'
|
130
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
131
|
+
requirements:
|
132
|
+
- - ">="
|
133
|
+
- !ruby/object:Gem::Version
|
134
|
+
version: '0'
|
135
|
+
requirements: []
|
136
|
+
rubyforge_project:
|
137
|
+
rubygems_version: 2.4.5.1
|
138
|
+
signing_key:
|
139
|
+
specification_version: 4
|
140
|
+
summary: A lita handler that helps with code review
|
141
|
+
test_files:
|
142
|
+
- spec/lita/handlers/reviewme_spec.rb
|
143
|
+
- spec/spec_helper.rb
|