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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b45590a9a30c53dfa993e986ce49f4bc33696f1a35f0e8587a4d9b54dad3f380
4
- data.tar.gz: 76e02456149f91f875c029add83bbdf1d31c7ce4b02fd9dd2729410911cf38b7
3
+ metadata.gz: 0f0b8b6d8c59749308e3c9dcd6ab96ae9db013bd388961e5ff5029a99a3d0e28
4
+ data.tar.gz: a1e4d49af97459f0b62b2b74379cbaf1ee21bbce80cf9ddda4562d41e63eaab9
5
5
  SHA512:
6
- metadata.gz: 25ccf695386404cf691960d213fed68402ba1269361af0ccf0a82a463d3dadb4065766e0380c6beaf83020eadf31241ee6b72a9041a339fd9930155959f3d664
7
- data.tar.gz: 31adbef7665e978f6292cbf8663f79bc77bc6828c76a8b9afeb740914d762cbefdf1a5c2f88b6d2495819169f9f1a689b355a110ae44597366983e0c6e84c311
6
+ metadata.gz: 8b005be1f8ee9db2d9385036e27233acbce5b5fc6d9eb0a6f875d76d5d7da095cf42d5dde54242b270efa2493ff30414d9f292263648d560e1f9305f0a23a2c5
7
+ data.tar.gz: c8b7c8026a6d880bca7021d36975f9c71eabcfd7375c4efde5821ff42840d045b333ab61c567b76aecdab61f58d2e48fac1199a9d095cbcddbde5d622bebdd93
data/Gemfile CHANGED
@@ -4,6 +4,7 @@ source "https://rubygems.org"
4
4
  gem "activesupport", ">= 2.3.5"
5
5
  gem "oj", "~> 3.13.12"
6
6
  gem "octokit", "~> 5.0"
7
+ gem "zip"
7
8
  gem "faraday-retry", "~> 1.0.3"
8
9
  gem "rake", "~> 13"
9
10
  # Add dependencies to develop your gem here.
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.11
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.11 ruby lib
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.11"
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 = "2024-07-09"
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 = find_or_create_config_dif
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
- full_reponame = upstream_repo(submission_url, token)
72
- set_upstream_remote(full_reponame)
73
- sync_specs_with_source(full_reponame)
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
- github_email = `git config user.email`.chomp
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 = find_or_create_config_dif
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
- if Octokit.repository?(full_reponame)
136
- repo_contents = Octokit.contents(full_reponame)
137
- remote_spec_folder = repo_contents.find { |git_object| git_object[:name] == 'spec' }
138
- if remote_spec_folder.blank?
139
- abort("The project #{full_reponame} does not have specs.")
140
- end
141
- remote_sha = remote_spec_folder[:sha]
142
- # Discard unstaged changes in spec folder
143
- `git checkout spec -q`
144
- `git clean spec -f -q`
145
- local_sha = `git ls-tree HEAD #{project_root.join('spec')}`.chomp.split[2]
146
-
147
- unless remote_sha == local_sha
148
- `git fetch upstream -q`
149
- # Remove local contents of spec folder
150
- `rm -rf spec/*`
151
- default_branch = `git remote show upstream | grep 'HEAD branch' | cut -d' ' -f5`.chomp
152
- # Overwrite local contents of spec folder with contents from upstream branch
153
- `git checkout upstream/#{default_branch} spec/ -q`
154
- # Unstage new spec file contents
155
- # - if wrong token is used, spec files can be removed properly when unstaged
156
- # - spec file changes committed by learner are removed and updated
157
- # - we are not committing spec file changes by default to avoid confusing the git history
158
- `git restore --staged spec/*`
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
- else
161
- abort("The project #{full_reponame} does not exist.")
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 find_or_create_config_dif
179
- config_dir_name = File.join(project_root, ".vscode")
180
- Dir.mkdir(config_dir_name) unless Dir.exist?(config_dir_name)
181
- config_dir_name
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
- result = Oj.load(res.body)
207
- result["repo_slug"]
242
+ Oj.load(res.body)
208
243
  rescue => e
209
244
  return false
210
245
  end
211
246
 
212
- def github_username(primary_email)
213
- return "" if primary_email.blank?
214
- username = `git config user.name`.chomp
215
- search_results = Octokit.search_users("#{primary_email} in:email").fetch(:items)
216
- if search_results.present?
217
- username = search_results.first.fetch(:login, username)
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.11
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: 2024-07-09 00:00:00.000000000 Z
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.4.6
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: []