Fork of the espurna firmware for `mhsw` switches
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1040 lines
30 KiB

Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
providers: relays, lights and buttons refactoring (#2414) - gpio module now tracks the known providers (right now, hardware and mcp expander) - refactored relay struct to use 'Provider' implementing setup,notify,change,boot instead of just BasePin actions - refactored button module to use gpio provider instead of referencing types itself - removed dual & stm code from buttons, migrate both to relay module - added status notify and change callbacks for relayStatus (i.e. 'notify' when relay status was called, but not changed. and 'changed' when it did) - relays runtime configuration keys - relay command now shows configured relays and current & target statuses - refactor the code using relayStatus(0, blah) under LIGHT_PROVIDER check to use lightState instead - remove rfbridge code form relay module. implement through a basic state listener in the rfbridge module, depend on RELAY_SUPPORT - allow to bind rf codes to real relays - drop tuya-specific lights provider, remove tuya code from relays and lights modules - integrate tuya via relay listeners and providers, use lights custom provider - implement channel transitions for tuya. disabled by default, and transition time and step are overridden to 2000 + 100. needs to be set to some value below the total time (i.e. `total transition time / step time == number of steps`, we need to figure out a correct time that serial comms could handle) - lights custom provider (global, not per-pin) and state listeners - remove lights code from relay module. implement through providers & listeners in the lights module, depend on RELAY_SUPPORT - lights per-channel relay provider (unused atm), depend on RELAY_SUPPORT - refactored channel transition - calculate step only once, make sure time + step values are sane, generate quick transitions with very small delay (10ms hardcoded) for transitions during OFF state i.e. we no longer waste 500ms (or whatever transition time is set to) on boot doing nothing - transition time + step parameter for the lightUpdate - report mask parameter for the lightUpdate - minor fixes across the board resolve #2222
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
providers: relays, lights and buttons refactoring (#2414) - gpio module now tracks the known providers (right now, hardware and mcp expander) - refactored relay struct to use 'Provider' implementing setup,notify,change,boot instead of just BasePin actions - refactored button module to use gpio provider instead of referencing types itself - removed dual & stm code from buttons, migrate both to relay module - added status notify and change callbacks for relayStatus (i.e. 'notify' when relay status was called, but not changed. and 'changed' when it did) - relays runtime configuration keys - relay command now shows configured relays and current & target statuses - refactor the code using relayStatus(0, blah) under LIGHT_PROVIDER check to use lightState instead - remove rfbridge code form relay module. implement through a basic state listener in the rfbridge module, depend on RELAY_SUPPORT - allow to bind rf codes to real relays - drop tuya-specific lights provider, remove tuya code from relays and lights modules - integrate tuya via relay listeners and providers, use lights custom provider - implement channel transitions for tuya. disabled by default, and transition time and step are overridden to 2000 + 100. needs to be set to some value below the total time (i.e. `total transition time / step time == number of steps`, we need to figure out a correct time that serial comms could handle) - lights custom provider (global, not per-pin) and state listeners - remove lights code from relay module. implement through providers & listeners in the lights module, depend on RELAY_SUPPORT - lights per-channel relay provider (unused atm), depend on RELAY_SUPPORT - refactored channel transition - calculate step only once, make sure time + step values are sane, generate quick transitions with very small delay (10ms hardcoded) for transitions during OFF state i.e. we no longer waste 500ms (or whatever transition time is set to) on boot doing nothing - transition time + step parameter for the lightUpdate - report mask parameter for the lightUpdate - minor fixes across the board resolve #2222
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
providers: relays, lights and buttons refactoring (#2414) - gpio module now tracks the known providers (right now, hardware and mcp expander) - refactored relay struct to use 'Provider' implementing setup,notify,change,boot instead of just BasePin actions - refactored button module to use gpio provider instead of referencing types itself - removed dual & stm code from buttons, migrate both to relay module - added status notify and change callbacks for relayStatus (i.e. 'notify' when relay status was called, but not changed. and 'changed' when it did) - relays runtime configuration keys - relay command now shows configured relays and current & target statuses - refactor the code using relayStatus(0, blah) under LIGHT_PROVIDER check to use lightState instead - remove rfbridge code form relay module. implement through a basic state listener in the rfbridge module, depend on RELAY_SUPPORT - allow to bind rf codes to real relays - drop tuya-specific lights provider, remove tuya code from relays and lights modules - integrate tuya via relay listeners and providers, use lights custom provider - implement channel transitions for tuya. disabled by default, and transition time and step are overridden to 2000 + 100. needs to be set to some value below the total time (i.e. `total transition time / step time == number of steps`, we need to figure out a correct time that serial comms could handle) - lights custom provider (global, not per-pin) and state listeners - remove lights code from relay module. implement through providers & listeners in the lights module, depend on RELAY_SUPPORT - lights per-channel relay provider (unused atm), depend on RELAY_SUPPORT - refactored channel transition - calculate step only once, make sure time + step values are sane, generate quick transitions with very small delay (10ms hardcoded) for transitions during OFF state i.e. we no longer waste 500ms (or whatever transition time is set to) on boot doing nothing - transition time + step parameter for the lightUpdate - report mask parameter for the lightUpdate - minor fixes across the board resolve #2222
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Terminal: change command-line parser (#2247) Change the underlying command line handling: - switch to a custom parser, inspired by redis / sds - update terminalRegisterCommand signature, pass only bare minimum - clean-up `help` & `commands`. update settings `set`, `get` and `del` - allow our custom test suite to run command-line tests - clean-up Stream IO to allow us to print large things into debug stream (for example, `eeprom.dump`) - send parsing errors to the debug log As a proof of concept, introduce `TERMINAL_MQTT_SUPPORT` and `TERMINAL_WEB_API_SUPPORT` - MQTT subscribes to the `<root>/cmd/set` and sends response to the `<root>/cmd`. We can't output too much, as we don't have any large-send API. - Web API listens to the `/api/cmd?apikey=...&line=...` (or PUT, params inside the body). This one is intended as a possible replacement of the `API_SUPPORT`. Internals introduce a 'task' around the AsyncWebServerRequest object that will simulate what WiFiClient does and push data into it continuously, switching between CONT and SYS. Both are experimental. We only accept a single command and not every command is updated to use Print `ctx.output` object. We are also somewhat limited by the Print / Stream overall, perhaps I am overestimating the usefulness of Arduino compatibility to such an extent :) Web API handler can also sometimes show only part of the result, whenever the command tries to yield() by itself waiting for something. Perhaps we would need to create a custom request handler for that specific use-case.
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
Update rpnlib to 0.23.x (#2274) ## [0.23.0] 2020-07-26 ### Added - `p` operator to print the top of the stack via debug function - `&var` syntax to create variable reference in expression - `=` operator for variable assignment in expression - `exists` operator to check for variable existance (only for references) - `deref` operator to convert variable reference into a value (only for references) - Allow to use either float or double as floating type, parse numbers in expressions as specified type - Add boolean type, parse `true` and `false` in expressions - Add null type, parse `null` in expressions - Add string type, parse double-quoted `"string"` in expressions - Add integer and unsigned integer type, used in operators - Allow to configure underlying types from rpnlib\_config.h and -D... flags - Return `rpn_error` from operators, split error types into categories - Create a new stack by using `[` keyword. Move stack contents into the previous stack + size by using `]`. ### Changed - Stack structure no longer holds raw `float`, but internal `rpn_value` type - rpn\_... setter and getter methods use `rpn_value` type - Operator functions return `rpn_error` type, allowing to return both value and operator errors - Variables in expressions are no longer required to exist when using `&var` Expression will automatically create the variable, set it to `null` and push it's reference on the stack - It is possible to create 'reference' stack values - Improve precision of `e` and `pi` ### Fixed - Proper value for `e` constant - Allow to use multiple contexts simultaniously, replace `rpn_error` and `rpn_debug_callback` with the current `rpn_context` members `error` and `debug_callback` respectively
4 years ago
  1. /*
  2. RPN RULES MODULE
  3. Use RPNLib library (https://github.com/xoseperez/rpnlib)
  4. Copyright (C) 2019 by Xose Pérez <xose dot perez at gmail dot com>
  5. */
  6. #include "rpnrules.h"
  7. #if RPN_RULES_SUPPORT
  8. #include <rpnlib.h>
  9. #include "broker.h"
  10. #include "light.h"
  11. #include "mqtt.h"
  12. #include "ntp.h"
  13. #include "ntp_timelib.h"
  14. #include "relay.h"
  15. #include "rfbridge.h"
  16. #include "rpc.h"
  17. #include "rtcmem.h"
  18. #include "sensor.h"
  19. #include "terminal.h"
  20. #include "wifi.h"
  21. #include "ws.h"
  22. #include <list>
  23. #include <type_traits>
  24. #include <vector>
  25. // -----------------------------------------------------------------------------
  26. // Custom commands
  27. // -----------------------------------------------------------------------------
  28. rpn_context _rpn_ctxt;
  29. bool _rpn_run = false;
  30. unsigned long _rpn_delay = RPN_DELAY;
  31. unsigned long _rpn_last = 0;
  32. struct RpnRunner {
  33. enum class Policy {
  34. OneShot,
  35. Periodic
  36. };
  37. RpnRunner(Policy policy_, uint32_t period_) :
  38. policy(policy_),
  39. period(period_),
  40. last(millis())
  41. {}
  42. Policy policy { Policy::Periodic };
  43. uint32_t period { 0ul };
  44. uint32_t last { 0ul };
  45. bool expired { false };
  46. };
  47. std::vector<RpnRunner> _rpn_runners;
  48. rpn_operator_error _rpnRunnerHandler(rpn_context & ctxt, RpnRunner::Policy policy, uint32_t time) {
  49. for (auto& runner : _rpn_runners) {
  50. if ((policy == runner.policy) && (time == runner.period)) {
  51. return runner.expired
  52. ? rpn_operator_error::Ok
  53. : rpn_operator_error::CannotContinue;
  54. }
  55. }
  56. _rpn_runners.emplace_back(policy, time);
  57. return rpn_operator_error::CannotContinue;
  58. }
  59. // -----------------------------------------------------------------------------
  60. bool _rpnWebSocketOnKeyCheck(const char * key, JsonVariant& value) {
  61. return (strncmp(key, "rpn", 3) == 0);
  62. }
  63. void _rpnWebSocketOnConnected(JsonObject& root) {
  64. root["rpnSticky"] = getSetting("rpnSticky", 1 == RPN_STICKY);
  65. root["rpnDelay"] = getSetting("rpnDelay", RPN_DELAY);
  66. JsonArray& rules = root.createNestedArray("rpnRules");
  67. unsigned char i = 0;
  68. String rule = getSetting({"rpnRule", i});
  69. while (rule.length()) {
  70. rules.add(rule);
  71. rule = getSetting({"rpnRule", ++i});
  72. }
  73. #if MQTT_SUPPORT
  74. i=0;
  75. JsonArray& topics = root.createNestedArray("rpnTopics");
  76. JsonArray& names = root.createNestedArray("rpnNames");
  77. String rpn_topic = getSetting({"rpnTopic", i});
  78. while (rpn_topic.length() > 0) {
  79. String rpn_name = getSetting({"rpnName", i});
  80. topics.add(rpn_topic);
  81. names.add(rpn_name);
  82. rpn_topic = getSetting({"rpnTopic", ++i});
  83. }
  84. #endif
  85. }
  86. #if MQTT_SUPPORT
  87. void _rpnMQTTSubscribe() {
  88. unsigned char i = 0;
  89. String rpn_topic = getSetting({"rpnTopic", i});
  90. while (rpn_topic.length()) {
  91. mqttSubscribeRaw(rpn_topic.c_str());
  92. rpn_topic = getSetting({"rpnTopic", ++i});
  93. }
  94. }
  95. void _rpnMQTTCallback(unsigned int type, const char * topic, const char * payload) {
  96. if (type == MQTT_CONNECT_EVENT) {
  97. _rpnMQTTSubscribe();
  98. }
  99. if (type == MQTT_MESSAGE_EVENT) {
  100. unsigned char i = 0;
  101. String rpn_topic = getSetting({"rpnTopic", i});
  102. while (rpn_topic.length()) {
  103. if (rpn_topic.equals(topic)) {
  104. String rpn_name = getSetting({"rpnName", i});
  105. if (rpn_name.length()) {
  106. rpn_value value { atof(payload) };
  107. rpn_variable_set(_rpn_ctxt, rpn_name, value);
  108. _rpn_run = true;
  109. break;
  110. }
  111. }
  112. rpn_topic = getSetting({"rpnTopic", ++i});
  113. }
  114. }
  115. }
  116. #endif // MQTT_SUPPORT
  117. void _rpnConfigure() {
  118. #if MQTT_SUPPORT
  119. if (mqttConnected()) _rpnMQTTSubscribe();
  120. #endif
  121. _rpn_delay = getSetting("rpnDelay", RPN_DELAY);
  122. }
  123. void _rpnBrokerCallback(const String& topic, unsigned char id, double value, const char*) {
  124. char name[32] = {0};
  125. snprintf(name, sizeof(name), "%s%u", topic.c_str(), id);
  126. if (topic == MQTT_TOPIC_RELAY) {
  127. rpn_variable_set(_rpn_ctxt, name, rpn_value(static_cast<bool>(value)));
  128. } else {
  129. rpn_variable_set(_rpn_ctxt, name, rpn_value(value));
  130. }
  131. _rpn_run = true;
  132. }
  133. void _rpnBrokerStatus(const String& topic, unsigned char id, unsigned int value) {
  134. _rpnBrokerCallback(topic, id, double(value), nullptr);
  135. }
  136. #if NTP_SUPPORT
  137. namespace {
  138. constexpr bool time_t_is_32bit { sizeof(time_t) == 4 };
  139. constexpr bool time_t_is_64bit { sizeof(time_t) == 8 };
  140. static_assert(time_t_is_32bit || time_t_is_64bit, "");
  141. template <typename T>
  142. using split_t = std::integral_constant<bool, sizeof(T) == 8>;
  143. using RpnNtpFunc = rpn_int(*)(time_t);
  144. rpn_error _rpnNtpPopTimestampPair(rpn_context& ctxt, RpnNtpFunc func) {
  145. rpn_value rhs = rpn_stack_pop(ctxt);
  146. rpn_value lhs = rpn_stack_pop(ctxt);
  147. auto timestamp = (static_cast<long long>(lhs.toInt()) << 32ll)
  148. | (static_cast<long long>(rhs.toInt()));
  149. rpn_value value(func(timestamp));
  150. rpn_stack_push(ctxt, value);
  151. return 0;
  152. }
  153. rpn_error _rpnNtpPopTimestampSingle(rpn_context& ctxt, RpnNtpFunc func) {
  154. rpn_value input = rpn_stack_pop(ctxt);
  155. rpn_value result(func(input.toInt()));
  156. rpn_stack_push(ctxt, result);
  157. return 0;
  158. }
  159. void _rpnNtpPushTimestampPair(rpn_context& ctxt, time_t timestamp) {
  160. rpn_value lhs(static_cast<rpn_int>((static_cast<long long>(timestamp) >> 32ll) & 0xffffffffll));
  161. rpn_value rhs(static_cast<rpn_int>(static_cast<long long>(timestamp) & 0xffffffffll));
  162. rpn_stack_push(ctxt, lhs);
  163. rpn_stack_push(ctxt, rhs);
  164. }
  165. void _rpnNtpPushTimestampSingle(rpn_context& ctxt, time_t timestamp) {
  166. rpn_value result(static_cast<rpn_int>(timestamp));
  167. rpn_stack_push(ctxt, result);
  168. }
  169. inline rpn_error _rpnNtpPopTimestamp(const std::true_type&, rpn_context& ctxt, RpnNtpFunc func) {
  170. return _rpnNtpPopTimestampPair(ctxt, func);
  171. }
  172. inline rpn_error _rpnNtpPopTimestamp(const std::false_type&, rpn_context& ctxt, RpnNtpFunc func) {
  173. return _rpnNtpPopTimestampSingle(ctxt, func);
  174. }
  175. rpn_error _rpnNtpPopTimestamp(rpn_context& ctxt, RpnNtpFunc func) {
  176. return _rpnNtpPopTimestamp(split_t<time_t>{}, ctxt, func);
  177. }
  178. inline void _rpnNtpPushTimestamp(const std::true_type&, rpn_context& ctxt, time_t timestamp) {
  179. _rpnNtpPushTimestampPair(ctxt, timestamp);
  180. }
  181. inline void _rpnNtpPushTimestamp(const std::false_type&, rpn_context& ctxt, time_t timestamp) {
  182. _rpnNtpPushTimestampSingle(ctxt, timestamp);
  183. }
  184. void _rpnNtpPushTimestamp(rpn_context& ctxt, time_t timestamp) {
  185. _rpnNtpPushTimestamp(split_t<time_t>{}, ctxt, timestamp);
  186. }
  187. rpn_error _rpnNtpNow(rpn_context & ctxt) {
  188. if (ntpSynced()) {
  189. _rpnNtpPushTimestamp(ctxt, now());
  190. return 0;
  191. }
  192. return rpn_operator_error::CannotContinue;
  193. }
  194. rpn_error _rpnNtpFunc(rpn_context & ctxt, RpnNtpFunc func) {
  195. return _rpnNtpPopTimestamp(ctxt, func);
  196. }
  197. bool _rpn_ntp_tick_minute { false };
  198. bool _rpn_ntp_tick_hour { false };
  199. rpn_error _rpnNtpTickMinute(rpn_context& ctxt) {
  200. if (_rpn_ntp_tick_minute) {
  201. _rpn_ntp_tick_minute = false;
  202. return 0;
  203. }
  204. return rpn_operator_error::CannotContinue;
  205. }
  206. rpn_error _rpnNtpTickHour(rpn_context& ctxt) {
  207. if (_rpn_ntp_tick_hour) {
  208. _rpn_ntp_tick_hour = false;
  209. return 0;
  210. }
  211. return rpn_operator_error::CannotContinue;
  212. }
  213. } // namespace
  214. #endif // NTP_SUPPORT
  215. String _rpnValueToString(const rpn_value& value) {
  216. String out;
  217. if (value.isString()) {
  218. out = value.toString();
  219. } else if (value.isFloat()) {
  220. out = String(value.toFloat(), 10);
  221. } else if (value.isInt()) {
  222. out = String(value.toInt(), 10);
  223. } else if (value.isUint()) {
  224. out = String(value.toUint(), 10);
  225. } else if (value.isBoolean()) {
  226. out = String(value.toBoolean() ? "true" : "false");
  227. } else if (value.isNull()) {
  228. out = F("(null)");
  229. }
  230. return out;
  231. }
  232. char _rpnStackTypeTag(rpn_stack_value::Type type) {
  233. switch (type) {
  234. case rpn_stack_value::Type::None:
  235. return 'N';
  236. case rpn_stack_value::Type::Variable:
  237. return '$';
  238. case rpn_stack_value::Type::Array:
  239. return 'A';
  240. case rpn_stack_value::Type::Value:
  241. default:
  242. return ' ';
  243. }
  244. }
  245. #if RELAY_SUPPORT
  246. rpn_error _rpnRelayStatus(rpn_context & ctxt, bool force) {
  247. rpn_value id;
  248. rpn_value status;
  249. rpn_stack_pop(ctxt, id);
  250. rpn_stack_pop(ctxt, status);
  251. rpn_uint value = status.toUint();
  252. if (value == 2) {
  253. relayToggle(id.toUint());
  254. } else if (relayStatusTarget(id.toUint()) != (value == 1)) {
  255. relayStatus(id.toUint(), value == 1);
  256. }
  257. return 0;
  258. }
  259. #endif // RELAY_SUPPORT
  260. #if RFB_SUPPORT
  261. struct rpn_rfbridge_code {
  262. unsigned char protocol;
  263. String raw;
  264. size_t count;
  265. decltype(millis()) last;
  266. };
  267. // TODO: in theory, we could do with forward_list. however, this would require a more complicated removal process,
  268. // as we would no longer know the previous element and would need to track 2 elements at a time
  269. static std::list<rpn_rfbridge_code> _rfb_codes;
  270. static uint32_t _rfb_code_repeat_window;
  271. static uint32_t _rfb_code_stale_delay;
  272. static uint32_t _rfb_code_match_window;
  273. struct rpn_rfbridge_match {
  274. unsigned char protocol;
  275. String raw;
  276. };
  277. rpn_error _rpnRfbSequence(rpn_context& ctxt) {
  278. auto raw_second = rpn_stack_pop(ctxt);
  279. auto proto_second = rpn_stack_pop(ctxt);
  280. auto raw_first = rpn_stack_pop(ctxt);
  281. auto proto_first = rpn_stack_pop(ctxt);
  282. // find 2 codes in the same order and save pointers
  283. rpn_rfbridge_match match[2] {
  284. {static_cast<unsigned char>(proto_first.toUint()), raw_first.toString()},
  285. {static_cast<unsigned char>(proto_second.toUint()), raw_second.toString()}
  286. };
  287. rpn_rfbridge_code* refs[2] {nullptr, nullptr};
  288. for (auto& recent : _rfb_codes) {
  289. if ((refs[0] != nullptr) && (refs[1] != nullptr)) {
  290. break;
  291. }
  292. for (int index = 0; index < 2; ++index) {
  293. if ((refs[index] == nullptr)
  294. && (match[index].protocol == recent.protocol)
  295. && (match[index].raw == recent.raw)) {
  296. refs[index] = &recent;
  297. }
  298. }
  299. }
  300. if ((refs[0] == nullptr) || (refs[1] == nullptr)) {
  301. return rpn_operator_error::CannotContinue;
  302. }
  303. // purge codes to avoid matching again on the next rules run
  304. if ((millis() - refs[0]->last) > (millis() - refs[1]->last)) {
  305. _rfb_codes.remove_if([&refs](rpn_rfbridge_code& code) {
  306. return (refs[0] == &code) || (refs[1] == &code);
  307. });
  308. return rpn_operator_error::Ok;
  309. }
  310. return rpn_operator_error::CannotContinue;
  311. }
  312. decltype(_rfb_codes)::iterator _rpnRfbFindCode(unsigned char protocol, const String& match) {
  313. return std::find_if(_rfb_codes.begin(), _rfb_codes.end(), [protocol, &match](const rpn_rfbridge_code& code) {
  314. return (code.protocol == protocol) && (code.raw == match);
  315. });
  316. }
  317. rpn_error _rpnRfbSend(rpn_context& ctxt) {
  318. auto code = rpn_stack_pop(ctxt);
  319. if (!code.isString()) {
  320. return rpn_operator_error::InvalidArgument;
  321. }
  322. rfbSend(code.toString());
  323. return rpn_operator_error::Ok;
  324. }
  325. rpn_error _rpnRfbPop(rpn_context& ctxt) {
  326. auto code = rpn_stack_pop(ctxt);
  327. auto proto = rpn_stack_pop(ctxt);
  328. auto result = _rpnRfbFindCode(proto.toUint(), code.toString());
  329. if (result == _rfb_codes.end()) {
  330. return rpn_operator_error::CannotContinue;
  331. }
  332. _rfb_codes.erase(result);
  333. return rpn_operator_error::Ok;
  334. }
  335. rpn_error _rpnRfbInfo(rpn_context& ctxt) {
  336. auto code = rpn_stack_pop(ctxt);
  337. auto proto = rpn_stack_pop(ctxt);
  338. auto result = _rpnRfbFindCode(proto.toUint(), code.toString());
  339. if (result == _rfb_codes.end()) {
  340. return rpn_operator_error::CannotContinue;
  341. }
  342. rpn_stack_push(ctxt, rpn_value(
  343. static_cast<rpn_uint>((*result).count)));
  344. rpn_stack_push(ctxt, rpn_value(
  345. static_cast<rpn_uint>((*result).last)));
  346. return rpn_operator_error::Ok;
  347. }
  348. rpn_error _rpnRfbWaitMatch(rpn_context& ctxt) {
  349. auto code = rpn_stack_pop(ctxt);
  350. auto proto = rpn_stack_pop(ctxt);
  351. auto count = rpn_stack_pop(ctxt);
  352. auto time = rpn_stack_pop(ctxt);
  353. auto result = _rpnRfbFindCode(proto.toUint(), code.toString());
  354. if (result == _rfb_codes.end()) {
  355. return rpn_operator_error::CannotContinue;
  356. }
  357. if ((*result).count < count.toUint()) {
  358. return rpn_operator_error::CannotContinue;
  359. }
  360. // purge code to avoid matching again on the next rules run
  361. if (rpn_operator_error::Ok == _rpnRunnerHandler(ctxt, RpnRunner::Policy::OneShot, time.toUint())) {
  362. _rfb_codes.erase(result);
  363. return rpn_operator_error::Ok;
  364. }
  365. return rpn_operator_error::CannotContinue;
  366. }
  367. rpn_error _rpnRfbMatcher(rpn_context& ctxt) {
  368. auto code = rpn_stack_pop(ctxt);
  369. auto proto = rpn_stack_pop(ctxt);
  370. auto count = rpn_stack_pop(ctxt);
  371. auto result = _rpnRfbFindCode(proto.toUint(), code.toString());
  372. if (result == _rfb_codes.end()) {
  373. return rpn_operator_error::CannotContinue;
  374. }
  375. // only process recent codes, ignore when rule is processing outside of this small window
  376. if (millis() - (*result).last >= _rfb_code_match_window) {
  377. return rpn_operator_error::CannotContinue;
  378. }
  379. // purge code to avoid matching again on the next rules run
  380. if ((*result).count == count.toUint()) {
  381. _rfb_codes.erase(result);
  382. return rpn_operator_error::Ok;
  383. }
  384. return rpn_operator_error::CannotContinue;
  385. }
  386. void _rpnBrokerRfbridgeCallback(unsigned char protocol, const char* raw_code) {
  387. // remove really old codes that we have not seen in a while to avoid memory exhaustion
  388. auto ts = millis();
  389. auto old = std::remove_if(_rfb_codes.begin(), _rfb_codes.end(), [ts](rpn_rfbridge_code& code) {
  390. return (ts - code.last) >= _rfb_code_stale_delay;
  391. });
  392. if (old != _rfb_codes.end()) {
  393. _rfb_codes.erase(old, _rfb_codes.end());
  394. }
  395. auto result = _rpnRfbFindCode(protocol, raw_code);
  396. if (result != _rfb_codes.end()) {
  397. // we also need to reset the counter at a certain point to allow next batch of repeats to go through
  398. if (millis() - (*result).last >= _rfb_code_repeat_window) {
  399. (*result).count = 0;
  400. }
  401. (*result).last = millis();
  402. (*result).count += 1u;
  403. } else {
  404. _rfb_codes.push_back({protocol, raw_code, 1u, millis()});
  405. }
  406. _rpn_run = true;
  407. }
  408. void _rpnRfbSetup() {
  409. // - Repeat window is an arbitrary time, just about 3-4 more times it takes for
  410. // a code to be sent again when holding a generic remote button
  411. // Code counter is reset to 0 when outside of the window.
  412. // - Stale delay allows broker callback to remove really old codes.
  413. // (TODO: can this happen in loop() cb instead?)
  414. _rfb_code_repeat_window = getSetting("rfbRepeatWindow", 2000ul);
  415. _rfb_code_match_window = getSetting("rfbMatchWindow", 2000ul);
  416. _rfb_code_stale_delay = getSetting("rfbStaleDelay", 10000ul);
  417. #if TERMINAL_SUPPORT
  418. terminalRegisterCommand(F("RFB.CODES"), [](const terminal::CommandContext& ctx) {
  419. for (auto& code : _rfb_codes) {
  420. char buffer[128] = {0};
  421. snprintf_P(buffer, sizeof(buffer),
  422. PSTR("proto=%u raw=\"%s\" count=%u last=%u"),
  423. code.protocol,
  424. code.raw.c_str(),
  425. code.count,
  426. code.last
  427. );
  428. ctx.output.println(buffer);
  429. }
  430. });
  431. #endif
  432. // Main bulk of the processing goes on in here
  433. RfbridgeBroker::Register(_rpnBrokerRfbridgeCallback);
  434. }
  435. #endif // RFB_SUPPORT
  436. void _rpnShowStack(Print& print) {
  437. print.println(F("Stack:"));
  438. auto index = rpn_stack_size(_rpn_ctxt);
  439. if (!index) {
  440. print.println(F(" (empty)"));
  441. return;
  442. }
  443. rpn_stack_foreach(_rpn_ctxt, [&index, &print](rpn_stack_value::Type type, const rpn_value& value) {
  444. print.printf("%c %02u: %s\n",
  445. _rpnStackTypeTag(type), index--,
  446. _rpnValueToString(value).c_str()
  447. );
  448. });
  449. }
  450. void _rpnInit() {
  451. // Init context
  452. rpn_init(_rpn_ctxt);
  453. // Time functions need NTP support
  454. // TODO: since 1.15.0, timelib+ntpclientlib are no longer used with latest Cores
  455. // `now` is always in UTC, `utc_...` functions to be used instead to convert time
  456. #if NTP_SUPPORT && !NTP_LEGACY_SUPPORT
  457. {
  458. constexpr size_t time_t_argc { split_t<time_t>{} ? 2 : 1 };
  459. rpn_operator_set(_rpn_ctxt, "tick_1h", 0, _rpnNtpTickHour);
  460. rpn_operator_set(_rpn_ctxt, "tick_1m", 0, _rpnNtpTickMinute);
  461. rpn_operator_set(_rpn_ctxt, "utc", 0, _rpnNtpNow);
  462. rpn_operator_set(_rpn_ctxt, "now", 0, _rpnNtpNow);
  463. rpn_operator_set(_rpn_ctxt, "utc_month", time_t_argc, [](rpn_context & ctxt) {
  464. return _rpnNtpFunc(ctxt, utc_month);
  465. });
  466. rpn_operator_set(_rpn_ctxt, "month", time_t_argc, [](rpn_context & ctxt) {
  467. return _rpnNtpFunc(ctxt, month);
  468. });
  469. rpn_operator_set(_rpn_ctxt, "utc_day", time_t_argc, [](rpn_context & ctxt) {
  470. return _rpnNtpFunc(ctxt, utc_day);
  471. });
  472. rpn_operator_set(_rpn_ctxt, "day", time_t_argc, [](rpn_context & ctxt) {
  473. return _rpnNtpFunc(ctxt, day);
  474. });
  475. rpn_operator_set(_rpn_ctxt, "utc_dow", time_t_argc, [](rpn_context & ctxt) {
  476. return _rpnNtpFunc(ctxt, utc_weekday);
  477. });
  478. rpn_operator_set(_rpn_ctxt, "dow", time_t_argc, [](rpn_context & ctxt) {
  479. return _rpnNtpFunc(ctxt, weekday);
  480. });
  481. rpn_operator_set(_rpn_ctxt, "utc_hour", time_t_argc, [](rpn_context & ctxt) {
  482. return _rpnNtpFunc(ctxt, utc_hour);
  483. });
  484. rpn_operator_set(_rpn_ctxt, "hour", time_t_argc, [](rpn_context & ctxt) {
  485. return _rpnNtpFunc(ctxt, hour);
  486. });
  487. rpn_operator_set(_rpn_ctxt, "utc_minute", time_t_argc, [](rpn_context & ctxt) {
  488. return _rpnNtpFunc(ctxt, utc_minute);
  489. });
  490. rpn_operator_set(_rpn_ctxt, "minute", time_t_argc, [](rpn_context & ctxt) {
  491. return _rpnNtpFunc(ctxt, minute);
  492. });
  493. }
  494. #endif
  495. // TODO: 1.14.0 weekday(...) conversion seemed to have 0..6 range with Monday as 0
  496. // using classic Sunday as first, but instead of 0 it is 1
  497. // Implementation above also uses 1 for Sunday, staying compatible with TimeLib
  498. #if NTP_SUPPORT && NTP_LEGACY_SUPPORT
  499. rpn_operator_set(_rpn_ctxt, "utc", 0, [](rpn_context & ctxt) -> rpn_error {
  500. if (!ntpSynced()) return rpn_operator_error::CannotContinue;
  501. rpn_value ts { static_cast<rpn_int>(ntpLocal2UTC(now())) };
  502. rpn_stack_push(ctxt, ts);
  503. return 0;
  504. });
  505. rpn_operator_set(_rpn_ctxt, "now", 0, _rpnNtpNow);
  506. rpn_operator_set(_rpn_ctxt, "month", 1, [](rpn_context & ctxt) {
  507. return _rpnNtpFunc(ctxt, month);
  508. });
  509. rpn_operator_set(_rpn_ctxt, "day", 1, [](rpn_context & ctxt) {
  510. return _rpnNtpFunc(ctxt, day);
  511. });
  512. rpn_operator_set(_rpn_ctxt, "dow", 1, [](rpn_context & ctxt) {
  513. return _rpnNtpFunc(ctxt, weekday);
  514. });
  515. rpn_operator_set(_rpn_ctxt, "hour", 1, [](rpn_context & ctxt) {
  516. return _rpnNtpFunc(ctxt, hour);
  517. });
  518. rpn_operator_set(_rpn_ctxt, "minute", 1, [](rpn_context & ctxt) {
  519. return _rpnNtpFunc(ctxt, minute);
  520. });
  521. #endif
  522. // Accept relay number and numeric API status value (0, 1 and 2)
  523. #if RELAY_SUPPORT
  524. // apply status and reset timers when called
  525. rpn_operator_set(_rpn_ctxt, "relay_reset", 2, [](rpn_context & ctxt) {
  526. return _rpnRelayStatus(ctxt, true);
  527. });
  528. // only update status when target status differs, keep running timers
  529. rpn_operator_set(_rpn_ctxt, "relay", 2, [](rpn_context & ctxt) {
  530. return _rpnRelayStatus(ctxt, false);
  531. });
  532. #endif // RELAY_SUPPORT == 1
  533. // Channel operators
  534. #if LIGHT_PROVIDER != LIGHT_PROVIDER_NONE
  535. rpn_operator_set(_rpn_ctxt, "update", 0, [](rpn_context & ctxt) -> rpn_error {
  536. lightUpdate();
  537. return 0;
  538. });
  539. rpn_operator_set(_rpn_ctxt, "black", 0, [](rpn_context & ctxt) -> rpn_error {
  540. lightColor(0ul);
  541. return 0;
  542. });
  543. rpn_operator_set(_rpn_ctxt, "channel", 2, [](rpn_context & ctxt) -> rpn_error {
  544. rpn_value value;
  545. rpn_value id;
  546. rpn_stack_pop(ctxt, id);
  547. rpn_stack_pop(ctxt, value);
  548. lightChannel(id.toUint(), id.toInt());
  549. return 0;
  550. });
  551. #endif
  552. #if RFB_SUPPORT
  553. rpn_operator_set(_rpn_ctxt, "rfb_send", 1, _rpnRfbSend);
  554. rpn_operator_set(_rpn_ctxt, "rfb_pop", 2, _rpnRfbPop);
  555. rpn_operator_set(_rpn_ctxt, "rfb_info", 2, _rpnRfbInfo);
  556. rpn_operator_set(_rpn_ctxt, "rfb_sequence", 4, _rpnRfbSequence);
  557. rpn_operator_set(_rpn_ctxt, "rfb_match", 3, _rpnRfbMatcher);
  558. rpn_operator_set(_rpn_ctxt, "rfb_match_wait", 4, _rpnRfbWaitMatch);
  559. #endif
  560. #if MQTT_SUPPORT
  561. rpn_operator_set(_rpn_ctxt, "mqtt_send", 2, [](rpn_context & ctxt) -> rpn_error {
  562. rpn_value message;
  563. rpn_stack_pop(ctxt, message);
  564. rpn_value topic;
  565. rpn_stack_pop(ctxt, topic);
  566. return mqttSendRaw(topic.toString().c_str(), message.toString().c_str())
  567. ? rpn_operator_error::Ok
  568. : rpn_operator_error::CannotContinue;
  569. });
  570. #endif
  571. // Some debugging. Dump stack contents
  572. #if TERMINAL_SUPPORT
  573. rpn_operator_set(_rpn_ctxt, "showstack", 0, [](rpn_context & ctxt) -> rpn_error {
  574. _rpnShowStack(terminalDefaultStream());
  575. return 0;
  576. });
  577. #endif
  578. // And, simple string logging
  579. #if DEBUG_SUPPORT
  580. rpn_operator_set(_rpn_ctxt, "dbgmsg", 1, [](rpn_context & ctxt) -> rpn_error {
  581. rpn_value message;
  582. rpn_stack_pop(ctxt, message);
  583. DEBUG_MSG_P(PSTR("[RPN] %s\n"), message.toString().c_str());
  584. return 0;
  585. });
  586. #endif
  587. rpn_operator_set(_rpn_ctxt, "mem?", 0, [](rpn_context & ctxt) -> rpn_error {
  588. rpn_stack_push(ctxt, rpn_value(rtcmemStatus()));
  589. return 0;
  590. });
  591. rpn_operator_set(_rpn_ctxt, "mem_write", 2, [](rpn_context & ctxt) -> rpn_error {
  592. auto addr = rpn_stack_pop(ctxt).toUint();
  593. auto value = rpn_stack_pop(ctxt).toUint();
  594. if (addr < RTCMEM_BLOCKS) {
  595. auto* rtcmem = reinterpret_cast<volatile uint32_t*>(RTCMEM_ADDR);
  596. *(rtcmem + addr) = value;
  597. return 0;
  598. }
  599. return rpn_operator_error::InvalidArgument;
  600. });
  601. rpn_operator_set(_rpn_ctxt, "mem_read", 1, [](rpn_context & ctxt) -> rpn_error {
  602. auto addr = rpn_stack_pop(ctxt).toUint();
  603. if (addr < RTCMEM_BLOCKS) {
  604. auto* rtcmem = reinterpret_cast<volatile uint32_t*>(RTCMEM_ADDR);
  605. rpn_uint result = *(rtcmem + addr);
  606. rpn_stack_push(ctxt, rpn_value(result));
  607. return 0;
  608. }
  609. return rpn_operator_error::InvalidArgument;
  610. });
  611. rpn_operator_set(_rpn_ctxt, "sleep", 2, [](rpn_context & ctxt) -> rpn_error {
  612. static bool once { false };
  613. if (once) return rpn_operator_error::CannotContinue;
  614. auto value = rpn_stack_pop(ctxt).checkedToUint();
  615. if (!value.ok()) {
  616. return value.error();
  617. }
  618. uint64_t duration = value.value();
  619. if (!duration) {
  620. return rpn_operator_error::CannotContinue;
  621. }
  622. auto mode = rpn_stack_pop(ctxt).toUint();
  623. once = true;
  624. schedule_function([duration, mode]() {
  625. wifiTurnOff();
  626. ESP.deepSleep(duration * 1000000ull, static_cast<RFMode>(mode));
  627. });
  628. return 0;
  629. });
  630. rpn_operator_set(_rpn_ctxt, "stations", 0, [](rpn_context & ctxt) -> rpn_error {
  631. if (!(WiFi.getMode() & WIFI_AP)) return rpn_operator_error::CannotContinue;
  632. rpn_stack_push(ctxt, rpn_value(static_cast<rpn_uint>(WiFi.softAPgetStationNum())));
  633. return 0;
  634. });
  635. rpn_operator_set(_rpn_ctxt, "disconnect", 0, [](rpn_context & ctxt) -> rpn_error {
  636. wifiDisconnect();
  637. yield();
  638. return 0;
  639. });
  640. rpn_operator_set(_rpn_ctxt, "rssi", 0, [](rpn_context & ctxt) -> rpn_error {
  641. if (!wifiConnected()) return rpn_operator_error::CannotContinue;
  642. rpn_stack_push(ctxt, rpn_value(static_cast<rpn_int>(WiFi.RSSI())));
  643. return 0;
  644. });
  645. rpn_operator_set(_rpn_ctxt, "delay", 1, [](rpn_context & ctxt) -> rpn_error {
  646. auto ms = rpn_stack_pop(ctxt);
  647. delay(ms.toUint());
  648. return 0;
  649. });
  650. rpn_operator_set(_rpn_ctxt, "yield", 0, [](rpn_context & ctxt) -> rpn_error {
  651. yield();
  652. return 0;
  653. });
  654. rpn_operator_set(_rpn_ctxt, "reset", 0, [](rpn_context & ctxt) -> rpn_error {
  655. static bool once = ([]() {
  656. deferredReset(100, CustomResetReason::Rule);
  657. return true;
  658. })();
  659. return once
  660. ? rpn_operator_error::CannotContinue
  661. : rpn_operator_error::Ok;
  662. });
  663. rpn_operator_set(_rpn_ctxt, "millis", 0, [](rpn_context & ctxt) -> rpn_error {
  664. rpn_stack_push(ctxt, rpn_value(static_cast<uint32_t>(millis())));
  665. return 0;
  666. });
  667. rpn_operator_set(_rpn_ctxt, "oneshot_ms", 1, [](rpn_context & ctxt) -> rpn_error {
  668. auto every = rpn_stack_pop(ctxt);
  669. return _rpnRunnerHandler(ctxt, RpnRunner::Policy::OneShot, every.toUint());
  670. });
  671. rpn_operator_set(_rpn_ctxt, "every_ms", 1, [](rpn_context & ctxt) -> rpn_error {
  672. auto every = rpn_stack_pop(ctxt);
  673. return _rpnRunnerHandler(ctxt, RpnRunner::Policy::Periodic, every.toUint());
  674. });
  675. // XXX: workaround for the vector 2x growth on push. will need to fix this in the rpnlib
  676. _rpn_ctxt.operators.shrink_to_fit();
  677. DEBUG_MSG_P(PSTR("[RPN] Registered %u operators\n"), _rpn_ctxt.operators.size());
  678. }
  679. #if TERMINAL_SUPPORT
  680. void _rpnInitCommands() {
  681. terminalRegisterCommand(F("RPN.RUNNERS"), [](const terminal::CommandContext& ctx) {
  682. if (!_rpn_runners.size()) {
  683. terminalError(ctx, F("No active runners"));
  684. return;
  685. }
  686. for (auto& runner : _rpn_runners) {
  687. char buffer[128] = {0};
  688. snprintf_P(buffer, sizeof(buffer), PSTR("%p %s %u ms, last %u ms"),
  689. &runner, (RpnRunner::Policy::Periodic == runner.policy) ? "every" : "one-shot",
  690. runner.period, runner.last
  691. );
  692. ctx.output.println(buffer);
  693. }
  694. terminalOK(ctx);
  695. });
  696. terminalRegisterCommand(F("RPN.VARS"), [](const terminal::CommandContext& ctx) {
  697. rpn_variables_foreach(_rpn_ctxt, [&ctx](const String& name, const rpn_value& value) {
  698. char buffer[256] = {0};
  699. snprintf_P(buffer, sizeof(buffer), PSTR(" %s: %s"), name.c_str(), _rpnValueToString(value).c_str());
  700. ctx.output.println(buffer);
  701. });
  702. terminalOK(ctx);
  703. });
  704. terminalRegisterCommand(F("RPN.OPS"), [](const terminal::CommandContext& ctx) {
  705. rpn_operators_foreach(_rpn_ctxt, [&ctx](const String& name, size_t argc, rpn_operator::callback_type) {
  706. char buffer[128] = {0};
  707. snprintf_P(buffer, sizeof(buffer), PSTR(" %s (%d)"), name.c_str(), argc);
  708. ctx.output.println(buffer);
  709. });
  710. terminalOK(ctx);
  711. });
  712. terminalRegisterCommand(F("RPN.TEST"), [](const terminal::CommandContext& ctx) {
  713. if (ctx.argc != 2) {
  714. terminalError(F("Wrong arguments"));
  715. return;
  716. }
  717. ctx.output.print(F("Running RPN expression: "));
  718. ctx.output.println(ctx.argv[1].c_str());
  719. if (!rpn_process(_rpn_ctxt, ctx.argv[1].c_str())) {
  720. rpn_stack_clear(_rpn_ctxt);
  721. char buffer[64] = {0};
  722. snprintf_P(buffer, sizeof(buffer), PSTR("position=%u category=%d code=%d"),
  723. _rpn_ctxt.error.position, static_cast<int>(_rpn_ctxt.error.category), _rpn_ctxt.error.code);
  724. terminalError(ctx, buffer);
  725. return;
  726. }
  727. _rpnShowStack(ctx.output);
  728. rpn_stack_clear(_rpn_ctxt);
  729. terminalOK(ctx);
  730. });
  731. }
  732. #endif
  733. // enables us to use rules without any events firing
  734. // notice: requires rpnRun to trigger at least once so that we can install runners
  735. void _rpnRunnersCheck() {
  736. auto ts = millis();
  737. for (auto& runner : _rpn_runners) {
  738. if (ts - runner.last >= runner.period) {
  739. runner.expired = true;
  740. runner.last = ts;
  741. _rpn_run = true;
  742. }
  743. }
  744. }
  745. void _rpnRunnersReset() {
  746. auto old = std::remove_if(_rpn_runners.begin(), _rpn_runners.end(), [](RpnRunner& runner) {
  747. return (RpnRunner::Policy::OneShot == runner.policy) && runner.expired;
  748. });
  749. if (old != _rpn_runners.end()) {
  750. _rpn_runners.erase(old, _rpn_runners.end());
  751. }
  752. for (auto& runner : _rpn_runners) {
  753. runner.expired = false;
  754. }
  755. }
  756. void _rpnRun() {
  757. if (!_rpn_run) {
  758. return;
  759. }
  760. if (millis() - _rpn_last <= _rpn_delay) {
  761. return;
  762. }
  763. _rpn_last = millis();
  764. _rpn_run = false;
  765. String rule;
  766. unsigned char i = 0;
  767. while ((rule = getSetting({"rpnRule", i++})).length()) {
  768. rpn_process(_rpn_ctxt, rule.c_str());
  769. rpn_stack_clear(_rpn_ctxt);
  770. }
  771. if (!getSetting("rpnSticky", 1 == RPN_STICKY)) {
  772. rpn_variables_clear(_rpn_ctxt);
  773. }
  774. }
  775. void _rpnLoop() {
  776. _rpnRunnersCheck();
  777. _rpnRun();
  778. _rpnRunnersReset();
  779. }
  780. void rpnSetup() {
  781. // Init context
  782. _rpnInit();
  783. // Load & cache settings
  784. _rpnConfigure();
  785. // Terminal commands
  786. #if TERMINAL_SUPPORT
  787. _rpnInitCommands();
  788. #endif
  789. // Websockets
  790. #if WEB_SUPPORT
  791. wsRegister()
  792. .onVisible([](JsonObject& root) { root["rpnVisible"] = 1; })
  793. .onConnected(_rpnWebSocketOnConnected)
  794. .onKeyCheck(_rpnWebSocketOnKeyCheck);
  795. #endif
  796. // MQTT
  797. #if MQTT_SUPPORT
  798. mqttRegister(_rpnMQTTCallback);
  799. #endif
  800. #if NTP_SUPPORT
  801. NtpBroker::Register([](NtpTick tick, time_t, const String&) {
  802. switch (tick) {
  803. case NtpTick::EveryMinute:
  804. _rpn_ntp_tick_minute = true;
  805. break;
  806. case NtpTick::EveryHour:
  807. _rpn_ntp_tick_hour = true;
  808. break;
  809. }
  810. _rpn_run = true;
  811. });
  812. #endif
  813. StatusBroker::Register(_rpnBrokerStatus);
  814. #if RFB_SUPPORT
  815. _rpnRfbSetup();
  816. #endif
  817. #if SENSOR_SUPPORT
  818. SensorReadBroker::Register(_rpnBrokerCallback);
  819. #endif
  820. espurnaRegisterReload(_rpnConfigure);
  821. espurnaRegisterLoop(_rpnLoop);
  822. _rpn_last = millis();
  823. _rpn_run = true;
  824. }
  825. #endif // RPN_RULES_SUPPORT