elemac 0.1.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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 3d7f4976b34a33440772fb7e5da0f6e485804ee5
4
+ data.tar.gz: e961679265faaff5f78fb19910c4a38fb5fa1727
5
+ SHA512:
6
+ metadata.gz: 7e52b89b7a83b87977e14be5de245d5e92cd3d9eb4254dc2798d2a7cbbbb554815cad6656b21694e67b536a780f96ffc70bc1e165f83391fbb868fba0841ab4e
7
+ data.tar.gz: 460092142f21f416c52a4d50e92144ab54044da3e9389628c6f2241c08642fcfd66b0bc768cd46630417af4df9c396cf9ce2530721ab862f89100aa797ce34d5
data/.gitignore ADDED
@@ -0,0 +1,50 @@
1
+ *.gem
2
+ *.rbc
3
+ /.config
4
+ /coverage/
5
+ /InstalledFiles
6
+ /pkg/
7
+ /spec/reports/
8
+ /spec/examples.txt
9
+ /test/tmp/
10
+ /test/version_tmp/
11
+ /tmp/
12
+
13
+ # Used by dotenv library to load environment variables.
14
+ # .env
15
+
16
+ ## Specific to RubyMotion:
17
+ .dat*
18
+ .repl_history
19
+ build/
20
+ *.bridgesupport
21
+ build-iPhoneOS/
22
+ build-iPhoneSimulator/
23
+
24
+ ## Specific to RubyMotion (use of CocoaPods):
25
+ #
26
+ # We recommend against adding the Pods directory to your .gitignore. However
27
+ # you should judge for yourself, the pros and cons are mentioned at:
28
+ # https://guides.cocoapods.org/using/using-cocoapods.html#should-i-check-the-pods-directory-into-source-control
29
+ #
30
+ # vendor/Pods/
31
+
32
+ ## Documentation cache and generated files:
33
+ /.yardoc/
34
+ /_yardoc/
35
+ /doc/
36
+ /rdoc/
37
+
38
+ ## Environment normalization:
39
+ /.bundle/
40
+ /vendor/bundle
41
+ /lib/bundler/man/
42
+
43
+ # for a library or gem, you might want to ignore these files since the code is
44
+ # intended to run in multiple environments; otherwise, check them in:
45
+ # Gemfile.lock
46
+ # .ruby-version
47
+ # .ruby-gemset
48
+
49
+ # unless supporting rvm < 1.11.0 or doing something fancy, ignore this:
50
+ .rvmrc
data/Gemfile ADDED
@@ -0,0 +1,6 @@
1
+ source "https://rubygems.org"
2
+
3
+ git_source(:github) {|repo_name| "https://github.com/#{repo_name}" }
4
+
5
+ # Specify your gem's dependencies in elemac.gemspec
6
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2017 Michał Prostko
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,28 @@
1
+ # Elemac SA-03 Ruby binding
2
+ Ruby binding for ELEMAC SA-03 aquarium controller.
3
+
4
+ Manufacturer: http://www.elemac.pl/
5
+
6
+ Gem provides a way to read memory of the aquarium controller. Memory addresses, offsets ect. were found from decompiling the original program which was written in C# and available only for MS Windows. Memory writing (currently disabled in code) is also possible although it might damage the controller and make it unusable.
7
+
8
+ :warning: **Warning:** This script is published for educational purposes only! Author will accept no responsibility for any consequence, damage or loss which might result from use.
9
+
10
+ ## Features:
11
+ You're able to view following details from the controller:
12
+ * Overview info (date, statuses, alarms)
13
+ * Lightning info
14
+ * Sensor info (Temperature and PH) # Redox is unavailable because I am unable to test this
15
+ * Output info (Power / PWM / TTL) # PWM and TTL are 'dumb' because I am unable to test this
16
+ * Timers info (ToDo)
17
+
18
+ ## Installation:
19
+ Clone the repo. Build gem, install and use in your ruby scripts.
20
+
21
+ ## Usage:
22
+
23
+ ```ruby
24
+ el = Elemac::Connection.new
25
+ x = Elemac::Sensors.new(device: el)
26
+ x.temp1.value
27
+ x.ph1.inspect
28
+ ```
data/Rakefile ADDED
@@ -0,0 +1,2 @@
1
+ require "bundler/gem_tasks"
2
+ task :default => :spec
data/bin/console ADDED
@@ -0,0 +1,14 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require "bundler/setup"
4
+ require "elemac"
5
+
6
+ # You can add fixtures and/or initialization code here to make experimenting
7
+ # with your gem easier. You can also use a different console, if you like.
8
+
9
+ # (If you use this, don't forget to add pry to your Gemfile!)
10
+ # require "pry"
11
+ # Pry.start
12
+
13
+ require "irb"
14
+ IRB.start(__FILE__)
data/bin/setup ADDED
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
data/elemac.gemspec ADDED
@@ -0,0 +1,25 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path("../lib", __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require "elemac/version"
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.authors = ["Michał Prostko"]
8
+ spec.license = "MIT"
9
+ spec.summary = 'Ruby binding for ELEMAC SA-03 aquarium controller'
10
+ spec.description = 'Ruby binding for ELEMAC SA-03 aquarium controller'
11
+ spec.homepage = "https://github.com/mprostko/elemac"
12
+
13
+ spec.name = 'elemac'
14
+ spec.version = Elemac::VERSION
15
+ spec.files = `git ls-files -z`.split("\x0").reject do |f|
16
+ f.match(%r{^(test|spec|features)/})
17
+ end
18
+ spec.executables = []
19
+ spec.require_paths = ["lib"]
20
+ spec.required_ruby_version = ">= 2.0"
21
+
22
+ spec.add_dependency "hidapi", "~> 0.1.9"
23
+ spec.add_development_dependency "bundler", "~> 1.15"
24
+ spec.add_development_dependency "rake", "~> 10.0"
25
+ end
data/lib/elemac.rb ADDED
@@ -0,0 +1,9 @@
1
+ require "elemac/version"
2
+ require "elemac/connection"
3
+ require "elemac/overview"
4
+ require "elemac/light"
5
+ require "elemac/outputs"
6
+ require "elemac/sensors"
7
+
8
+ module Elemac
9
+ end
@@ -0,0 +1,69 @@
1
+ require "hidapi"
2
+
3
+ module Elemac
4
+ class Connection
5
+ def initialize(vendor: 0x4d8, product: 0x3f, debug: false)
6
+ @debug = debug
7
+ @dev = HIDAPI::open(vendor, product)
8
+ throw 'Cannot open ELEMAC HID' unless @dev
9
+ end
10
+
11
+ def request_data(address)
12
+ bytes = Array.new(64, 0)
13
+ length = address.length - 1
14
+ bytes[0..length] = address.bytes
15
+ # Write command to device
16
+ @dev.write(bytes)
17
+ # Read result.
18
+ response = read_data # this hex form, transform it to number
19
+ response.hex
20
+ end
21
+
22
+ # Reading data is in form 'rXYYY' where X is bit length, YYY is the address in hex
23
+ # Read data ends on byte 0
24
+ def read_data
25
+ raw = @dev.read
26
+ response = ""
27
+ debug_read(raw) if @debug
28
+
29
+ raw.each_byte { |x|
30
+ break if x == 0
31
+ response += x.chr # Get char (for hex)
32
+ }
33
+ # Reverse bit order
34
+ response = response.scan(/../).reverse.join
35
+ response
36
+ end
37
+
38
+ =begin
39
+ # Writing to controller is in form 'wXYYY=value' where X is bit length, YYY is the address in hex
40
+ def write_data(address, value=32896)
41
+ throw 'USE ONLY IN EMERGENCY CASES!'
42
+ bytes = Array.new(64, 0)
43
+ address[0] = 'w' # change read to write
44
+ value = value.to_s(16) # convert to hex
45
+ # Reverse bit order
46
+ value = value.scan(/../).reverse.join
47
+ message = "#{address}=#{value}"
48
+ data_size = message.length - 1
49
+ bytes[0..data_size] = message.bytes
50
+ @dev.write(bytes)
51
+ raw = @dev.read
52
+ puts raw
53
+ puts raw.inspect
54
+ end
55
+ =end
56
+
57
+ private
58
+ def debug_read(raw)
59
+ dbg = []
60
+ count = 0
61
+ raw.each_byte { |x|
62
+ count += 1
63
+ dbg << x
64
+ }
65
+ puts "bits: #{count}"
66
+ puts dbg.inspect
67
+ end
68
+ end
69
+ end
@@ -0,0 +1,65 @@
1
+ require 'date'
2
+
3
+ module Elemac
4
+ class Converter
5
+ BIT32_HIGH_MASK = 4278190080
6
+ BIT32_LOW_MASK = 16711680
7
+ BIT16_HIGH_MASK = 65280
8
+ BIT16_LOW_MASK = 255
9
+ def self.bit32_high(value)
10
+ (value & BIT16_HIGH_MASK) >> 24
11
+ end
12
+ def self.bit32_low(value)
13
+ (value & BIT32_LOW_MASK) >> 16
14
+ end
15
+ def self.bit16_high(value)
16
+ (value & BIT16_HIGH_MASK) >> 8
17
+ end
18
+ def self.bit16_low(value)
19
+ (value & BIT16_LOW_MASK)
20
+ end
21
+ def self.int_to_hour(value)
22
+ hour = bit16_low(value).to_s
23
+ minute = bit16_high(value).to_s
24
+ "#{hour.rjust(2,'0')}:#{minute.rjust(2,'0')}"
25
+ end
26
+ def self.seconds_to_hsm(value)
27
+ value = 14400 if value > 14400
28
+ Time.at(value).utc.strftime("%H:%M:%S")
29
+ end
30
+ def self.long_to_date(value)
31
+ day = bit32_low(value).to_s
32
+ month = bit16_high(value).to_s
33
+ year = (2000 + bit16_low(value).to_i).to_s
34
+ "#{year}-#{month.rjust(2,'0')}-#{day.rjust(2,'0')}"
35
+ end
36
+ # calibration date should be in format YYYY-MM-DD
37
+ # reminder is the amount of months that ph has to be calibrated
38
+ def self.ph_reminder(value, calibration_date, reminder=0)
39
+ value = value & 3
40
+ ph7, ph49, out_of_date = false
41
+ case(value)
42
+ when 0
43
+ ph7, ph49, out_of_date = false, false, false
44
+ when 1
45
+ ph7, ph49, out_of_date = true, false, false
46
+ when 2
47
+ ph7, ph49, out_of_date = false, true, false
48
+ when 3
49
+ ph7, ph49 = true, true
50
+ next_calibration_date = Date.parse(calibration_date) << -reminder
51
+ out_of_date = next_calibration_date < Date.today
52
+ end
53
+ "PH7: #{ph7.to_s}, PH49: #{ph49.to_s}, Calibrated: #{out_of_date.to_s}"
54
+ end
55
+ def self.format_date(year,month,date)
56
+ "#{year.to_s.rjust(2,'0')}-#{month.to_s.rjust(2,'0')}-#{date.to_s.rjust(2,'0')}"
57
+ end
58
+ def self.format_time(hour,minute,second)
59
+ "#{hour.to_s.rjust(2,'0')}:#{minute.to_s.rjust(2,'0')}:#{second.to_s.rjust(2,'0')}"
60
+ end
61
+ def self.ph_9_enabled(value)
62
+ value & 4 > 0
63
+ end
64
+ end
65
+ end
@@ -0,0 +1,74 @@
1
+ require "elemac/property"
2
+ require "elemac/converter"
3
+
4
+ module Elemac
5
+ class Light
6
+ SUNRISE = 92
7
+ SUNSET = 94
8
+ DAY_BREAK_START = 160
9
+ DAY_BREAK_END = 162
10
+ NIGHT_BREAK_START = 164
11
+ NIGHT_BREAK_END = 166
12
+ DAYLIGHT_SAVE = 360
13
+ LAMP_POWER_ON_DELAY = 91
14
+ OFFSET = 0
15
+
16
+ def initialize(device:)
17
+ @device = device
18
+ end
19
+
20
+ def get_offset
21
+ OFFSET
22
+ end
23
+
24
+ def sunrise
25
+ prop = Property.new(offset: get_offset, address: SUNRISE, type: :short)
26
+ Converter.int_to_hour(get_data(prop.to_s))
27
+ end
28
+
29
+ def sunset
30
+ prop = Property.new(offset: get_offset, address: SUNSET, type: :short)
31
+ Converter.int_to_hour(get_data(prop.to_s))
32
+ end
33
+
34
+ def day_break_start
35
+ prop = Property.new(offset: get_offset, address: DAY_BREAK_START, type: :short)
36
+ Converter.int_to_hour(get_data(prop.to_s))
37
+ end
38
+
39
+ def day_break_end
40
+ prop = Property.new(offset: get_offset, address: DAY_BREAK_END, type: :short)
41
+ Converter.int_to_hour(get_data(prop.to_s))
42
+ end
43
+
44
+ def night_break_start
45
+ prop = Property.new(offset: get_offset, address: NIGHT_BREAK_START, type: :short)
46
+ Converter.int_to_hour(get_data(prop.to_s))
47
+ end
48
+
49
+ def night_break_end
50
+ prop = Property.new(offset: get_offset, address: NIGHT_BREAK_END, type: :short)
51
+ Converter.int_to_hour(get_data(prop.to_s))
52
+ end
53
+
54
+ def light_power_on_delay
55
+ prop = Property.new(offset: get_offset, address: LAMP_POWER_ON_DELAY, type: :char)
56
+ get_data(prop.to_s)
57
+ end
58
+
59
+ def inspect
60
+ puts "SUNRISE:\t#{sunrise}"
61
+ puts "D-BREAK START:\t#{day_break_start}"
62
+ puts "D-BREAK END:\t#{day_break_end}"
63
+ puts "SUNSET:\t\t#{sunset}"
64
+ puts "N-BREAK START:\t#{night_break_start}"
65
+ puts "N-BREAK END:\t#{night_break_end}"
66
+ puts "HQI DELAY:\t#{light_power_on_delay}"
67
+ end
68
+
69
+ private
70
+ def get_data(address)
71
+ @device.request_data(address)
72
+ end
73
+ end
74
+ end
@@ -0,0 +1,104 @@
1
+ require "elemac/property"
2
+
3
+ module Elemac
4
+ module Output
5
+ class Default
6
+ OFFSET = 2048
7
+ DATA_SIZE = 1
8
+ TYPE = 0
9
+ MODE = 256
10
+ DAY_1_START = 25
11
+ DAY_1_STOP = 281
12
+ DAY_2_START = 100
13
+ DAY_2_STOP = 356
14
+ DAY_3_START = 175
15
+ DAY_3_STOP = 431
16
+ WORK = 512
17
+ PAUSE = 587
18
+ TIMER = 662
19
+ POWER_FLAG = 1
20
+ VOLTAGE_FLAG = 14 # bits 1110
21
+ DIMMABLE_FLAG = 16
22
+ NEGATIVE_FLAG = 32
23
+ TIMER_ON_FLAG = 64
24
+ PRESENT_FLAG = 128
25
+ def initialize(device:, index: 0)
26
+ @device = device
27
+ @index = index
28
+ @voltage = 0
29
+ @dimmable = false
30
+ @dim_value = 0
31
+ end
32
+
33
+ def get_offset
34
+ throw "Bad offset. Can be only 0..25. #{@index} given." unless (0..25).cover?(@index)
35
+ OFFSET + (@index * DATA_SIZE)
36
+ end
37
+
38
+ def name
39
+ (@index + 'A'.ord).chr
40
+ end
41
+
42
+ def type
43
+ prop = Property.new(offset: get_offset, address: TYPE, type: :short)
44
+ get_data(prop.to_s)
45
+ end
46
+
47
+ def powered
48
+ type & POWER_FLAG != 0
49
+ end
50
+
51
+ def board_dimmable
52
+ type & DIMMABLE_FLAG != 0
53
+ end
54
+
55
+ def board_present
56
+ type & 128 != 0
57
+ end
58
+
59
+ def board_voltage
60
+ val = (type & VOLTAGE_FLAG) >> 1
61
+ case(val)
62
+ when 0
63
+ 230
64
+ when 1
65
+ 12
66
+ when 2
67
+ 5
68
+ when 3
69
+ 3
70
+ else
71
+ 0
72
+ end
73
+ end
74
+
75
+ def negative
76
+ type & NEGATIVE_FLAG != 0
77
+ end
78
+
79
+ def timer_on
80
+ type & TIMER_ON_FLAG != 0
81
+ end
82
+
83
+ def dimmable
84
+ @dimmable
85
+ end
86
+
87
+ def inspect
88
+ puts "NAME:\t\t#{name}"
89
+ puts "TYPE:\t\t#{type}"
90
+ puts "POWERED:\t#{powered}"
91
+ puts "TIMER:\t\t#{timer_on}"
92
+ puts "NEGATIVE:\t#{negative}"
93
+ puts "BRD_PRESENT:\t#{board_present}"
94
+ puts "BRD_VOLTAGE:\t#{board_voltage}"
95
+ puts "BRD_DIMMABLE:\t#{board_dimmable}"
96
+ end
97
+
98
+ private
99
+ def get_data(address)
100
+ @device.request_data(address)
101
+ end
102
+ end
103
+ end
104
+ end
@@ -0,0 +1,10 @@
1
+ require "elemac/output/default"
2
+
3
+ module Elemac
4
+ class Output::Power < Output::Default
5
+ def initialize(device:, index: 0)
6
+ super(device: device, index: index)
7
+ @voltage = 230
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,61 @@
1
+ require "elemac/output/default"
2
+ require "elemac/converter"
3
+
4
+ module Elemac
5
+ class Output::Pwm < Output::Default
6
+ # Reg -> regulated -> PWM
7
+ REG_OFFSET = 1792
8
+ REG_DATA_SIZE = 8
9
+ REG_LEVEL = 1
10
+ REG_MIN_LEVEL = 2
11
+ REG_MAX_LEVEL = 3
12
+ REG_TRANS_TIME = 4
13
+ REG_TRANS_COUNT = 6
14
+
15
+ def initialize(device:, index: 0, dim_index: 0)
16
+ super(device: device, index: index)
17
+ @dimm_index = dim_index
18
+ @voltage = 12
19
+ @dimmable = true
20
+ end
21
+
22
+ def get_reg_offset
23
+ throw "Bad offset. Can be only 0..8. #{@index} given." unless (0..8).cover?(@index)
24
+ REG_OFFSET + (@dimm_index * REG_DATA_SIZE)
25
+ end
26
+
27
+ def level
28
+ prop = Property.new(offset: get_reg_offset, address: REG_LEVEL, type: :char)
29
+ get_data(prop.to_s)
30
+ end
31
+
32
+ def min_level
33
+ prop = Property.new(offset: get_reg_offset, address: REG_MIN_LEVEL, type: :char)
34
+ get_data(prop.to_s)
35
+ end
36
+
37
+ def max_level
38
+ prop = Property.new(offset: get_reg_offset, address: REG_MAX_LEVEL, type: :char)
39
+ get_data(prop.to_s)
40
+ end
41
+
42
+ def trans_time
43
+ prop = Property.new(offset: get_reg_offset, address: REG_TRANS_TIME, type: :short)
44
+ get_data(prop.to_s)
45
+ end
46
+
47
+ def trans_count
48
+ prop = Property.new(offset: get_reg_offset, address: REG_TRANS_COUNT, type: :char)
49
+ get_data(prop.to_s)
50
+ end
51
+
52
+ def inspect
53
+ super
54
+ puts "LEVEL:\t\t#{level}"
55
+ puts "MIN LEVEL:\t#{min_level}"
56
+ puts "MAX_LEVEL:\t#{max_level}"
57
+ puts "TRANS TIME:\t#{Converter.seconds_to_hsm(trans_time)}"
58
+ puts "TRANS COUNT:\t#{trans_count}"
59
+ end
60
+ end
61
+ end
@@ -0,0 +1,10 @@
1
+ require "elemac/output/default"
2
+
3
+ module Elemac
4
+ class Output::Ttl < Output::Default
5
+ def initialize(device:, index: 0)
6
+ super(device: device, index: index)
7
+ @voltage = 3.3
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,66 @@
1
+ require "elemac/output/power"
2
+ require "elemac/output/pwm"
3
+ require "elemac/output/ttl"
4
+
5
+ module Elemac
6
+ class Outputs
7
+ def initialize(device:)
8
+ @device = device
9
+ # TODO: check loaded sensors
10
+ @out_a = @out_b = @out_c = @out_d = @out_e = true
11
+ @out_f = true
12
+ @out_g = true
13
+ # more unavailable now
14
+ end
15
+
16
+ def out_a
17
+ throw 'Output A unavailable' unless @out_a
18
+ PowerOutput.new(device: @device, index: 0)
19
+ end
20
+ def out_b
21
+ throw 'Output B unavailable' unless @out_b
22
+ PowerOutput.new(device: @device, index: 1)
23
+ end
24
+ def out_c
25
+ throw 'Output C unavailable' unless @out_c
26
+ PowerOutput.new(device: @device, index: 2)
27
+ end
28
+ def out_d
29
+ throw 'Output D unavailable' unless @out_d
30
+ PowerOutput.new(device: @device, index: 3)
31
+ end
32
+ def out_e
33
+ throw 'Output E unavailable' unless @out_e
34
+ PowerOutput.new(device: @device, index: 4)
35
+ end
36
+ def out_f
37
+ throw 'Output F unavailable' unless @out_f
38
+ PwmOutput.new(device: @device, index: 5, dim_index: 0)
39
+ end
40
+ def out_g
41
+ throw 'Output G unavailable' unless @out_g
42
+ TtlOutput.new(device: @device, index: 6)
43
+ end
44
+
45
+ def outputs_available
46
+ ['a','b','c','d','e','f','g'] # and more...
47
+ end
48
+
49
+ def power_outputs
50
+ outputs = []
51
+ outputs << out_a if @out_a
52
+ outputs << out_b if @out_b
53
+ outputs << out_c if @out_c
54
+ outputs << out_d if @out_d
55
+ outputs << out_e if @out_e
56
+ end
57
+
58
+ def dimm_outputs
59
+ [out_f] # and more...
60
+ end
61
+
62
+ def ttl_outputs
63
+ [out_g] # and more...
64
+ end
65
+ end
66
+ end
@@ -0,0 +1,114 @@
1
+ require "elemac/property"
2
+ require "elemac/converter"
3
+
4
+ module Elemac
5
+ class Overview
6
+ PROP_CURRENT_HOUR = 62
7
+ PROP_CURRENT_MINUTE = 63
8
+ PROP_CURRENT_SECOND = 64
9
+ PROP_CURRENT_DAY = 371
10
+ PROP_CURRENT_MONTH = 370
11
+ PROP_CURRENT_YEAR = 369
12
+ PROP_CURRENT_DOW = 368
13
+ DAY_NIGHT_FLAG = 84
14
+ DAY_NIGHT_LIGHT_FLAG = 86
15
+ DAY_FLAG = 4
16
+ DAY_LIGHT_FLAG = 1
17
+ NIGHT_LIGHT_FLAG = 64
18
+ OFFSET = 0
19
+
20
+ def initialize(device:)
21
+ @device = device
22
+ end
23
+
24
+ def get_offset
25
+ OFFSET
26
+ end
27
+
28
+ def current_day
29
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_DAY, type: :char)
30
+ get_data(prop.to_s).to_i
31
+ end
32
+
33
+ def current_month
34
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_MONTH, type: :char)
35
+ get_data(prop.to_s).to_i
36
+ end
37
+
38
+ def current_year
39
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_YEAR, type: :char)
40
+ 2000 + get_data(prop.to_s).to_i
41
+ end
42
+
43
+ def current_date
44
+ Converter.format_date(current_year, current_month, current_day)
45
+ end
46
+
47
+ def current_hour
48
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_HOUR, type: :char)
49
+ get_data(prop.to_s).to_i
50
+ end
51
+
52
+ def current_minute
53
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_MINUTE, type: :char)
54
+ get_data(prop.to_s).to_i
55
+ end
56
+
57
+ def current_second
58
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_SECOND, type: :char)
59
+ get_data(prop.to_s).to_i
60
+ end
61
+
62
+ def current_time
63
+ Converter.format_time(current_hour, current_minute, current_second)
64
+ end
65
+
66
+ def current_day_of_week
67
+ prop = Property.new(offset: get_offset, address: PROP_CURRENT_DOW, type: :char)
68
+ get_data(prop.to_s)
69
+ end
70
+
71
+ def day_night_flags
72
+ prop = Property.new(offset: get_offset, address: DAY_NIGHT_FLAG, type: :char)
73
+ get_data(prop.to_s)
74
+ end
75
+
76
+ def day_night_light_flags
77
+ prop = Property.new(offset: get_offset, address: DAY_NIGHT_LIGHT_FLAG, type: :char)
78
+ get_data(prop.to_s)
79
+ end
80
+
81
+ def day
82
+ day_night_flags & DAY_FLAG != 0
83
+ end
84
+
85
+ def night
86
+ !day
87
+ end
88
+
89
+ def day_light
90
+ day_night_light_flags & DAY_LIGHT_FLAG != 0
91
+ end
92
+
93
+ def night_light
94
+ day_night_light_flags & NIGHT_LIGHT_FLAG != 0
95
+ end
96
+
97
+ def inspect
98
+ puts "DATE:\t\t#{current_date}"
99
+ puts "TIME:\t\t#{current_time}"
100
+ puts "DAY OF WEEK:\t#{current_day_of_week}"
101
+ puts "D/N FLG:\t#{day_night_flags} (#{day_night_flags.to_s(2)})"
102
+ puts "D/N LIGHT FLG:\t#{day_night_light_flags} (#{day_night_light_flags.to_s(2)})"
103
+ puts "IS DAY:\t\t#{day}"
104
+ puts "IS NIGHT:\t#{night}"
105
+ puts "DAY LIGHT:\t#{day_light}"
106
+ puts "NIGHT LIGHT:\t#{night_light}"
107
+ end
108
+
109
+ private
110
+ def get_data(address)
111
+ @device.request_data(address)
112
+ end
113
+ end
114
+ end
@@ -0,0 +1,27 @@
1
+ module Elemac
2
+ class Property
3
+ def initialize(address:, offset: 0, type: :int)
4
+ @address = address
5
+ @offset = offset
6
+ @type = type
7
+ end
8
+ def get_type(type)
9
+ case(type)
10
+ when :char # 1 byte
11
+ 'r0'
12
+ when :short # 2 bytes
13
+ 'r1'
14
+ when :int # 3 bytes
15
+ 'r2'
16
+ when :long # 4 bytes
17
+ 'r3'
18
+ end
19
+ end
20
+ def get_address
21
+ (@offset + @address).to_s(16).upcase.rjust(3, '0')
22
+ end
23
+ def to_s
24
+ "#{get_type(@type)}#{get_address}"
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,114 @@
1
+ require "elemac/property"
2
+
3
+ module Elemac
4
+ module Sensor
5
+ class Default
6
+ FLAGS = 0
7
+ MEASURMENT = 2
8
+ HYSTERESIS = 10
9
+ DAY_HIGH = 6
10
+ NIGHT_HIGH = 8
11
+ DAY_LOW = 13
12
+ NIGHT_LOW = 11
13
+ ALARM_HIGH = 4
14
+ ALARM_LOW = 15
15
+ DATA_SIZE = 17
16
+ FLAG_PRESENT = 1
17
+ FLAG_RISE = 2
18
+ FLAG_LOWER = 4
19
+ FLAG_ALARM_HIGH = 8
20
+ FLAG_ALARM_LOW = 16
21
+ OFFSET = 1864
22
+
23
+ def initialize(device:, index: 0, divider: 1.0)
24
+ throw 'Index has to be in range 0-5' if !(0..5).cover?(index)
25
+ @device = device
26
+ @index = index
27
+ @offset = get_offset
28
+ @divider = divider
29
+ end
30
+ def get_offset
31
+ # this does not include rh sensor yet
32
+ throw "Bad offset. Can be only 0,1,2,3 for TEMP or 4,5 for PH. #{@index} given." unless (0..5).cover?(@index)
33
+ OFFSET + (@index * DATA_SIZE)
34
+ end
35
+ def value
36
+ prop = Property.new(offset: get_offset, address: MEASURMENT, type: :short)
37
+ get_data(prop.to_s) / @divider
38
+ end
39
+ def hysteresis
40
+ prop = Property.new(offset: get_offset, address: HYSTERESIS, type: :char)
41
+ get_data(prop.to_s) / @divider
42
+ end
43
+ def day_high
44
+ prop = Property.new(offset: get_offset, address: DAY_HIGH, type: :short)
45
+ get_data(prop.to_s) / @divider
46
+ end
47
+ def day_low
48
+ prop = Property.new(offset: get_offset, address: DAY_LOW, type: :short)
49
+ get_data(prop.to_s) / @divider
50
+ end
51
+ def night_high
52
+ prop = Property.new(offset: get_offset, address: NIGHT_HIGH, type: :short)
53
+ get_data(prop.to_s) / @divider
54
+ end
55
+ def night_low
56
+ prop = Property.new(offset: get_offset, address: NIGHT_LOW, type: :short)
57
+ get_data(prop.to_s) / @divider
58
+ end
59
+ def alarm_high
60
+ prop = Property.new(offset: get_offset, address: ALARM_HIGH, type: :short)
61
+ get_data(prop.to_s) / @divider
62
+ end
63
+ def alarm_low
64
+ prop = Property.new(offset: get_offset, address: ALARM_LOW, type: :short)
65
+ get_data(prop.to_s) / @divider
66
+ end
67
+ def flags
68
+ prop = Property.new(offset: get_offset, address: FLAGS, type: :short)
69
+ get_data(prop.to_s)
70
+ end
71
+ def flag_present
72
+ flags & FLAG_PRESENT != 0
73
+ end
74
+ def flag_rise
75
+ return false unless flag_present
76
+ flags & FLAG_RISE != 0
77
+ end
78
+ def flag_lower
79
+ return false unless flag_present
80
+ flags & FLAG_LOWER != 0
81
+ end
82
+ def flag_alarm_high
83
+ return false unless flag_present
84
+ flags & FLAG_ALARM_HIGH != 0
85
+ end
86
+ def flag_alarm_low
87
+ return false unless flag_present
88
+ flags & FLAG_ALARM_LOW != 0
89
+ end
90
+
91
+ def inspect
92
+ puts "MEASURMENT\t#{value}"
93
+ puts "HYSTERESIS\t#{hysteresis}"
94
+ puts "DAY LOW\t\t#{day_low}"
95
+ puts "DAY HIGH\t#{day_high}"
96
+ puts "NIGHT LOW\t#{night_low}"
97
+ puts "NIGHT HIGH\t#{night_high}"
98
+ puts "ALARM LOW\t#{alarm_low}"
99
+ puts "ALARM HIGH\t#{alarm_high}"
100
+ puts "FLAGS:\t\t#{flags} (#{flags.to_s(2)})"
101
+ puts "F_PRESENT:\t#{flag_present}"
102
+ puts "F_RISE:\t\t#{flag_rise}"
103
+ puts "F_LOWER:\t#{flag_lower}"
104
+ puts "F_ALARM_H:\t#{flag_alarm_high}"
105
+ puts "F_ALARM_L:\t#{flag_alarm_low}"
106
+ end
107
+
108
+ private
109
+ def get_data(address)
110
+ @device.request_data(address)
111
+ end
112
+ end
113
+ end
114
+ end
@@ -0,0 +1,107 @@
1
+ require "elemac/sensor/default"
2
+ require "elemac/converter"
3
+
4
+ # Index can be only 0 or 1, defined by Length in PhSettings.cs
5
+ # to jest inny index (tylko dla PH) bo odwołujemy się do innego offsetu (ph offset)
6
+ # CALIBR_7_OFFSET = 8 # not used in elemac
7
+ # CALIBR_49_OFFSET = 9 # not used in elemac
8
+ module Elemac
9
+ class Sensor::Ph < Sensor::Default
10
+ PH_OFFSET = 2786
11
+ PH_DATA_SIZE = 15
12
+ CALIBR_7 = 1
13
+ CALIBR_49 = 3
14
+ CALIBR_DATE = 5
15
+ CALIBR_FLAGS = 0 # zwraca random, czasem 0,24,16 - brane sa 3 pierwsze bity pod uwage
16
+ SLOPE = 10
17
+ ELZERO = 12
18
+ REMINDER = 14
19
+ ADC_OFFSET = 0
20
+ ADC1 = 228
21
+ ADC2 = 1540
22
+
23
+ # -4 poniewaz index sensora jest 4 lub 5 dla ph, a dane offsetowe do kalibracji moga
24
+ # przyjmowac index 0 lub 1 wyłącznie co odpowiada indeksowi 0 lub 1 w zaleznosci od czujnika
25
+ def get_ph_offset
26
+ index = (@index - 4)
27
+ throw "Bad offset. Can be only 0 or 1. #{index} given." unless (0..1).cover?(index)
28
+ PH_OFFSET + (index * PH_DATA_SIZE)
29
+ end
30
+
31
+ def calibr_7
32
+ prop = Property.new(offset: get_ph_offset, address: CALIBR_7, type: :short)
33
+ get_data(prop.to_s)
34
+ end
35
+
36
+ # TODO: chyba nie sprowadza dobrych wartosci dla ph9 (218) - wyswietla tyle co w 4
37
+ def calibr_49
38
+ prop = Property.new(offset: get_ph_offset, address: CALIBR_49, type: :short)
39
+ get_data(prop.to_s)
40
+ end
41
+
42
+ def calibr_flags
43
+ prop = Property.new(offset: get_ph_offset, address: CALIBR_FLAGS, type: :char)
44
+ get_data(prop.to_s)
45
+ end
46
+
47
+ def calibr_date
48
+ prop = Property.new(offset: get_ph_offset, address: CALIBR_DATE, type: :int)
49
+ Converter.long_to_date(get_data(prop.to_s))
50
+ end
51
+
52
+ # a warning should be shown when this value is > 22240
53
+ def slope
54
+ prop = Property.new(offset: get_ph_offset, address: SLOPE, type: :short)
55
+ get_data(prop.to_s)
56
+ end
57
+
58
+ def slope_mv_ph
59
+ current_slope = slope
60
+ if current_slope == 0
61
+ 0
62
+ else
63
+ 1638400.0 / current_slope * (625.0 / 128.0) / 11.9090900421143
64
+ end
65
+ end
66
+
67
+ def elzero
68
+ prop = Property.new(offset: get_ph_offset, address: ELZERO, type: :short)
69
+ get_data(prop.to_s)
70
+ end
71
+
72
+ def reminder
73
+ prop = Property.new(offset: get_ph_offset, address: REMINDER, type: :char)
74
+ get_data(prop.to_s)
75
+ end
76
+
77
+ def calibr_status
78
+ Converter.ph_reminder(calibr_flags, calibr_date, reminder)
79
+ end
80
+
81
+ def ph_9_enabled
82
+ Converter.ph_9_enabled(calibr_flags)
83
+ end
84
+
85
+ def adc
86
+ # depending on current ph 0 or 1, get adc value
87
+ adc_address = (@index - 4) == 0 ? ADC1 : ADC2
88
+ prop = Property.new(offset: ADC_OFFSET, address: ADC1, type: :short)
89
+ get_data(prop.to_s)
90
+ end
91
+
92
+ def inspect
93
+ super
94
+ puts "ADC\t\t#{adc}"
95
+ puts "CALIBR 7\t#{calibr_7}"
96
+ puts "CALIBR 49\t#{calibr_49}"
97
+ puts "PH9 ENABLED\t#{ph_9_enabled}"
98
+ puts "CALIBR_FLAGS\t#{calibr_flags} (#{(calibr_flags & 7).to_s(2)})"
99
+ puts "CALIBR_DATE\t#{calibr_date}"
100
+ puts "SLOPE\t\t#{slope}"
101
+ puts "SLOPE mV/pH\t#{slope_mv_ph}"
102
+ puts "ELZERO\t\t#{elzero}"
103
+ puts "REMINDER\t#{reminder}"
104
+ puts "STATUS\t\t#{calibr_status}"
105
+ end
106
+ end
107
+ end
@@ -0,0 +1,9 @@
1
+ require "elemac/sensor/default"
2
+
3
+ module Elemac
4
+ class Sensor::Redox < Sensor::Default
5
+ def initialize
6
+ throw 'Not yet supported'
7
+ end
8
+ end
9
+ end
@@ -0,0 +1,6 @@
1
+ require "elemac/sensor/default"
2
+
3
+ module Elemac
4
+ class Sensor::Temperature < Sensor::Default
5
+ end
6
+ end
@@ -0,0 +1,51 @@
1
+ # Sensors are described as 4 blocks of memory starting from offset 1864
2
+ # those sensors are set as: temp1, temp2, temp3, temp4, ph1... dunno about ph2 and redox
3
+ # measure is on +2. size of 17. so first reading is at 1864+2
4
+ # ph starts at 1864 + 4*17. +2 is the ph measure
5
+
6
+ require "elemac/sensor/temperature"
7
+ require "elemac/sensor/ph"
8
+ require "elemac/sensor/redox"
9
+
10
+ module Elemac
11
+ class Sensors
12
+ def initialize(device:)
13
+ @device = device
14
+ # TODO: check loaded sensors
15
+ @temp1_available = true
16
+ @temp2_available = false
17
+ @temp3_available = false
18
+ @temp4_available = false
19
+ @ph1_available = true
20
+ @ph2_available = false
21
+ end
22
+ def temp1
23
+ throw 'Sensor temp1 unavailable' unless @temp1_available
24
+ Sensor::Temperature.new(device: @device, index: 0, divider: 10.0)
25
+ end
26
+ def temp2
27
+ throw 'Sensor temp2 unavailable' unless @temp2_available
28
+ Sensor::Temperature.new(device: @device, index: 1, divider: 10.0)
29
+ end
30
+ def temp3
31
+ throw 'Sensor temp3 unavailable' unless @temp3_available
32
+ Sensor::Temperature.new(device: @device, index: 2, divider: 10.0)
33
+ end
34
+ def temp4
35
+ throw 'Sensor temp4 unavailable' unless @temp4_available
36
+ Sensor::Temperature.new(device: @device, index: 3, divider: 10.0)
37
+ end
38
+ def ph1
39
+ throw 'Sensor ph1 unavailable' unless @ph1_available
40
+ Sensor::Ph.new(device: @device, index: 4, divider: 100.0)
41
+ end
42
+ def ph2
43
+ throw 'Sensor ph1 unavailable' unless @ph2_available
44
+ Sensor::Ph.new(device: @device, index: 5, divider: 100.0)
45
+ end
46
+ #def rh
47
+ # throw 'Sensor rh unavailable' unless @ph2_available
48
+ # Sensor::Redox.new(device: @device, index: 6, divider: 100.0)
49
+ #end
50
+ end
51
+ end
@@ -0,0 +1,3 @@
1
+ module Elemac
2
+ VERSION = "0.1.0"
3
+ end
metadata ADDED
@@ -0,0 +1,110 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: elemac
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Michał Prostko
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2017-09-08 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: hidapi
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: 0.1.9
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: 0.1.9
27
+ - !ruby/object:Gem::Dependency
28
+ name: bundler
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '1.15'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '1.15'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rake
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '10.0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '10.0'
55
+ description: Ruby binding for ELEMAC SA-03 aquarium controller
56
+ email:
57
+ executables: []
58
+ extensions: []
59
+ extra_rdoc_files: []
60
+ files:
61
+ - ".gitignore"
62
+ - Gemfile
63
+ - LICENSE
64
+ - README.md
65
+ - Rakefile
66
+ - bin/console
67
+ - bin/setup
68
+ - elemac.gemspec
69
+ - lib/elemac.rb
70
+ - lib/elemac/connection.rb
71
+ - lib/elemac/converter.rb
72
+ - lib/elemac/light.rb
73
+ - lib/elemac/output/default.rb
74
+ - lib/elemac/output/power.rb
75
+ - lib/elemac/output/pwm.rb
76
+ - lib/elemac/output/ttl.rb
77
+ - lib/elemac/outputs.rb
78
+ - lib/elemac/overview.rb
79
+ - lib/elemac/property.rb
80
+ - lib/elemac/sensor/default.rb
81
+ - lib/elemac/sensor/ph.rb
82
+ - lib/elemac/sensor/redox.rb
83
+ - lib/elemac/sensor/temperature.rb
84
+ - lib/elemac/sensors.rb
85
+ - lib/elemac/version.rb
86
+ homepage: https://github.com/mprostko/elemac
87
+ licenses:
88
+ - MIT
89
+ metadata: {}
90
+ post_install_message:
91
+ rdoc_options: []
92
+ require_paths:
93
+ - lib
94
+ required_ruby_version: !ruby/object:Gem::Requirement
95
+ requirements:
96
+ - - ">="
97
+ - !ruby/object:Gem::Version
98
+ version: '2.0'
99
+ required_rubygems_version: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - ">="
102
+ - !ruby/object:Gem::Version
103
+ version: '0'
104
+ requirements: []
105
+ rubyforge_project:
106
+ rubygems_version: 2.6.11
107
+ signing_key:
108
+ specification_version: 4
109
+ summary: Ruby binding for ELEMAC SA-03 aquarium controller
110
+ test_files: []