grade_runner 0.0.11 → 0.0.13
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/Gemfile +1 -0
- data/Gemfile.lock +2 -0
- data/VERSION +1 -1
- data/grade_runner.gemspec +5 -3
- data/lib/grade_runner/formatters/hint_formatter.rb +11 -0
- data/lib/grade_runner/formatters/json_output_formatter.rb +67 -0
- data/lib/grade_runner.rb +24 -0
- data/lib/tasks/grade.rake +94 -49
- metadata +22 -6
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 0f0b8b6d8c59749308e3c9dcd6ab96ae9db013bd388961e5ff5029a99a3d0e28
|
4
|
+
data.tar.gz: a1e4d49af97459f0b62b2b74379cbaf1ee21bbce80cf9ddda4562d41e63eaab9
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 8b005be1f8ee9db2d9385036e27233acbce5b5fc6d9eb0a6f875d76d5d7da095cf42d5dde54242b270efa2493ff30414d9f292263648d560e1f9305f0a23a2c5
|
7
|
+
data.tar.gz: c8b7c8026a6d880bca7021d36975f9c71eabcfd7375c4efde5821ff42840d045b333ab61c567b76aecdab61f58d2e48fac1199a9d095cbcddbde5d622bebdd93
|
data/Gemfile
CHANGED
data/Gemfile.lock
CHANGED
@@ -145,6 +145,7 @@ GEM
|
|
145
145
|
webrick (1.7.0)
|
146
146
|
yard (0.9.28)
|
147
147
|
webrick (~> 1.7.0)
|
148
|
+
zip (2.0.2)
|
148
149
|
|
149
150
|
PLATFORMS
|
150
151
|
ruby
|
@@ -166,6 +167,7 @@ DEPENDENCIES
|
|
166
167
|
rdoc (~> 6.1)
|
167
168
|
rspec (~> 3.5.0)
|
168
169
|
simplecov
|
170
|
+
zip
|
169
171
|
|
170
172
|
BUNDLED WITH
|
171
173
|
2.1.4
|
data/VERSION
CHANGED
@@ -1 +1 @@
|
|
1
|
-
0.0.
|
1
|
+
0.0.13
|
data/grade_runner.gemspec
CHANGED
@@ -2,16 +2,16 @@
|
|
2
2
|
# DO NOT EDIT THIS FILE DIRECTLY
|
3
3
|
# Instead, edit Juwelier::Tasks in Rakefile, and run 'rake gemspec'
|
4
4
|
# -*- encoding: utf-8 -*-
|
5
|
-
# stub: grade_runner 0.0.
|
5
|
+
# stub: grade_runner 0.0.13 ruby lib
|
6
6
|
|
7
7
|
Gem::Specification.new do |s|
|
8
8
|
s.name = "grade_runner".freeze
|
9
|
-
s.version = "0.0.
|
9
|
+
s.version = "0.0.13"
|
10
10
|
|
11
11
|
s.required_rubygems_version = Gem::Requirement.new(">= 0".freeze) if s.respond_to? :required_rubygems_version=
|
12
12
|
s.require_paths = ["lib".freeze]
|
13
13
|
s.authors = ["Raghu Betina".freeze, "Jelani Woods".freeze]
|
14
|
-
s.date = "
|
14
|
+
s.date = "2025-03-17"
|
15
15
|
s.description = "This gem runs your RSpec test suite and posts the JSON output to grades.firstdraft.com.".freeze
|
16
16
|
s.email = ["raghu@firstdraft.com".freeze, "jelani@firstdraft.com".freeze]
|
17
17
|
s.extra_rdoc_files = [
|
@@ -48,6 +48,7 @@ Gem::Specification.new do |s|
|
|
48
48
|
s.add_runtime_dependency(%q<activesupport>.freeze, [">= 2.3.5"])
|
49
49
|
s.add_runtime_dependency(%q<oj>.freeze, ["~> 3.13.12"])
|
50
50
|
s.add_runtime_dependency(%q<octokit>.freeze, ["~> 5.0"])
|
51
|
+
s.add_runtime_dependency(%q<zip>.freeze, [">= 0"])
|
51
52
|
s.add_runtime_dependency(%q<faraday-retry>.freeze, ["~> 1.0.3"])
|
52
53
|
s.add_runtime_dependency(%q<rake>.freeze, ["~> 13"])
|
53
54
|
s.add_development_dependency(%q<rspec>.freeze, ["~> 3.5.0"])
|
@@ -65,6 +66,7 @@ Gem::Specification.new do |s|
|
|
65
66
|
s.add_dependency(%q<activesupport>.freeze, [">= 2.3.5"])
|
66
67
|
s.add_dependency(%q<oj>.freeze, ["~> 3.13.12"])
|
67
68
|
s.add_dependency(%q<octokit>.freeze, ["~> 5.0"])
|
69
|
+
s.add_dependency(%q<zip>.freeze, [">= 0"])
|
68
70
|
s.add_dependency(%q<faraday-retry>.freeze, ["~> 1.0.3"])
|
69
71
|
s.add_dependency(%q<rake>.freeze, ["~> 13"])
|
70
72
|
s.add_dependency(%q<rspec>.freeze, ["~> 3.5.0"])
|
@@ -0,0 +1,11 @@
|
|
1
|
+
RSpec::Support.require_rspec_core "formatters/documentation_formatter"
|
2
|
+
|
3
|
+
class HintFormatter < RSpec::Core::Formatters::DocumentationFormatter
|
4
|
+
RSpec::Core::Formatters.register self, :example_failed
|
5
|
+
|
6
|
+
def example_failed(failure)
|
7
|
+
super
|
8
|
+
@output.puts "\n\nHint: #{failure.example.metadata[:hint][0]}" if failure.example.metadata[:hint].present?
|
9
|
+
end
|
10
|
+
|
11
|
+
end
|
@@ -0,0 +1,67 @@
|
|
1
|
+
RSpec::Support.require_rspec_core "formatters/json_formatter"
|
2
|
+
require "oj"
|
3
|
+
class JsonOutputFormatter < RSpec::Core::Formatters::JsonFormatter
|
4
|
+
RSpec::Core::Formatters.register self, :dump_summary
|
5
|
+
|
6
|
+
def dump_summary(summary)
|
7
|
+
total_points = summary.
|
8
|
+
examples.
|
9
|
+
map { |example| example.metadata.fetch(:points, GradeRunner.default_points).to_i }.
|
10
|
+
sum
|
11
|
+
|
12
|
+
earned_points = summary.
|
13
|
+
examples.
|
14
|
+
select { |example| example.execution_result.status == :passed }.
|
15
|
+
map { |example| example.metadata.fetch(:points, GradeRunner.default_points).to_i }.
|
16
|
+
sum
|
17
|
+
|
18
|
+
score = (earned_points.to_f / total_points).round(4)
|
19
|
+
score = 0 if score.nan?
|
20
|
+
|
21
|
+
@output_hash[:summary] = {
|
22
|
+
duration: summary.duration,
|
23
|
+
example_count: summary.example_count,
|
24
|
+
errors_outside_of_examples_count: summary.errors_outside_of_examples_count,
|
25
|
+
failure_count: summary.failure_count,
|
26
|
+
pending_count: summary.pending_count,
|
27
|
+
total_points: total_points,
|
28
|
+
earned_points: earned_points,
|
29
|
+
score: score
|
30
|
+
}
|
31
|
+
result = (@output_hash[:summary][:score] * 100).round(2)
|
32
|
+
|
33
|
+
if summary.errors_outside_of_examples_count.positive?
|
34
|
+
result = "An error occurred while running tests"
|
35
|
+
else
|
36
|
+
result = result.to_s + "%"
|
37
|
+
end
|
38
|
+
|
39
|
+
|
40
|
+
@output_hash[:summary_line] = [
|
41
|
+
"#{summary.example_count} #{summary.example_count == 1 ? "test" : "tests"}",
|
42
|
+
"#{summary.failure_count} failures",
|
43
|
+
"#{earned_points}/#{total_points} points",
|
44
|
+
result,
|
45
|
+
].join(", ")
|
46
|
+
end
|
47
|
+
|
48
|
+
def close(_notification)
|
49
|
+
output.write Oj.dump @output_hash
|
50
|
+
end
|
51
|
+
|
52
|
+
private
|
53
|
+
|
54
|
+
def format_example(example)
|
55
|
+
{
|
56
|
+
description: example.description,
|
57
|
+
full_description: example.full_description,
|
58
|
+
hint: example.metadata[:hint],
|
59
|
+
status: example.execution_result.status.to_s,
|
60
|
+
points: example.metadata.fetch(:points, GradeRunner.default_points),
|
61
|
+
file_path: example.metadata[:file_path],
|
62
|
+
line_number: example.metadata[:line_number],
|
63
|
+
run_time: example.execution_result.run_time,
|
64
|
+
}
|
65
|
+
end
|
66
|
+
|
67
|
+
end
|
data/lib/grade_runner.rb
CHANGED
@@ -1,2 +1,26 @@
|
|
1
1
|
require "grade_runner/runner"
|
2
2
|
require "grade_runner/railtie" if defined?(Rails)
|
3
|
+
|
4
|
+
module GradeRunner
|
5
|
+
class Error < StandardError; end
|
6
|
+
|
7
|
+
class << self
|
8
|
+
attr_writer :default_points, :override_local_specs
|
9
|
+
|
10
|
+
def default_points
|
11
|
+
@default_points || 1
|
12
|
+
end
|
13
|
+
|
14
|
+
def override_local_specs
|
15
|
+
if @override_local_specs.nil?
|
16
|
+
true
|
17
|
+
else
|
18
|
+
@override_local_specs
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
def config
|
23
|
+
yield self
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
data/lib/tasks/grade.rake
CHANGED
@@ -2,6 +2,9 @@ require "active_support/core_ext/object/blank"
|
|
2
2
|
require "grade_runner/runner"
|
3
3
|
require "octokit"
|
4
4
|
require "yaml"
|
5
|
+
require "zip"
|
6
|
+
require "fileutils"
|
7
|
+
require "open-uri"
|
5
8
|
|
6
9
|
desc "Alias for \"grade:next\"."
|
7
10
|
task grade: "grade:all" do
|
@@ -14,10 +17,11 @@ namespace :grade do
|
|
14
17
|
input_token = ARGV[1]
|
15
18
|
file_token = nil
|
16
19
|
|
17
|
-
config_dir_name =
|
20
|
+
config_dir_name = find_or_create_directory(".vscode")
|
18
21
|
config_file_name = "#{config_dir_name}/.ltici_apitoken.yml"
|
19
22
|
student_config = {}
|
20
23
|
student_config["submission_url"] = "https://grades.firstdraft.com"
|
24
|
+
student_config["github_username"] = retrieve_github_username
|
21
25
|
|
22
26
|
if File.exist?(config_file_name)
|
23
27
|
begin
|
@@ -47,7 +51,6 @@ namespace :grade do
|
|
47
51
|
while new_personal_access_token == "" do
|
48
52
|
print "> "
|
49
53
|
new_personal_access_token = $stdin.gets.chomp.strip
|
50
|
-
|
51
54
|
if new_personal_access_token!= "" && is_valid_token?(submission_url, new_personal_access_token) == false
|
52
55
|
puts "Please enter valid token"
|
53
56
|
new_personal_access_token = ""
|
@@ -67,17 +70,21 @@ namespace :grade do
|
|
67
70
|
student_config["personal_access_token"] = nil
|
68
71
|
update_config_file(config_file_name, student_config)
|
69
72
|
puts "Your access token looked invalid, so we've reset it to be blank. Please re-run rails grade and, when asked, copy-paste your token carefully from the assignment page."
|
70
|
-
else
|
71
|
-
|
72
|
-
|
73
|
-
|
73
|
+
else
|
74
|
+
if GradeRunner.override_local_specs
|
75
|
+
resource_info = upstream_repo(submission_url, token)
|
76
|
+
full_reponame = resource_info.fetch("repo_slug")
|
77
|
+
remote_spec_folder_sha = resource_info.fetch("spec_folder_sha")
|
78
|
+
source_code_url = resource_info.fetch("source_code_url")
|
79
|
+
set_upstream_remote(full_reponame)
|
80
|
+
sync_specs_with_source(full_reponame, remote_spec_folder_sha, source_code_url)
|
81
|
+
end
|
74
82
|
|
75
83
|
path = File.join(project_root, "/tmp/output/#{Time.now.to_i}.json")
|
76
84
|
`bin/rails db:migrate RAILS_ENV=test` if defined?(Rails)
|
77
85
|
`RAILS_ENV=test bundle exec rspec --format JsonOutputFormatter --out #{path}`
|
78
86
|
rspec_output_json = Oj.load(File.read(path))
|
79
|
-
|
80
|
-
username = github_username(github_email)
|
87
|
+
username = retrieve_github_username
|
81
88
|
reponame = project_root.to_s.split("/").last
|
82
89
|
sha = `git rev-parse HEAD`.slice(0..7)
|
83
90
|
|
@@ -102,12 +109,13 @@ namespace :grade do
|
|
102
109
|
|
103
110
|
desc "Reset access token saved in YAML file."
|
104
111
|
task :reset_token do
|
105
|
-
config_dir_name =
|
112
|
+
config_dir_name = find_or_create_directory(".vscode")
|
106
113
|
config_file_name = "#{config_dir_name}/.ltici_apitoken.yml"
|
107
114
|
submission_url = "https://grades.firstdraft.com"
|
108
115
|
|
109
116
|
student_config = {}
|
110
117
|
student_config["submission_url"] = submission_url
|
118
|
+
student_config["github_username"] = retrieve_github_username
|
111
119
|
puts "Enter your access token for this project"
|
112
120
|
new_personal_access_token = ""
|
113
121
|
|
@@ -131,34 +139,62 @@ namespace :grade do
|
|
131
139
|
|
132
140
|
end
|
133
141
|
|
134
|
-
def sync_specs_with_source(full_reponame)
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
142
|
+
def sync_specs_with_source(full_reponame, remote_sha, repo_url)
|
143
|
+
# Unstage staged changes in spec folder
|
144
|
+
`git restore --staged spec/* `
|
145
|
+
# Discard unstaged changes in spec folder
|
146
|
+
`git checkout spec -q`
|
147
|
+
`git clean spec -f -q`
|
148
|
+
local_sha = `git ls-tree HEAD #{project_root.join('spec')}`.chomp.split[2]
|
149
|
+
|
150
|
+
unless remote_sha == local_sha
|
151
|
+
find_or_create_directory("tmp")
|
152
|
+
find_or_create_directory("tmp/backup")
|
153
|
+
files_and_subfolders_inside_specs = Dir.glob("spec/*")
|
154
|
+
# Temporarily move specs
|
155
|
+
FileUtils.mv(files_and_subfolders_inside_specs, "tmp/backup")
|
156
|
+
|
157
|
+
download_file(repo_url, "tmp/spec.zip")
|
158
|
+
extracted_zip_folder = extract_zip("tmp/spec.zip", "tmp")
|
159
|
+
source_directory = extracted_zip_folder.join("spec")
|
160
|
+
overwrite_spec_folder(source_directory)
|
161
|
+
|
162
|
+
FileUtils.rm(project_root.join("tmp/spec.zip"))
|
163
|
+
FileUtils.rm_rf(extracted_zip_folder)
|
164
|
+
FileUtils.rm_rf("tmp/backup")
|
165
|
+
`git add spec/`
|
166
|
+
`git commit spec/ -m "Update spec/ folder to latest version" --author "First Draft <grades@firstdraft.com>"`
|
167
|
+
end
|
168
|
+
end
|
169
|
+
|
170
|
+
def download_file(url, destination)
|
171
|
+
download = URI.open(url)
|
172
|
+
IO.copy_stream(download, destination)
|
173
|
+
end
|
174
|
+
|
175
|
+
def extract_zip(folder, destination)
|
176
|
+
extracted_file_path = project_root.join(destination)
|
177
|
+
Zip::File.open(folder) do |zip_file|
|
178
|
+
zip_file.each_with_index do |file, index|
|
179
|
+
# Get name of root folder in zip file
|
180
|
+
if index == 0
|
181
|
+
extracted_file_path = extracted_file_path.join(file.name)
|
182
|
+
end
|
183
|
+
file_path = File.join(destination, file.name)
|
184
|
+
FileUtils.mkdir_p(File.dirname(file_path))
|
185
|
+
file.extract(file_path)
|
159
186
|
end
|
160
|
-
|
161
|
-
|
187
|
+
end
|
188
|
+
extracted_file_path
|
189
|
+
end
|
190
|
+
|
191
|
+
def overwrite_spec_folder(source_directory)
|
192
|
+
destination_directory = "spec"
|
193
|
+
# Get all files in the source directory
|
194
|
+
files = Dir.glob("#{source_directory}/*")
|
195
|
+
# Move each file to the destination directory
|
196
|
+
files.each do |file|
|
197
|
+
FileUtils.mv(file, destination_directory)
|
162
198
|
end
|
163
199
|
end
|
164
200
|
|
@@ -175,10 +211,10 @@ def update_config_file(config_file_name, config)
|
|
175
211
|
File.write(config_file_name, YAML.dump(config))
|
176
212
|
end
|
177
213
|
|
178
|
-
def
|
179
|
-
|
180
|
-
Dir.mkdir(
|
181
|
-
|
214
|
+
def find_or_create_directory(directory_name)
|
215
|
+
directory = File.join(project_root, directory_name)
|
216
|
+
Dir.mkdir(directory) unless Dir.exist?(directory)
|
217
|
+
directory
|
182
218
|
end
|
183
219
|
|
184
220
|
def is_valid_token?(root_url, token)
|
@@ -203,20 +239,29 @@ def upstream_repo(root_url, token)
|
|
203
239
|
res = Net::HTTP.start(uri.hostname, uri.port, use_ssl: true) do |http|
|
204
240
|
http.request(req)
|
205
241
|
end
|
206
|
-
|
207
|
-
result["repo_slug"]
|
242
|
+
Oj.load(res.body)
|
208
243
|
rescue => e
|
209
244
|
return false
|
210
245
|
end
|
211
246
|
|
212
|
-
def
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
247
|
+
def retrieve_github_username
|
248
|
+
config_dir_name = find_or_create_directory(".vscode")
|
249
|
+
config_file_name = "#{config_dir_name}/.ltici_apitoken.yml"
|
250
|
+
if File.exist?(config_file_name)
|
251
|
+
config = YAML.load_file(config_file_name)
|
252
|
+
if config["github_username"].present?
|
253
|
+
return config["github_username"]
|
254
|
+
end
|
255
|
+
else
|
256
|
+
github_email = `git config user.email`.chomp
|
257
|
+
return "" if github_email.blank?
|
258
|
+
username = `git config user.name`.chomp
|
259
|
+
search_results = Octokit.search_users("#{github_email} in:email").fetch(:items)
|
260
|
+
if search_results.present?
|
261
|
+
username = search_results.first.fetch(:login, username)
|
262
|
+
end
|
263
|
+
return username
|
218
264
|
end
|
219
|
-
username
|
220
265
|
end
|
221
266
|
|
222
267
|
def project_root
|
metadata
CHANGED
@@ -1,15 +1,15 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: grade_runner
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.0.
|
4
|
+
version: 0.0.13
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Raghu Betina
|
8
8
|
- Jelani Woods
|
9
|
-
autorequire:
|
9
|
+
autorequire:
|
10
10
|
bindir: bin
|
11
11
|
cert_chain: []
|
12
|
-
date:
|
12
|
+
date: 2025-03-18 00:00:00.000000000 Z
|
13
13
|
dependencies:
|
14
14
|
- !ruby/object:Gem::Dependency
|
15
15
|
name: activesupport
|
@@ -53,6 +53,20 @@ dependencies:
|
|
53
53
|
- - "~>"
|
54
54
|
- !ruby/object:Gem::Version
|
55
55
|
version: '5.0'
|
56
|
+
- !ruby/object:Gem::Dependency
|
57
|
+
name: zip
|
58
|
+
requirement: !ruby/object:Gem::Requirement
|
59
|
+
requirements:
|
60
|
+
- - ">="
|
61
|
+
- !ruby/object:Gem::Version
|
62
|
+
version: '0'
|
63
|
+
type: :runtime
|
64
|
+
prerelease: false
|
65
|
+
version_requirements: !ruby/object:Gem::Requirement
|
66
|
+
requirements:
|
67
|
+
- - ">="
|
68
|
+
- !ruby/object:Gem::Version
|
69
|
+
version: '0'
|
56
70
|
- !ruby/object:Gem::Dependency
|
57
71
|
name: faraday-retry
|
58
72
|
requirement: !ruby/object:Gem::Requirement
|
@@ -255,6 +269,8 @@ files:
|
|
255
269
|
- VERSION
|
256
270
|
- grade_runner.gemspec
|
257
271
|
- lib/grade_runner.rb
|
272
|
+
- lib/grade_runner/formatters/hint_formatter.rb
|
273
|
+
- lib/grade_runner/formatters/json_output_formatter.rb
|
258
274
|
- lib/grade_runner/railtie.rb
|
259
275
|
- lib/grade_runner/runner.rb
|
260
276
|
- lib/tasks/grade.rake
|
@@ -264,7 +280,7 @@ homepage: http://github.com/firstdraft/grade_runner
|
|
264
280
|
licenses:
|
265
281
|
- MIT
|
266
282
|
metadata: {}
|
267
|
-
post_install_message:
|
283
|
+
post_install_message:
|
268
284
|
rdoc_options: []
|
269
285
|
require_paths:
|
270
286
|
- lib
|
@@ -279,8 +295,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
279
295
|
- !ruby/object:Gem::Version
|
280
296
|
version: '0'
|
281
297
|
requirements: []
|
282
|
-
rubygems_version: 3.
|
283
|
-
signing_key:
|
298
|
+
rubygems_version: 3.1.6
|
299
|
+
signing_key:
|
284
300
|
specification_version: 4
|
285
301
|
summary: A Ruby client for [firstdraft Grades](https://grades.firstdraft.com)
|
286
302
|
test_files: []
|