work_hours_calculator 0.1.4 → 0.3.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: 2d7296652efae4aa80651a3887ac25154161d71a72f9535f527934c4d760aa7f
4
- data.tar.gz: 8a469a806bc11f9c01e9adcc4877016e52e223086b94a2d80f3d4f2305ec0df3
3
+ metadata.gz: '01250922e8ef25f907d2c9bf9b39da140e432d3089b5f7133c3c63de063bd7e5'
4
+ data.tar.gz: e1e29fa8dddf4d95c4c64a558dd9ba932f1180adaa65b5bfcee98636bb0f024d
5
5
  SHA512:
6
- metadata.gz: 8ce0a70e34578886ed9db9cc90f2106a79a600f82513ba2ffc6ebd3313f37b88c0d7977162379e8b31dbf29f4e50d49d7d7f9813a05cca440e2822bc748ae748
7
- data.tar.gz: cb636ced3ad71141105816df20d7a1e902129750c29d7d0d58ccf2f6b99af120db63f941d6a600f912d6b69d0bb99f7d67b6951945667ed43bf02be0b0c41aab
6
+ metadata.gz: bdadeee003eeee6c5ae9a4bf44519119805e306c61391bbfba1da1ec4e3a5177c1d0d9c5513a9f822fb012435967e22deaff218886624972cdbae71c137f1679
7
+ data.tar.gz: 644efc89b85c8bc878e720a00e3d1b50eb9031603f64aa91ec68b6d78c257228cb4d315a1f6511f514aa95aed0692e0fc8a18ba0516eef7f11b3baa4046abf10
data/README.md CHANGED
@@ -12,6 +12,7 @@ This is a Ruby command-line tool for calculating the total work hours, break hou
12
12
  - Return net work hours after subtracting break times.
13
13
  - Return net break hours.
14
14
  - Support for csv import or export.
15
+ - Log your work hours for the day.
15
16
 
16
17
  ## Installation
17
18
 
@@ -25,6 +26,14 @@ To calculate your work hours, run the following command:
25
26
  work_calculator -s "9:30:00 AM" -e "7:00:00 PM" -b "12:49:00 PM-1:26:00 PM,3:42:00 PM-4:35:00 PM"
26
27
  ```
27
28
 
29
+ ### Expected Output
30
+ ```bash
31
+ Total Work Decimal Hours: 9.5 hours
32
+ Total Break Decimal Hours: 1.5 hours
33
+ Total Break Hours: 1:30 minutes
34
+ Net Work Decimal Hours: 8.0 hours
35
+ ```
36
+
28
37
  ## Using CSV Input
29
38
  You can also use a CSV file to provide the input data. The CSV file should have the following format:
30
39
  ```text
@@ -40,13 +49,33 @@ You can export the results to a CSV file by specifying the --csv-output option:
40
49
  ```bash
41
50
  work_calculator -s "9:30:00 AM" -e "7:00:00 PM" -b "12:49:00 PM-1:26:00 PM,3:42:00 PM-4:35:00 PM" --csv-output path/to/your/output.csv
42
51
  ```
52
+ ## Logging Work Hours for the Day
53
+ You can log your work hours throughout the day, much like a diary of your work. To log your work, run the following command:
54
+ ```bash
55
+ work_calculator --log some-description-goes-here
56
+ ```
57
+ Example:
58
+ ```bash
59
+ work_calculator --log "working on an interesting project"
60
+ # > Logged work: 2025-02-22 19:47:24 - working on an interesting project
43
61
 
44
- # Expected Output
62
+ work_calculator --log "break"
63
+ # > Logged work: 2025-02-22 19:48:11 - break
64
+
65
+ work_calculator --log "end"
66
+ # > Logged work: 2025-02-22 19:50:01 - end
67
+ ```
68
+ Please note of the system keywords for the description:
69
+ - "break" : considered the record log as a break
70
+ - "end" : considered the record log as finished or has ended work for the day.
71
+
72
+ ### Calculate the hours from your work log
45
73
  ```bash
46
- Total Work Decimal Hours: 9.5 hours
47
- Total Break Decimal Hours: 1.5 hours
48
- Total Break Hours: 1:30 minutes
49
- Net Work Decimal Hours: 8.0 hours
74
+ work_calculator --calculate-log "2025-02-22"
75
+ # > Total Work Decimal Hours: 3.2 hours
76
+ # > Total Break Decimal Hours: 2.15 hours
77
+ # > Total Break Hours: 2:09 minutes
78
+ # > Net Work Decimal Hours: 1.05 hours
50
79
  ```
51
80
 
52
81
  ## Usage
@@ -60,8 +89,13 @@ Run the script from the command line with the required options for start time, e
60
89
  | -b or --breaks | Specifies break periods in comma-separated start_time-end_time format | `-b "12:49:00 PM-1:26:00 PM,3:42:00 PM-4:35:00 PM"` |
61
90
  | --csv-input | Specifies the CSV input file | `--csv-input path/to/your/input.csv`s |
62
91
  | --csv-output | Specifies the CSV output file | `--csv-output path/to/your/output.csv` |
92
+ | --log DESCRIPTION | Log work with description | `--log "working on a project"` |
93
+ | --log-dir DIRECTORY | Specify a directory to store log files | `--lod-dir` <br><br>or export it as an ENV variable so you don't have to specify the directory arg everytime. <br> `export WORK_HOURS_LOG_DIR="/some/path"`|
94
+ | --calculate-log DATE | Calculate hours from the log file for the specified date (e.g., '2023-10-01') | `--calculate-log 2025-02-01` |
63
95
  | -h or --help | Displays help instructions | `-h` |
64
96
 
97
+
98
+
65
99
  ## TODOS
66
100
  - Add support for overtimes
67
101
  - Add support for daily or weekly or monthly summary for csv inputs
data/bin/work_calculator CHANGED
@@ -4,21 +4,28 @@
4
4
  require_relative '../lib/work_hours_calculator'
5
5
  require_relative '../lib/work_hours_calculator/parser'
6
6
  require_relative '../lib/work_hours_calculator/csv_handler'
7
+ require_relative '../lib/work_hours_calculator/logger'
7
8
 
8
9
  options = WorkHoursCalculator::Parser.parse_options
9
10
 
10
- if options[:csv_input]
11
- data = WorkHoursCalculator::CSVHandler.import(options[:csv_input])
12
- results = WorkHoursCalculator.calculate(data[:work_start], data[:work_end], data[:breaks])
11
+ if options[:log]
12
+ WorkHoursCalculator::Logger.log_work(options[:description])
13
13
  else
14
- results = WorkHoursCalculator.calculate(options[:start_time], options[:end_time], options[:breaks])
15
- end
14
+ if options[:calculate_log]
15
+ results = WorkHoursCalculator.calculate_hours_from_log(options[:log_date])
16
+ elsif options[:csv_input]
17
+ data = WorkHoursCalculator::CSVHandler.import(options[:csv_input])
18
+ results = WorkHoursCalculator.calculate(data[:work_start], data[:work_end], data[:breaks])
19
+ else
20
+ results = WorkHoursCalculator.calculate(options[:start_time], options[:end_time], options[:breaks])
21
+ end
16
22
 
17
- if options[:csv_output]
18
- WorkHoursCalculator::CSVHandler.export(options[:csv_output], results)
19
- else
20
- puts "Total Work Decimal Hours: #{results[:total_work_decimal_hours].round(2)} hours"
21
- puts "Total Break Decimal Hours: #{results[:total_break_decimal_hours].round(2)} hours"
22
- puts "Total Break Hours: #{results[:total_break_hours]} minutes"
23
- puts "Net Work Decimal Hours: #{results[:net_work_decimal_hours].round(2)} hours"
23
+ if options[:csv_output]
24
+ WorkHoursCalculator::CSVHandler.export(options[:csv_output], results)
25
+ else
26
+ puts "Total Work Decimal Hours: #{results[:total_work_decimal_hours].round(2)} hours"
27
+ puts "Total Break Decimal Hours: #{results[:total_break_decimal_hours].round(2)} hours"
28
+ puts "Total Break Hours: #{results[:total_break_hours]} minutes"
29
+ puts "Net Work Decimal Hours: #{results[:net_work_decimal_hours].round(2)} hours"
30
+ end
24
31
  end
@@ -0,0 +1,83 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "csv"
4
+ require "date"
5
+ require "time"
6
+ module WorkHoursCalculator
7
+ class InvalidFile < StandardError; end
8
+
9
+ class InvalidDir < StandardError; end
10
+
11
+ class InvalidRecord < StandardError; end
12
+
13
+ class MissingRecord < StandardError; end
14
+
15
+ class Logger
16
+ DEFAULT_LOG_DIR = File.join(Dir.home, "work_hours_logs")
17
+ LOG_DIR = ENV["WORK_HOURS_LOG_DIR"] || DEFAULT_LOG_DIR
18
+
19
+ def self.log_work(description, log_dir = LOG_DIR)
20
+ raise InvalidRecord, "Description cannot be empty" if description.nil? || description.strip.empty?
21
+
22
+ begin
23
+ Dir.mkdir(log_dir) unless Dir.exist?(log_dir)
24
+ rescue SystemCallError => e
25
+ raise InvalidDir, "Failed to create log directory: #{e.message}"
26
+ end
27
+
28
+ log_file = File.join(log_dir, "#{Date.today}.csv")
29
+
30
+ begin
31
+ CSV.open(log_file, "a") do |csv|
32
+ time = Time.now.strftime("%Y-%m-%d %H:%M:%S")
33
+ csv << [time, description]
34
+ end
35
+ rescue SystemCallError => e
36
+ raise InvalidFile, "Failed to write to log file: #{e.message}"
37
+ end
38
+
39
+ puts "Logged work: #{Time.now.strftime("%Y-%m-%d %H:%M:%S")} - #{description}"
40
+ end
41
+
42
+ def self.get_hours_from_log(date, log_dir = DEFAULT_LOG_DIR)
43
+ raise InvalidRecord, "Date cannot be nil" if date.nil?
44
+
45
+ log_file = File.join(log_dir, "#{date}.csv")
46
+ unless File.exist?(log_file)
47
+ puts "Log file for #{date} not found."
48
+ exit 1
49
+ end
50
+
51
+ entries = CSV.read(log_file, headers: false).map do |row|
52
+ {time: Time.parse(row[0]), description: row[1].strip}
53
+ end
54
+
55
+ work_start = entries.find { |entry| entry[:description].downcase != "break" && entry[:description].downcase != "end" }
56
+ work_end = entries.reverse.find { |entry| entry[:description].downcase == "end" }
57
+
58
+ raise MissingRecord, "Work start record not found" if work_start.nil?
59
+ raise MissingRecord, "Work end record not found" if work_end.nil?
60
+
61
+ work_start_time = work_start[:time]
62
+ work_end_time = work_end[:time]
63
+
64
+ breaks = []
65
+ break_start = nil
66
+
67
+ entries.each do |entry|
68
+ if entry[:description].downcase == "break" || entry[:description].downcase == "end"
69
+ break_start ||= entry[:time]
70
+ elsif break_start
71
+ breaks << [break_start, entry[:time]]
72
+ break_start = nil
73
+ end
74
+ end
75
+
76
+ {
77
+ work_start: work_start_time.strftime("%I:%M:%S %p"),
78
+ work_end: work_end_time.strftime("%I:%M:%S %p"),
79
+ breaks: breaks.map { |start, end_time| [start.strftime("%I:%M:%S %p"), end_time.strftime("%I:%M:%S %p")] }
80
+ }
81
+ end
82
+ end
83
+ end
@@ -15,17 +15,31 @@ module WorkHoursCalculator
15
15
  opts.on("-b", "--breaks x,y", Array, "Break periods as comma-separated pairs (e.g., '12:49:00 PM-1:26:00 PM,3:42:00 PM-4:35:00 PM')") { |v| options[:breaks] = v.map { |pair| pair.split("-") } }
16
16
  opts.on("--csv-input FILE", "CSV input file") { |v| options[:csv_input] = v }
17
17
  opts.on("--csv-output FILE", "CSV output file") { |v| options[:csv_output] = v }
18
+ opts.on("--log DESCRIPTION", "Log work with description") { |v|
19
+ options[:log] = true
20
+ options[:description] = v
21
+ }
22
+ opts.on("--log-dir DIRECTORY", "Directory to store log files") { |v| options[:log_dir] = v }
23
+ opts.on("--calculate-log DATE", "Calculate hours from the log file for the specified date (e.g., '2023-10-01')") { |v|
24
+ options[:calculate_log] = true
25
+ options[:log_date] = v
26
+ }
18
27
  opts.on("-h", "--help", "Show help") {
19
28
  puts opts
20
29
  exit
21
30
  }
22
31
  end.parse!(args, into: options)
23
32
 
24
- if options[:csv_input].nil? && (options[:start_time].nil? || options[:end_time].nil? || options[:breaks].nil?)
33
+ if (options[:csv_input].nil? && options[:log].nil? && options[:calculate_log].nil?) && (options[:start_time].nil? || options[:end_time].nil? || options[:breaks].nil?)
25
34
  puts "Please provide start time, end time, and break times, or a CSV input file."
26
35
  exit
27
36
  end
28
37
 
38
+ if options[:log] && options[:description].nil?
39
+ puts "Error: Description is required when logging work hours."
40
+ exit 1
41
+ end
42
+
29
43
  options
30
44
  end
31
45
  end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module WorkHoursCalculator
4
- VERSION = "0.1.4"
4
+ VERSION = "0.3.0"
5
5
  end
@@ -2,6 +2,7 @@
2
2
 
3
3
  require_relative "work_hours_calculator/version"
4
4
  require_relative "work_hours_calculator/calculate"
5
+ require_relative "work_hours_calculator/logger"
5
6
 
6
7
  module WorkHoursCalculator
7
8
  class Error < StandardError; end
@@ -9,4 +10,9 @@ module WorkHoursCalculator
9
10
  def self.calculate(work_start, work_end, breaks)
10
11
  WorkHoursCalculator::Calculate.new(work_start, work_end, breaks).execute
11
12
  end
13
+
14
+ def self.calculate_hours_from_log(date)
15
+ results = WorkHoursCalculator::Logger.get_hours_from_log(date)
16
+ WorkHoursCalculator::Calculate.new(results[:work_start], results[:work_end], results[:breaks]).execute
17
+ end
12
18
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: work_hours_calculator
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.4
4
+ version: 0.3.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Gerda Decio
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2025-02-15 00:00:00.000000000 Z
11
+ date: 2025-02-22 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: csv
@@ -68,6 +68,7 @@ files:
68
68
  - lib/work_hours_calculator.rb
69
69
  - lib/work_hours_calculator/calculate.rb
70
70
  - lib/work_hours_calculator/csv_handler.rb
71
+ - lib/work_hours_calculator/logger.rb
71
72
  - lib/work_hours_calculator/parser.rb
72
73
  - lib/work_hours_calculator/version.rb
73
74
  homepage: https://github.com/gerdadecio/work-hours-calculator-ruby
@@ -79,7 +80,15 @@ metadata:
79
80
  documentation_uri: https://github.com/gerdadecio/work-hours-calculator-ruby
80
81
  bug_tracker_uri: https://github.com/gerdadecio/work-hours-calculator-ruby/issues
81
82
  changelog_uri: https://github.com/gerdadecio/work-hours-calculator-ruby/blob/main/CHANGELOG.md
82
- post_install_message:
83
+ post_install_message: |
84
+ Thank you for installing the Work Hours Calculator gem!
85
+
86
+ You can set the log directory using the WORK_HOURS_LOG_DIR environment variable. If this variable is not set, the default log directory will be ~/work_hours_logs.
87
+
88
+ Example:
89
+ export WORK_HOURS_LOG_DIR="/path/to/custom/log_directory"
90
+
91
+ For more information, please refer to the README.md file.
83
92
  rdoc_options: []
84
93
  require_paths:
85
94
  - lib