deadfinder 1.5.0 → 1.6.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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b8024d88f4693ba6418aab2cca93f3c267fa823f2fa7b47116d39d7d6adb3d40
4
- data.tar.gz: 0c494d74bbbb448eb482204b2525478847d13b62e32c4b2cbd97f16315890973
3
+ metadata.gz: 5a9dba525c92f09cb14019fe890f73dc85eaa975179ac19efca534f0fdc76b14
4
+ data.tar.gz: fee3168a2912dacd9a2740b1aaf2c495b0b68b48b4d024c208523d8a8a5bb5e1
5
5
  SHA512:
6
- metadata.gz: b231b730274c98b22672a4169cb943054db76b6ae0fec70f66eee4eb9e314217942b71d0d4da854556d8f7ec81e33d29be8cb9a758ea12ab3d7aec4ce40907b0
7
- data.tar.gz: 55ae74b443389b62dbd34150deed7e9de433540ca8e2796ed753d90366849f8690d6025d21cd513154e2379cf31c582817c7dd5119ecf629da42dd1d2571296a
6
+ metadata.gz: be806719c1ed5ca905885222d485f376c7e1a61dff0388f812e05d52e279e21315af08fdcfb5b66e9773c88be2f1a5c8c434eba62e6340f91db76ea29f4643db
7
+ data.tar.gz: d966f9237d2a4f62ba98684346b045c49bd14b024723eda22ac5f0c9c2fb0e62fc358d6607d924c0cdd80cd10e8e5f4eb37a13a067188d8394d142fe0476001b
@@ -4,44 +4,51 @@ require 'colorize'
4
4
 
5
5
  class Logger
6
6
  @silent = false
7
+ @mutex = Mutex.new
7
8
 
8
9
  def self.set_silent
9
- @silent = true
10
+ @mutex.synchronize { @silent = true }
10
11
  end
11
12
 
12
13
  def self.unset_silent
13
- @silent = false
14
+ @mutex.synchronize { @silent = false }
14
15
  end
15
16
 
16
17
  def self.silent?
17
- @silent
18
+ @mutex.synchronize { @silent }
19
+ end
20
+
21
+ def self.log(prefix, text, color)
22
+ return if silent?
23
+
24
+ puts prefix.colorize(color) + text.to_s.colorize(:"light_#{color}")
18
25
  end
19
26
 
20
27
  def self.info(text)
21
- puts 'ℹ '.colorize(:blue) + text.to_s.colorize(:light_blue) unless silent?
28
+ log('ℹ ', text, :blue)
22
29
  end
23
30
 
24
31
  def self.error(text)
25
- puts '⚠︎ '.colorize(:red) + text.to_s unless silent?
32
+ log('⚠︎ ', text, :red)
26
33
  end
27
34
 
28
35
  def self.target(text)
29
- puts '► '.colorize(:green) + text.to_s.colorize(:light_green) unless silent?
36
+ log('► ', text, :green)
30
37
  end
31
38
 
32
39
  def self.sub_info(text)
33
- puts ' ● '.colorize(:blue) + text.to_s.colorize(:light_blue) unless silent?
40
+ log(' ● ', text, :blue)
34
41
  end
35
42
 
36
43
  def self.sub_done(text)
37
- puts ' ✓ '.colorize(:blue) + text.to_s.colorize(:light_blue) unless silent?
44
+ log(' ✓ ', text, :blue)
38
45
  end
39
46
 
40
47
  def self.found(text)
41
- puts "#{text}".colorize(:red) unless silent?
48
+ log('', text, :red)
42
49
  end
43
50
 
44
51
  def self.verbose(text)
45
- puts ' ➜ '.colorize(:yellow) + text.to_s.colorize(:light_yellow) unless silent?
52
+ log(' ➜ ', text, :yellow)
46
53
  end
47
54
  end
@@ -4,41 +4,24 @@ require 'uri'
4
4
 
5
5
  def generate_url(text, base_url)
6
6
  node = text.to_s
7
+ return node if node.start_with?('http://', 'https://')
8
+
7
9
  begin
8
- unless node.start_with?('http://', 'https://')
9
- uri = URI(base_url)
10
- if node.start_with? '//'
11
- return "#{uri.scheme}:#{node}"
12
- elsif node.start_with? '/'
13
- return "#{uri.scheme}://#{uri.host}#{node}"
14
- elsif ignore_scheme? node
15
- return nil
16
- else
17
- return "#{extract_directory(uri)}#{node}"
18
- end
10
+ uri = URI(base_url)
11
+ if node.start_with?('//')
12
+ "#{uri.scheme}:#{node}"
13
+ elsif node.start_with?('/')
14
+ "#{uri.scheme}://#{uri.host}#{node}"
15
+ elsif ignore_scheme?(node)
16
+ nil
17
+ else
18
+ URI.join(base_url, node).to_s
19
19
  end
20
20
  rescue StandardError
21
- # puts e
21
+ nil
22
22
  end
23
- node
24
23
  end
25
24
 
26
25
  def ignore_scheme?(url)
27
26
  url.start_with?('mailto:', 'tel:', 'sms:', 'data:', 'file:')
28
27
  end
29
-
30
- def extract_directory(uri)
31
- return "#{uri.scheme}://#{uri.host}#{uri.path}" if uri.path.end_with?('/')
32
-
33
- path_components = uri.path.split('/')
34
- path_components.last
35
- path_components.pop
36
-
37
- directory_path = path_components.join('/')
38
-
39
- if directory_path.start_with?('/')
40
- "#{uri.scheme}://#{uri.host}#{directory_path}/"
41
- else
42
- "#{uri.scheme}://#{uri.host}/#{directory_path}/"
43
- end
44
- end
@@ -1,3 +1,3 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- VERSION = '1.5.0'
3
+ VERSION = '1.6.0'
data/lib/deadfinder.rb CHANGED
@@ -10,6 +10,8 @@ require 'deadfinder/version'
10
10
  require 'concurrent-edge'
11
11
  require 'sitemap-parser'
12
12
  require 'json'
13
+ require 'yaml'
14
+ require 'csv'
13
15
 
14
16
  Channel = Concurrent::Channel
15
17
  CACHE_SET = Concurrent::Map.new
@@ -22,6 +24,7 @@ class DeadFinderRunner
22
24
  'concurrency' => 50,
23
25
  'timeout' => 10,
24
26
  'output' => '',
27
+ 'output_format' => 'json',
25
28
  'headers' => [],
26
29
  'worker_headers' => [],
27
30
  'silent' => true,
@@ -188,19 +191,40 @@ def run_sitemap(sitemap_url, options)
188
191
  end
189
192
 
190
193
  def gen_output(options)
194
+ return if options['output'].empty?
195
+
191
196
  output_data = OUTPUT.to_h
192
- File.write(options['output'], JSON.pretty_generate(output_data)) unless options['output'].empty?
197
+ format = options['output_format'].to_s.downcase
198
+
199
+ content = case format
200
+ when 'yaml', 'yml'
201
+ output_data.to_yaml
202
+ when 'csv'
203
+ CSV.generate do |csv|
204
+ csv << ['target', 'url']
205
+ output_data.each do |target, urls|
206
+ Array(urls).each do |url|
207
+ csv << [target, url]
208
+ end
209
+ end
210
+ end
211
+ else
212
+ JSON.pretty_generate(output_data)
213
+ end
214
+
215
+ File.write(options['output'], content)
193
216
  end
194
217
 
195
218
  class DeadFinder < Thor
196
219
  class_option :include30x, aliases: :r, default: false, type: :boolean, desc: 'Include 30x redirections'
197
220
  class_option :concurrency, aliases: :c, default: 50, type: :numeric, desc: 'Number of concurrency'
198
221
  class_option :timeout, aliases: :t, default: 10, type: :numeric, desc: 'Timeout in seconds'
199
- class_option :output, aliases: :o, default: '', type: :string, desc: 'File to write JSON result'
222
+ class_option :output, aliases: :o, default: '', type: :string, desc: 'File to write result (e.g., json, yaml, csv)'
223
+ class_option :output_format, aliases: :f, default: 'json', type: :string, desc: 'Output format'
200
224
  class_option :headers, aliases: :H, default: [], type: :array,
201
225
  desc: 'Custom HTTP headers to send with initial request'
202
226
  class_option :worker_headers, default: [], type: :array, desc: 'Custom HTTP headers to send with worker requests'
203
- class_option :user_agent, default: 'Mozilla/5.0 (compatible; DeadFinder/1.5.0;)', type: :string,
227
+ class_option :user_agent, default: 'Mozilla/5.0 (compatible; DeadFinder/1.6.0;)', type: :string,
204
228
  desc: 'User-Agent string to use for requests'
205
229
  class_option :proxy, aliases: :p, default: '', type: :string, desc: 'Proxy server to use for requests'
206
230
  class_option :silent, aliases: :s, default: false, type: :boolean, desc: 'Silent mode'
metadata CHANGED
@@ -1,14 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: deadfinder
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.5.0
4
+ version: 1.6.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - hahwul
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2024-10-17 00:00:00.000000000 Z
10
+ date: 2025-02-17 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
13
  name: colorize
@@ -190,7 +189,6 @@ licenses:
190
189
  metadata:
191
190
  rubygems_mfa_required: 'true'
192
191
  source_code_uri: https://github.com/hahwul/deadfinder
193
- post_install_message:
194
192
  rdoc_options: []
195
193
  require_paths:
196
194
  - lib
@@ -205,8 +203,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
205
203
  - !ruby/object:Gem::Version
206
204
  version: '0'
207
205
  requirements: []
208
- rubygems_version: 3.5.3
209
- signing_key:
206
+ rubygems_version: 3.6.3
210
207
  specification_version: 4
211
208
  summary: Find dead-links (broken links)
212
209
  test_files: []