learn-test 3.2.3 → 3.3.0.pre.4

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: d7d56bdb688e05fbb81c50ba01c290faff91f67647e04d70e837f2ca444555b1
4
- data.tar.gz: 4d4bf3e106248a1d57408d990ccbd6d0a17c48080db7de8fb971c91a16e557b9
3
+ metadata.gz: ce0ace7de8deb6d1ed1c75f04e3a4bab456ca7e359b04862ae3130c5055ec2b5
4
+ data.tar.gz: 1126e77a12912921eba8cdb78491a1061d25e5534cf9de3f51a8be0150d874ce
5
5
  SHA512:
6
- metadata.gz: e73cea00a3e2a7fbafaa01271aeb155de1775b88eb0e8392810bfeae485b8c84d644d8a1d1c70ddb2d3df7df9a79ec7a40dc9d48bf56675ca6bee28551a273c0
7
- data.tar.gz: ea17876778a49cc46a03ba853f39d1edde098f24f57e67606e9f632a5144c1bb94438f810073adad296b7ad4dfc09e750f7dbbcdb630bf5eb5cbec825f23395a
6
+ metadata.gz: d7bca7e41c472fbd244ffa1a4622958fb002f35e39fe5315ff10f3b78764d31f0061354661c73d2c85a0328380ffebfd212ed90d15d1b6d9796fa3276da27d78
7
+ data.tar.gz: cc8978a4410022a1f8089db3c17b87eec877234f9b91e138eb3c4e0c4043d169fc1686d419b71f40c35e5f605a5c5525ba54cd59b19234b0827a21bbba614a65
@@ -0,0 +1,27 @@
1
+ Layout/EmptyLinesAroundAttributeAccessor:
2
+ Enabled: true
3
+
4
+ Layout/FirstHashElementIndentation:
5
+ EnforcedStyle: consistent
6
+
7
+ Layout/SpaceAroundMethodCallOperator:
8
+ Enabled: true
9
+
10
+ Lint/RaiseException:
11
+ Enabled: true
12
+
13
+ Metrics/AbcSize:
14
+ Enabled: false
15
+
16
+ Metrics/BlockLength:
17
+ Enabled: false
18
+
19
+ Metrics/MethodLength:
20
+ Enabled: false
21
+
22
+ Metrics/PerceivedComplexity:
23
+ Enabled: false
24
+
25
+ # TODO: Add top-level module documentation
26
+ Style/Documentation:
27
+ Enabled: false
data/Gemfile CHANGED
@@ -3,7 +3,9 @@
3
3
  source 'https://rubygems.org'
4
4
 
5
5
  group :test do
6
+ gem 'aruba', '~> 1.0.3'
6
7
  gem 'rspec_junit_formatter', '~> 0.4.1'
8
+ gem 'simplecov', require: false
7
9
  end
8
10
 
9
11
  gemspec
@@ -29,4 +29,5 @@ Gem::Specification.new do |spec|
29
29
  spec.add_runtime_dependency "faraday", "~> 1.0"
30
30
  spec.add_runtime_dependency "crack", "~> 0.4.3"
31
31
  spec.add_runtime_dependency "colorize", "~> 0.8.1"
32
+ spec.add_runtime_dependency "zeitwerk", "~> 2.4.0"
32
33
  end
@@ -3,42 +3,23 @@
3
3
  require 'fileutils'
4
4
  require 'oj'
5
5
  require 'colorize'
6
+ require 'zeitwerk'
6
7
 
7
- require_relative 'learn_test/version'
8
- require_relative 'learn_test/netrc_interactor'
9
- require_relative 'learn_test/git_wip'
10
- require_relative 'learn_test/github_interactor'
11
- require_relative 'learn_test/user_id_parser'
12
- require_relative 'learn_test/username_parser'
13
- require_relative 'learn_test/learn_oauth_token_parser'
14
- require_relative 'learn_test/repo_parser'
15
- require_relative 'learn_test/file_finder'
16
- require_relative 'learn_test/runner'
17
-
18
- require_relative 'learn_test/dependency'
19
-
20
- require_relative 'learn_test/strategy'
21
- require_relative 'learn_test/js_strategy'
22
- require_relative 'learn_test/strategies/rspec'
23
- require_relative 'learn_test/strategies/karma'
24
- require_relative 'learn_test/strategies/protractor'
25
- require_relative 'learn_test/strategies/java_junit'
26
- require_relative 'learn_test/strategies/csharp_nunit'
27
- require_relative 'learn_test/strategies/mocha'
28
- require_relative 'learn_test/strategies/pytest'
29
- require_relative 'learn_test/strategies/none'
8
+ loader = Zeitwerk::Loader.for_gem
9
+ loader.inflector.inflect(
10
+ 'csharp' => 'CSharp',
11
+ 'csharp_nunit' => 'CSharpNunit',
12
+ 'nodejs' => 'NodeJS',
13
+ 'phantomjs' => 'PhantomJS'
14
+ )
15
+ loader.setup
30
16
 
31
17
  module LearnTest
32
- module Dependencies
33
- autoload :NodeJS, 'learn_test/dependencies/nodejs'
34
- autoload :PhantomJS, 'learn_test/dependencies/phantomjs'
35
- autoload :Karma, 'learn_test/dependencies/karma'
36
- autoload :Protractor, 'learn_test/dependencies/protractor'
37
- autoload :Java, 'learn_test/dependencies/java'
38
- autoload :CSharp, 'learn_test/dependencies/csharp'
39
- autoload :Ant, 'learn_test/dependencies/ant'
40
- autoload :Imagemagick, 'learn_test/dependencies/imagemagick'
41
- autoload :SeleniumServer, 'learn_test/dependencies/selenium_server'
42
- autoload :Pytest, 'learn_test/dependencies/pytest'
18
+ def self.root
19
+ File.dirname __dir__
20
+ end
21
+
22
+ def self.bin
23
+ File.join root, 'bin'
43
24
  end
44
25
  end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'git'
4
+
5
+ module LearnTest
6
+ module Git
7
+ def self.open(directory: './', options: {})
8
+ Base.open(directory, options)
9
+ end
10
+
11
+ class Base < ::Git::Base
12
+ def wip(message:)
13
+ wip = Wip::Base.new(base: self, message: message)
14
+ wip.process!
15
+ wip
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,88 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'fileutils'
4
+ require 'git'
5
+ require 'logger'
6
+ require 'tempfile'
7
+
8
+ module LearnTest
9
+ module Git
10
+ module Wip
11
+ class Base < ::Git::Path
12
+ TEMPFILE = '.wip'
13
+ PREFIX = 'refs/wip/'
14
+
15
+ attr_reader :working_branch, :wip_branch
16
+
17
+ def initialize(base:, message:)
18
+ @base = base
19
+ @message = message
20
+ @success = nil
21
+
22
+ current_branch = @base.current_branch
23
+
24
+ raise NoCommitsError, 'master' if current_branch.nil? # TODO: Swap to `main`?
25
+
26
+ @tmp = Tempfile.new(TEMPFILE)
27
+ @working_branch = Branch.new(base: @base, name: current_branch)
28
+ @wip_branch = Branch.new(base: @base, name: "#{PREFIX}#{current_branch}")
29
+ end
30
+
31
+ def process!
32
+ if @wip_branch.last_revision
33
+ merge = @base.merge_base(@wip_branch.last_revision, @working_branch.last_revision)
34
+
35
+ @wip_branch.parent = if merge == @working_branch.last_revision
36
+ @wip_branch.last_revision
37
+ else
38
+ @working_branch.last_revision
39
+ end
40
+ else
41
+ @wip_branch.parent = @working_branch.last_revision
42
+ end
43
+
44
+ new_tree = build_new_tree(@wip_branch.parent)
45
+ @base.diff(new_tree, @wip_branch.parent)
46
+
47
+ # tree_diff = @base.diff(new_tree, @wip_branch.parent)
48
+ # raise NoChangesError, @wip_branch if tree_diff.count.zero?
49
+
50
+ commit = @base.commit_tree(new_tree, parent: @wip_branch.parent)
51
+
52
+ @base.lib.send(:command, 'update-ref', ['-m', @message, @wip_branch, commit.objectish])
53
+
54
+ @success = true
55
+ ensure
56
+ cleanup
57
+ end
58
+
59
+ def success?
60
+ @success
61
+ end
62
+
63
+ private
64
+
65
+ def build_new_tree(wip_parent)
66
+ index = "#{@tmp.path}-index"
67
+
68
+ FileUtils.rm(index, force: true)
69
+ FileUtils.cp("#{@base.dir.path}/.git/index", index)
70
+
71
+ @base.read_tree(wip_parent)
72
+ @base.lib.send(:command, 'add', ['--update', '--', '.'])
73
+ @base.add(all: true)
74
+
75
+ new_tree_obj = @base.write_tree
76
+
77
+ FileUtils.rm(index, force: true)
78
+
79
+ new_tree_obj
80
+ end
81
+
82
+ def cleanup
83
+ FileUtils.rm("#{@tmp.path}-*", force: true)
84
+ end
85
+ end
86
+ end
87
+ end
88
+ end
@@ -0,0 +1,36 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LearnTest
4
+ module Git
5
+ module Wip
6
+ class Branch
7
+ attr_accessor :parent
8
+
9
+ def initialize(base:, name:)
10
+ @base = base
11
+ @name = name
12
+ end
13
+
14
+ def last_revision(raise_no_commits: false)
15
+ @last_revision ||= begin
16
+ begin
17
+ @base.revparse(@name)
18
+ rescue ::Git::GitExecuteError => e
19
+ if raise_no_commits
20
+ raise e.message.match(NoCommitsError::REGEX) ? NoCommitsError.new(@name) : e
21
+ end
22
+
23
+ raise unless e.message.match(NoCommitsError::REGEX)
24
+
25
+ false
26
+ end
27
+ end
28
+ end
29
+
30
+ def to_s
31
+ @name
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LearnTest
4
+ module Git
5
+ module Wip
6
+ class Error < StandardError; end
7
+
8
+ class NoChangesError < Error
9
+ def initialize(branch)
10
+ super "No changes found on `#{branch}`"
11
+ end
12
+ end
13
+
14
+ class NoCommitsError < Error
15
+ REGEX = /unknown revision or path not in the working tree/i.freeze
16
+
17
+ def initialize(branch)
18
+ super "Branch `#{branch}` doesn't have any commits. Please commit and try again."
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -6,7 +6,7 @@ module LearnTest
6
6
  class RepoParser
7
7
  def self.get_repo
8
8
  begin
9
- repo = Git.open(FileUtils.pwd)
9
+ repo = ::Git.open(FileUtils.pwd)
10
10
  rescue
11
11
  puts "You don't appear to be in a Learn lesson's directory. Please enter 'learn open' or cd to an appropriate directory and try again."
12
12
  die
@@ -3,9 +3,6 @@
3
3
  require 'fileutils'
4
4
  require 'json'
5
5
 
6
- require_relative 'client'
7
- require_relative 'git_wip'
8
-
9
6
  module LearnTest
10
7
  class Reporter
11
8
  attr_accessor :output_path, :debug
@@ -50,16 +47,30 @@ module LearnTest
50
47
  endpoint = strategy.service_endpoint
51
48
  augment_results!(results)
52
49
 
53
- if client.post_results(endpoint, results)
54
- if !LearnTest::GitWip.run! && @debug
55
- puts 'There was a problem connecting to Github. Not pushing current branch state.'.red
56
- end
57
- else
58
- if @debug
59
- puts 'There was a problem connecting to Learn. Not pushing test results.'.red
60
- end
50
+ unless client.post_results(endpoint, results)
51
+ puts 'There was a problem connecting to Learn. Not pushing test results.'.red if @debug
61
52
 
62
53
  save_failed_attempt(endpoint, results)
54
+ return
55
+ end
56
+
57
+ logger = @debug ? Logger.new(STDOUT, level: Logger::DEBUG) : false
58
+ repo = LearnTest::Git.open(options: { log: logger })
59
+
60
+ res = repo.wip(message: 'Automatic test submission')
61
+
62
+ unless res.success?
63
+ puts 'There was a problem creating your WIP branch. Not pushing current branch state.'.red if @debug
64
+ return
65
+ end
66
+
67
+ begin
68
+ repo.push('origin', "#{res.wip_branch}:refs/heads/fis-wip", { force: true })
69
+ rescue ::Git::GitExecuteError => e
70
+ if @debug
71
+ puts 'There was a problem connecting to GitHub. Not pushing current branch state.'.red
72
+ puts e.message
73
+ end
63
74
  end
64
75
  end
65
76
 
@@ -89,7 +100,7 @@ module LearnTest
89
100
 
90
101
  def augment_results!(results)
91
102
  if File.exist?("#{FileUtils.pwd}/.learn")
92
- dot_learn = YAML.load(File.read("#{FileUtils.pwd}/.learn"))
103
+ dot_learn = YAML.safe_load(File.read("#{FileUtils.pwd}/.learn"))
93
104
 
94
105
  unless dot_learn['github'].nil?
95
106
  results[:github] = dot_learn['github']
@@ -9,7 +9,6 @@ module LearnTest
9
9
  def initialize(repo, options = {})
10
10
  @repo = repo
11
11
  @options = options
12
- die unless strategy
13
12
  end
14
13
 
15
14
  def run
@@ -34,7 +33,11 @@ module LearnTest
34
33
  end
35
34
 
36
35
  def strategy
37
- @strategy ||= strategies.map { |s| s.new(self) }.detect(&:detect)
36
+ return @strategy if @strategy
37
+
38
+ detected = strategies.map { |s| s.new(self) }.detect(&:detect)
39
+
40
+ @strategy = detected || LearnTest::Strategies::None.new(self)
38
41
  end
39
42
 
40
43
  private
@@ -57,8 +60,7 @@ module LearnTest
57
60
  LearnTest::Strategies::Protractor,
58
61
  LearnTest::Strategies::JavaJunit,
59
62
  LearnTest::Strategies::Mocha,
60
- LearnTest::Strategies::Pytest,
61
- LearnTest::Strategies::None
63
+ LearnTest::Strategies::Pytest
62
64
  ]
63
65
  end
64
66
 
@@ -69,10 +71,5 @@ module LearnTest
69
71
  def local_test_run?
70
72
  options.include?('-h') || options.include?('--local')
71
73
  end
72
-
73
- def die
74
- puts "This directory doesn't appear to have any specs in it."
75
- exit
76
- end
77
74
  end
78
75
  end
@@ -7,12 +7,12 @@ module LearnTest
7
7
  '/e/flatiron_none'
8
8
  end
9
9
 
10
- def detect
11
- runner.files.include?('.canvas')
12
- end
13
-
14
10
  def run
15
- puts 'Your assignment was submitted. You can resubmit by running `learn test` again.'
11
+ puts <<~MSG
12
+ This directory doesn't appear to have any specs in it, so there’s no test to run.
13
+
14
+ If you are working on Canvas, this assignment has been submitted. You can resubmit by running `learn test` again.
15
+ MSG
16
16
  end
17
17
 
18
18
  def results
@@ -63,7 +63,7 @@ module LearnTest
63
63
  if npm_install
64
64
  system(command)
65
65
  else
66
- Open3.popen3(command) do |stdin, stdout, stderr, wait_thr|
66
+ Open3.popen3(command) do |_, stdout, stderr, wait_thr|
67
67
  while out = stdout.gets do
68
68
  puts out
69
69
  end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module LearnTest
4
- VERSION = '3.2.3'
4
+ VERSION = '3.3.0.pre.4'
5
5
  end
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- describe 'Running a RSpec Unit Test' do
4
- before(:all) do
3
+ describe 'Running a RSpec Unit Test', type: :aruba do
4
+ before :all do
5
5
  # While it doesn't cause these tests to fail, nasty messages occur (and more)
6
6
  # when either a ~/.netrc entry or file itself doesn't exist. This aims to correct that,
7
7
  # and will only ever be called once.
@@ -9,9 +9,23 @@ describe 'Running a RSpec Unit Test' do
9
9
  LearnTest::UsernameParser.get_username
10
10
  end
11
11
 
12
+ before :each do
13
+ copy '%/rspec-unit-spec', 'example'
14
+ cd 'example'
15
+
16
+ git_init
17
+ git_add
18
+ git_commit 'Initial Commit'
19
+ end
20
+
21
+ def run(flags = '')
22
+ run_command_and_stop("#{File.join(LearnTest.bin, 'learn-test')} #{flags}")
23
+ last_command_started.output
24
+ end
25
+
12
26
  context 'a basic rspec unit test' do
13
27
  it 'runs the spec with 0 failures' do
14
- output = `cd ./spec/fixtures/rspec-unit-spec && ./../../../bin/learn-test --local --test`
28
+ output = run('--local --test')
15
29
 
16
30
  expect(output).to include('3 examples, 0 failures')
17
31
  expect(output).to_not include('1 failures')
@@ -20,14 +34,14 @@ describe 'Running a RSpec Unit Test' do
20
34
 
21
35
  context 'with the --example flag' do
22
36
  it 'runs only the appropriate tests' do
23
- output = `cd ./spec/fixtures/rspec-unit-spec && ./../../../bin/learn-test --local --test --example multiple`
37
+ output = run('--local --test --example multiple')
24
38
 
25
39
  expect(output).to include('1 example, 0 failures')
26
40
  expect(output).to_not include('2 examples')
27
41
  end
28
42
 
29
- it 'accepts multiple examples' do
30
- output = `cd ./spec/fixtures/rspec-unit-spec && ./../../../bin/learn-test --local --test --example multiple --example accepts`
43
+ it 'accepts multiple examples' do
44
+ output = run(' --local --test --example multiple --example accepts')
31
45
 
32
46
  expect(output).to include('2 examples, 0 failures')
33
47
  expect(output).to_not include('3 examples')