gurney_client 0.3.0 → 0.5.0
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/CHANGELOG.md +32 -3
- data/Gemfile.lock +10 -8
- data/README.md +30 -21
- data/lib/gurney/api.rb +6 -5
- data/lib/gurney/cli.rb +77 -90
- data/lib/gurney/dependency_collector.rb +64 -0
- data/lib/gurney/git_file_reader.rb +23 -0
- data/lib/gurney/source/npm.rb +58 -0
- data/lib/gurney/source/pnpm.rb +54 -0
- data/lib/gurney/version.rb +1 -1
- data/lib/gurney.rb +4 -0
- metadata +11 -7
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 1097c1c98d3799765d40e6a5263a1fe58baa6a4df542570f91654247218b7d64
|
4
|
+
data.tar.gz: 953fe191235e0c338cb71a690284ff5beed7e36f26a511fe1cb3f6311493ef82
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 41cd13cf0b65dca7b568cd00f2a3dc95045ae3a6cdb8e896b24e9dcc755efc21fd24d3a328388604d3e783ce2a64f1dfe864d17f745df6ae957f032a4a1131b5
|
7
|
+
data.tar.gz: b00a53d0aa50995c7be657090d50c906e5914cf5ee1dd5d3b14c74679400f356b49ded4fc653c525ca962d1a2755ceba12be27f32c0361b358c6011d19477cad
|
data/CHANGELOG.md
CHANGED
@@ -1,8 +1,37 @@
|
|
1
|
-
#
|
1
|
+
# Changelog
|
2
2
|
|
3
|
-
|
3
|
+
All notable changes to this project will be documented in this file.
|
4
|
+
This project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html).
|
5
|
+
|
6
|
+
## Unreleased
|
7
|
+
|
8
|
+
### Compatible changes
|
9
|
+
|
10
|
+
### Breaking changes
|
11
|
+
|
12
|
+
|
13
|
+
## 0.5.0 2025-03-26
|
14
|
+
|
15
|
+
### Compatible changes
|
16
|
+
* Dependencies are also parsed from package-lock.json and pnpm-lock.yaml if present.
|
17
|
+
|
18
|
+
|
19
|
+
## 0.4.0 2024-11-15
|
20
|
+
|
21
|
+
### Compatible changes
|
22
|
+
* Added: Reporting of the repository path as identifier. Should it ever change,
|
23
|
+
it is an indicator for an unchanged gurney.yml in a project fork, and the
|
24
|
+
API may respond with an error.
|
25
|
+
|
26
|
+
|
27
|
+
## 0.3.0 2024-11-14
|
28
|
+
|
29
|
+
### Compatible changes
|
4
30
|
* Added: Compatibility with Ruby 3
|
5
31
|
* Fixed: Support UTF-8 chars in branch names
|
6
32
|
|
7
|
-
|
33
|
+
|
34
|
+
## 0.2.3 2023-05-24
|
35
|
+
|
36
|
+
### Compatible changes
|
8
37
|
* Added: Suppress Bundler's "the Bundler version is older than the one in the lockfile" warning.
|
data/Gemfile.lock
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
gurney_client (0.
|
4
|
+
gurney_client (0.5.0)
|
5
5
|
bundler (< 3)
|
6
6
|
colorize (~> 0.8)
|
7
7
|
git (~> 1.5)
|
@@ -10,22 +10,24 @@ PATH
|
|
10
10
|
GEM
|
11
11
|
remote: https://rubygems.org/
|
12
12
|
specs:
|
13
|
-
addressable (2.8.
|
14
|
-
public_suffix (>= 2.0.2, <
|
13
|
+
addressable (2.8.7)
|
14
|
+
public_suffix (>= 2.0.2, < 7.0)
|
15
15
|
byebug (11.0.1)
|
16
16
|
colorize (0.8.1)
|
17
17
|
diff-lcs (1.3)
|
18
|
-
git (1.
|
18
|
+
git (1.19.1)
|
19
19
|
addressable (~> 2.8)
|
20
20
|
rchardet (~> 1.8)
|
21
21
|
httparty (0.17.3)
|
22
22
|
mime-types (~> 3.0)
|
23
23
|
multi_xml (>= 0.5.2)
|
24
|
-
|
24
|
+
logger (1.6.1)
|
25
|
+
mime-types (3.6.0)
|
26
|
+
logger
|
25
27
|
mime-types-data (~> 3.2015)
|
26
|
-
mime-types-data (3.
|
28
|
+
mime-types-data (3.2024.1105)
|
27
29
|
multi_xml (0.6.0)
|
28
|
-
public_suffix (5.
|
30
|
+
public_suffix (5.1.1)
|
29
31
|
rake (13.0.1)
|
30
32
|
rchardet (1.8.0)
|
31
33
|
rspec (3.9.0)
|
@@ -52,4 +54,4 @@ DEPENDENCIES
|
|
52
54
|
rspec
|
53
55
|
|
54
56
|
BUNDLED WITH
|
55
|
-
2.
|
57
|
+
2.3.19
|
data/README.md
CHANGED
@@ -1,30 +1,31 @@
|
|
1
|
-
|
1
|
+
# Gurney
|
2
2
|
|
3
|
-
Gurney is a small tool to extract dependencies from project files and report
|
4
|
-
|
3
|
+
Gurney is a small tool to extract dependencies from project files and report
|
4
|
+
them to a web API. Modes:
|
5
|
+
- normal
|
6
|
+
- local pre-push hook
|
7
|
+
- remote post-receive hook
|
8
|
+
Usually, we configure the latter on our Git server to automatically run on each
|
9
|
+
push, with the API url passed as command line option.
|
5
10
|
|
6
|
-
When run as a
|
7
|
-
|
8
|
-
|
11
|
+
When run as a post-receive hook, Gurney will make a bare copy of the project and
|
12
|
+
look for a gurney.yml file. If present, Gurney looks at the configured branches
|
13
|
+
and collects their dependencies. These are reported to the web API.
|
9
14
|
|
10
|
-
|
15
|
+
## Usage
|
11
16
|
```
|
12
17
|
Usage: gurney [options]
|
13
|
-
--api-url [API URL]
|
14
|
-
|
15
|
-
|
16
|
-
--api-token [API TOKEN]
|
17
|
-
Token to be send to the api in the X-AuthToken header
|
18
|
+
--api-url [API URL] Url for web API call, can have parameters for <project_id> and <branch>
|
19
|
+
Example: --api-url "https://example.com/project/<project_id>/branch/<branch>"
|
20
|
+
--api-token [API TOKEN] Token to be sent to the API in the X-AuthToken header
|
18
21
|
-c, --config [CONFIG FILE] Config file to use
|
19
|
-
-h, --hook Run as a
|
20
|
-
--client-hook
|
21
|
-
|
22
|
-
|
23
|
-
--help
|
24
|
-
Prints this help
|
22
|
+
-h, --hook Run as a Git post-receive hook
|
23
|
+
--client-hook Run as a Git pre-push hook
|
24
|
+
-p, --project-id [PROJECT ID] Specify project id for API
|
25
|
+
--help Print this help
|
25
26
|
```
|
26
27
|
|
27
|
-
|
28
|
+
## Sample Config
|
28
29
|
```yaml
|
29
30
|
project_id: 1
|
30
31
|
branches:
|
@@ -34,5 +35,13 @@ api_url: http://example.com/dep_reporter/project/<project_id>/branch/<branch>
|
|
34
35
|
api_token: 1234567890
|
35
36
|
```
|
36
37
|
|
37
|
-
|
38
|
-
|
38
|
+
## Running as a global Git hook
|
39
|
+
See https://docs.gitlab.com/ee/administration/server_hooks.html#create-global-server-hooks-for-all-repositories
|
40
|
+
|
41
|
+
## Development
|
42
|
+
You can run Gurney locally from another directory like this:
|
43
|
+
|
44
|
+
```bash
|
45
|
+
cd some/real/project
|
46
|
+
ruby -I path/to/gurney/lib path/to/gurney/exe/gurney
|
47
|
+
```
|
data/lib/gurney/api.rb
CHANGED
@@ -9,13 +9,14 @@ module Gurney
|
|
9
9
|
@token = token
|
10
10
|
end
|
11
11
|
|
12
|
-
def post_dependencies(dependencies:, branch:, project_id:)
|
13
|
-
data = {
|
14
|
-
|
15
|
-
|
12
|
+
def post_dependencies(dependencies:, branch:, project_id:, repo_path: nil)
|
13
|
+
data = { dependencies: dependencies }
|
14
|
+
data[:repository_path] = repo_path if repo_path
|
15
|
+
|
16
16
|
url = base_url
|
17
17
|
url.gsub! '<project_id>', CGI.escape(project_id)
|
18
18
|
url.gsub! '<branch>', CGI.escape(branch)
|
19
|
+
|
19
20
|
post_json(url, data.to_json)
|
20
21
|
end
|
21
22
|
|
@@ -31,7 +32,7 @@ module Gurney
|
|
31
32
|
)
|
32
33
|
unless response.success?
|
33
34
|
if response.code == 404
|
34
|
-
raise ApiError.new("#{response.code}
|
35
|
+
raise ApiError.new("#{response.code} API url is probably wrong")
|
35
36
|
else
|
36
37
|
raise ApiError.new("#{response.code} #{response.body}")
|
37
38
|
end
|
data/lib/gurney/cli.rb
CHANGED
@@ -6,118 +6,105 @@ require 'git'
|
|
6
6
|
require 'fileutils'
|
7
7
|
|
8
8
|
module Gurney
|
9
|
+
|
10
|
+
class Error < StandardError; end
|
11
|
+
|
9
12
|
class CLI
|
10
13
|
HOOK_STDIN_REGEX = /(?<old>[0-9a-f]{40}) (?<new>[0-9a-f]{40}) refs\/heads\/(?<ref>\w+)/m
|
11
14
|
CLIENT_HOOK_STDIN_REGEX = /refs\/heads\/(?<ref>\w+) (?<new>[0-9a-f]{40}) refs\/heads\/(?<remote_ref>\w+) (?<remote_sha>[0-9a-f]{40})/m
|
12
15
|
MAIN_BRANCHES = ['master', 'main'].freeze
|
13
16
|
|
14
17
|
def self.run(cmd_parameter=[])
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
file = read_file(g, options.hook, branch, options.config_file)
|
28
|
-
break file if file
|
29
|
-
end
|
30
|
-
if !config_file && options.hook
|
31
|
-
# dont run as a hook with no config
|
32
|
-
exit 0
|
33
|
-
end
|
34
|
-
config_file ||= '---'
|
35
|
-
config = Gurney::Config.from_yaml(config_file)
|
36
|
-
|
37
|
-
options.branches ||= config&.branches
|
38
|
-
options.branches ||= config&.branches
|
39
|
-
options.api_token ||= config&.api_token
|
40
|
-
options.api_url ||= config&.api_url
|
41
|
-
options.project_id ||= config&.project_id
|
42
|
-
|
43
|
-
if [options.project_id, options.branches, options.api_url, options.api_token].any?(&:nil?)
|
44
|
-
raise Gurney::Error.new("Either provide in a config file or set the flags for project id, branches, api url and api token")
|
45
|
-
end
|
46
|
-
|
47
|
-
branches = []
|
48
|
-
if options.hook || options.client_hook
|
49
|
-
# we get passed changed branches and refs via stdin
|
50
|
-
$stdin.each_line do |line|
|
51
|
-
regex = options.client_hook ? CLIENT_HOOK_STDIN_REGEX : HOOK_STDIN_REGEX
|
52
|
-
line.force_encoding(Encoding::UTF_8)
|
53
|
-
matches = line.match(regex)
|
54
|
-
if matches && matches[:new] != '0' * 40
|
55
|
-
if options.branches.include? matches[:ref]
|
56
|
-
branches << matches[:ref]
|
57
|
-
end
|
58
|
-
end
|
59
|
-
end
|
18
|
+
new(cmd_parameter).run
|
19
|
+
rescue SystemExit
|
20
|
+
# Do nothing
|
21
|
+
rescue Gurney::ApiError => e
|
22
|
+
puts "Gurney API error".red
|
23
|
+
puts e.message.red
|
24
|
+
rescue Gurney::Error => e
|
25
|
+
puts "Gurney error: #{e.message}".red
|
26
|
+
rescue Exception
|
27
|
+
puts "Gurney: an unexpected error occurred".red
|
28
|
+
raise
|
29
|
+
end
|
60
30
|
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
31
|
+
def initialize(cmd_parameter=[])
|
32
|
+
@options = Gurney::CLI::OptionParser.parse(cmd_parameter)
|
33
|
+
@git = if options.hook
|
34
|
+
Git.bare(ENV['GIT_DIR'] || Dir.pwd)
|
35
|
+
else
|
36
|
+
unless Dir.exist? './.git'
|
37
|
+
raise Gurney::Error.new('Must be run within a git repository')
|
67
38
|
end
|
39
|
+
Git.open('.')
|
40
|
+
end
|
68
41
|
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
42
|
+
config_file = MAIN_BRANCHES.find do |branch|
|
43
|
+
git_file_reader = GitFileReader.new(git, branch, read_from_git: options.hook)
|
44
|
+
file = git_file_reader.read(options.config_file)
|
45
|
+
break file if file
|
46
|
+
end
|
47
|
+
if options.hook && !config_file
|
48
|
+
# Git hooks are activated by the config file. Without, do nothing.
|
49
|
+
exit 0
|
50
|
+
end
|
51
|
+
config_file ||= '---'
|
52
|
+
config = Gurney::Config.from_yaml(config_file)
|
53
|
+
|
54
|
+
options.branches ||= config&.branches
|
55
|
+
options.branches ||= config&.branches
|
56
|
+
options.api_token ||= config&.api_token
|
57
|
+
options.api_url ||= config&.api_url
|
58
|
+
options.project_id ||= config&.project_id
|
59
|
+
|
60
|
+
missing_options = [:project_id, :branches, :api_url, :api_token].select { |option| options.send(option).nil? }
|
61
|
+
# Use the line below in development
|
62
|
+
# missing_options = [:project_id, :branches, :api_token].select { |option| options.send(option).nil? }
|
63
|
+
raise Gurney::Error.new("Incomplete config - missing #{missing_options.map(&:inspect).join(', ')}.") unless missing_options.empty?
|
64
|
+
end
|
82
65
|
|
83
|
-
|
84
|
-
|
66
|
+
def run
|
67
|
+
reporting_branches.each do |branch|
|
68
|
+
git_file_reader = GitFileReader.new(git, branch, read_from_git: options.hook || options.client_hook)
|
69
|
+
dependencies = DependencyCollector.new(git_file_reader).collect_all
|
85
70
|
|
86
|
-
|
87
|
-
|
88
|
-
end
|
71
|
+
api = Gurney::Api.new(base_url: options.api_url, token: options.api_token)
|
72
|
+
api.post_dependencies(dependencies: dependencies, branch: branch, project_id: options.project_id, repo_path: git.repo.path)
|
89
73
|
|
90
|
-
|
91
|
-
|
92
|
-
puts "Gurney: api error".red
|
93
|
-
puts e.message.red
|
94
|
-
rescue Gurney::Error => e
|
95
|
-
puts "Gurney: error".red
|
96
|
-
puts e.message.red
|
97
|
-
rescue Exception => e
|
98
|
-
puts "Gurney: an unexpected error occurred".red
|
99
|
-
raise
|
74
|
+
dependency_counts = dependencies.group_by(&:ecosystem).map{|ecosystem, dependencies| "#{ecosystem}: #{dependencies.count}" }.join(', ')
|
75
|
+
puts "Gurney: reported dependencies (#{dependency_counts})"
|
100
76
|
end
|
101
77
|
end
|
102
78
|
|
103
79
|
private
|
104
80
|
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
81
|
+
attr_accessor :git, :options
|
82
|
+
|
83
|
+
def reporting_branches
|
84
|
+
branches = []
|
85
|
+
if options.hook || options.client_hook
|
86
|
+
# We get changed branches and refs via stdin
|
87
|
+
# See https://git-scm.com/docs/githooks#post-receive
|
88
|
+
$stdin.each_line do |line|
|
89
|
+
regex = options.client_hook ? CLIENT_HOOK_STDIN_REGEX : HOOK_STDIN_REGEX
|
90
|
+
line.force_encoding(Encoding::UTF_8)
|
91
|
+
matches = line.match(regex)
|
92
|
+
if matches && matches[:new] != '0' * 40
|
93
|
+
if options.branches.include? matches[:ref]
|
94
|
+
branches << matches[:ref]
|
95
|
+
end
|
96
|
+
end
|
111
97
|
end
|
112
98
|
else
|
113
|
-
|
114
|
-
|
99
|
+
current_branch = git.current_branch
|
100
|
+
unless options.branches.nil? || options.branches.include?(current_branch)
|
101
|
+
raise Gurney::Error.new('The current branch is not specified in the config.')
|
115
102
|
end
|
103
|
+
branches << current_branch
|
116
104
|
end
|
117
|
-
end
|
118
105
|
|
119
|
-
|
106
|
+
branches
|
107
|
+
end
|
120
108
|
|
121
|
-
class Error < Exception
|
122
109
|
end
|
123
110
|
end
|
@@ -0,0 +1,64 @@
|
|
1
|
+
module Gurney
|
2
|
+
class DependencyCollector
|
3
|
+
|
4
|
+
def initialize(git_file_reader)
|
5
|
+
@git_file_reader = git_file_reader
|
6
|
+
end
|
7
|
+
|
8
|
+
def collect_all
|
9
|
+
dependencies = []
|
10
|
+
|
11
|
+
dependencies.concat npm_dependencies
|
12
|
+
dependencies.concat bundler_dependencies
|
13
|
+
dependencies.concat ruby_version_dependencies
|
14
|
+
|
15
|
+
dependencies.compact
|
16
|
+
end
|
17
|
+
|
18
|
+
private
|
19
|
+
|
20
|
+
def bundler_dependencies
|
21
|
+
bundler_source = Gurney::Source::Bundler.new(gemfile_lock: @git_file_reader.read('Gemfile.lock'))
|
22
|
+
bundler_source.dependencies || []
|
23
|
+
end
|
24
|
+
|
25
|
+
def ruby_version_dependencies
|
26
|
+
ruby_version_source = Gurney::Source::RubyVersion.new(ruby_version: @git_file_reader.read('.ruby-version'))
|
27
|
+
ruby_version_source.dependencies || []
|
28
|
+
end
|
29
|
+
|
30
|
+
def npm_dependencies
|
31
|
+
npm_dependencies = []
|
32
|
+
|
33
|
+
if yarn_lock
|
34
|
+
yarn_source = Gurney::Source::Yarn.new(yarn_lock: yarn_lock)
|
35
|
+
npm_dependencies.concat(yarn_source.dependencies || [])
|
36
|
+
end
|
37
|
+
|
38
|
+
if package_lock_json
|
39
|
+
npm_source = Gurney::Source::Npm.new(package_lock_json: package_lock_json)
|
40
|
+
npm_dependencies.concat(npm_source.dependencies || [])
|
41
|
+
end
|
42
|
+
|
43
|
+
if pnpm_lock
|
44
|
+
pnpm_source = Gurney::Source::Pnpm.new(pnpm_lock: pnpm_lock)
|
45
|
+
npm_dependencies.concat(pnpm_source.dependencies || [])
|
46
|
+
end
|
47
|
+
|
48
|
+
npm_dependencies
|
49
|
+
end
|
50
|
+
|
51
|
+
def yarn_lock
|
52
|
+
@yarn_lock ||= @git_file_reader.read('yarn.lock')
|
53
|
+
end
|
54
|
+
|
55
|
+
def package_lock_json
|
56
|
+
@package_lock_json = @git_file_reader.read('package-lock.json')
|
57
|
+
end
|
58
|
+
|
59
|
+
def pnpm_lock
|
60
|
+
@pnpm_lock = @git_file_reader.read('pnpm-lock.yaml')
|
61
|
+
end
|
62
|
+
|
63
|
+
end
|
64
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module Gurney
|
2
|
+
class GitFileReader
|
3
|
+
|
4
|
+
def initialize(git, branch, read_from_git:)
|
5
|
+
@git = git
|
6
|
+
@branch = branch
|
7
|
+
@read_from_git = read_from_git
|
8
|
+
end
|
9
|
+
|
10
|
+
def read(filename)
|
11
|
+
if @read_from_git
|
12
|
+
begin
|
13
|
+
@git.show("#{@branch}:#{filename}")
|
14
|
+
rescue Git::GitExecuteError
|
15
|
+
# happens if branch does not exist
|
16
|
+
end
|
17
|
+
else
|
18
|
+
File.read(filename) if File.exist?(filename)
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,58 @@
|
|
1
|
+
require 'json'
|
2
|
+
require 'colorize'
|
3
|
+
|
4
|
+
module Gurney
|
5
|
+
module Source
|
6
|
+
class Npm < Base
|
7
|
+
|
8
|
+
SUPPORTED_LOCKFILE_VERSIONS = [2, 3].freeze
|
9
|
+
|
10
|
+
def initialize(package_lock_json:)
|
11
|
+
@package_lock_json = package_lock_json
|
12
|
+
end
|
13
|
+
|
14
|
+
def present?
|
15
|
+
!@package_lock_json&.empty?
|
16
|
+
end
|
17
|
+
|
18
|
+
def dependencies
|
19
|
+
if present?
|
20
|
+
parsed_lock = JSON.parse(@package_lock_json)
|
21
|
+
|
22
|
+
if SUPPORTED_LOCKFILE_VERSIONS.include?(parsed_lock['lockfileVersion'])
|
23
|
+
extract_dependencies(parsed_lock)
|
24
|
+
else
|
25
|
+
puts "package-lock.json: Lockfile version #{parsed_lock['lockfileVersion']} is unsupported. No npm dependencies reported.".yellow
|
26
|
+
[]
|
27
|
+
end
|
28
|
+
end
|
29
|
+
rescue JSON::ParserError => e
|
30
|
+
raise Gurney::Error.new("Invalid package-lock.json format: #{e.message}")
|
31
|
+
end
|
32
|
+
|
33
|
+
private
|
34
|
+
|
35
|
+
attr_reader :package_lock_json
|
36
|
+
|
37
|
+
def extract_dependencies(parsed_lock)
|
38
|
+
dependencies = []
|
39
|
+
|
40
|
+
if parsed_lock['packages']
|
41
|
+
parsed_lock['packages'].each do |path_to_package, details|
|
42
|
+
next if path_to_package == ''
|
43
|
+
|
44
|
+
name = path_to_package.sub(/^node_modules\//, '') # remove "node_modules/" prefix to get package name
|
45
|
+
dependencies << Dependency.new(
|
46
|
+
ecosystem: 'npm',
|
47
|
+
name: name,
|
48
|
+
version: details['version']
|
49
|
+
)
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
dependencies
|
54
|
+
end
|
55
|
+
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -0,0 +1,54 @@
|
|
1
|
+
require 'yaml'
|
2
|
+
require 'colorize'
|
3
|
+
|
4
|
+
module Gurney
|
5
|
+
module Source
|
6
|
+
class Pnpm < Base
|
7
|
+
def initialize(pnpm_lock:)
|
8
|
+
@pnpm_lock = pnpm_lock
|
9
|
+
end
|
10
|
+
|
11
|
+
def present?
|
12
|
+
!@pnpm_lock&.empty?
|
13
|
+
end
|
14
|
+
|
15
|
+
def dependencies
|
16
|
+
if present?
|
17
|
+
parsed_lock = YAML.safe_load(@pnpm_lock)
|
18
|
+
|
19
|
+
major_version = parsed_lock['lockfileVersion'].split('.').first
|
20
|
+
if major_version == '9'
|
21
|
+
extract_dependencies(parsed_lock)
|
22
|
+
else
|
23
|
+
puts "pnpm-lock.yaml: Lockfile version #{major_version} is unsupported. No npm dependencies reported.".yellow
|
24
|
+
[]
|
25
|
+
end
|
26
|
+
end
|
27
|
+
rescue Psych::SyntaxError => e
|
28
|
+
raise Gurney::Error.new("Invalid pnpm-lock.yaml format: #{e.message}")
|
29
|
+
end
|
30
|
+
|
31
|
+
private
|
32
|
+
|
33
|
+
attr_reader :pnpm_lock
|
34
|
+
|
35
|
+
def extract_dependencies(parsed_lock)
|
36
|
+
dependencies = []
|
37
|
+
|
38
|
+
# dependency_id has format <scoped_pkg_name>@<pkg_version>
|
39
|
+
# see https://github.com/pnpm/spec/blob/master/lockfile/9.0.md#packages
|
40
|
+
parsed_lock['packages'].each_key do |dependency_id|
|
41
|
+
name, _, version = dependency_id.rpartition('@')
|
42
|
+
dependencies << Dependency.new(
|
43
|
+
ecosystem: 'npm',
|
44
|
+
name: name,
|
45
|
+
version: version
|
46
|
+
)
|
47
|
+
end
|
48
|
+
|
49
|
+
dependencies
|
50
|
+
end
|
51
|
+
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
data/lib/gurney/version.rb
CHANGED
data/lib/gurney.rb
CHANGED
@@ -1,8 +1,12 @@
|
|
1
1
|
require_relative 'gurney/version'
|
2
2
|
require_relative 'gurney/config'
|
3
|
+
require_relative 'gurney/git_file_reader'
|
3
4
|
require_relative 'gurney/dependency'
|
5
|
+
require_relative 'gurney/dependency_collector'
|
4
6
|
require_relative 'gurney/source/base'
|
5
7
|
require_relative 'gurney/source/yarn'
|
8
|
+
require_relative 'gurney/source/npm'
|
9
|
+
require_relative 'gurney/source/pnpm'
|
6
10
|
require_relative 'gurney/source/bundler'
|
7
11
|
require_relative 'gurney/source/ruby_version'
|
8
12
|
require_relative 'gurney/api'
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: gurney_client
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.5.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Martin Schaflitzl
|
8
|
-
autorequire:
|
8
|
+
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2025-03-26 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: colorize
|
@@ -66,7 +66,7 @@ dependencies:
|
|
66
66
|
- - "~>"
|
67
67
|
- !ruby/object:Gem::Version
|
68
68
|
version: '1.5'
|
69
|
-
description:
|
69
|
+
description:
|
70
70
|
email:
|
71
71
|
- martin.schaflitzl@makandra.de
|
72
72
|
executables:
|
@@ -93,8 +93,12 @@ files:
|
|
93
93
|
- lib/gurney/cli/option_parser.rb
|
94
94
|
- lib/gurney/config.rb
|
95
95
|
- lib/gurney/dependency.rb
|
96
|
+
- lib/gurney/dependency_collector.rb
|
97
|
+
- lib/gurney/git_file_reader.rb
|
96
98
|
- lib/gurney/source/base.rb
|
97
99
|
- lib/gurney/source/bundler.rb
|
100
|
+
- lib/gurney/source/npm.rb
|
101
|
+
- lib/gurney/source/pnpm.rb
|
98
102
|
- lib/gurney/source/ruby_version.rb
|
99
103
|
- lib/gurney/source/yarn.rb
|
100
104
|
- lib/gurney/version.rb
|
@@ -103,7 +107,7 @@ licenses:
|
|
103
107
|
- MIT
|
104
108
|
metadata:
|
105
109
|
rubygems_mfa_required: 'true'
|
106
|
-
post_install_message:
|
110
|
+
post_install_message:
|
107
111
|
rdoc_options: []
|
108
112
|
require_paths:
|
109
113
|
- lib
|
@@ -118,8 +122,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
118
122
|
- !ruby/object:Gem::Version
|
119
123
|
version: '0'
|
120
124
|
requirements: []
|
121
|
-
rubygems_version: 3.0
|
122
|
-
signing_key:
|
125
|
+
rubygems_version: 3.1.0
|
126
|
+
signing_key:
|
123
127
|
specification_version: 4
|
124
128
|
summary: Gurney is a small tool to extract yarn and RubyGems dependencies from project
|
125
129
|
files and report them to a web api.
|