commit 76408a894a49bd9276cdda443fef42a86d79ee12
parent 8c5e43e1761bf958cf4d152523e6e4710bcf5431
Author: lash <dev@holbrook.no>
Date: Mon, 1 May 2023 22:33:51 +0100
Truncate readme
Diffstat:
M | README.md | | | 298 | +++---------------------------------------------------------------------------- |
1 file changed, 8 insertions(+), 290 deletions(-)
diff --git a/README.md b/README.md
@@ -1,11 +1,8 @@
# vise: A Constrained Size Output Virtual Machine
-An attempt at defining a small VM to handle menu interaction for size-constrained clients and servers.
+An attempt at defining a small VM to handle menu interaction for size-constrained clients.
-
-## Rationale
-
-Original motivation was to create a simple templating renderer for USSD clients, combined with an agnostic data-retrieval reference that may conceal any level of complexity.
+Original motivation for this project was to create a simple templating renderer for USSD clients, enhanced with a pluggable and agnostic interface for external data-retrieval.
## Features
@@ -41,294 +38,15 @@ Original motivation was to create a simple templating renderer for USSD clients,
* Key/value database reference example.
-## Opcodes
-
-The VM defines the following opcode symbols, alphabetically:
-
-* `CATCH <symbol> <signal> <matchmode>` - Jump to symbol if signal is set (see `signals` below). If match, has same side-effect as move.
-* `CROAK <signal> <matchmode>` - Clear state and restart execution from top if signal is set (see `signals` below). If match, has same side-effect as move.
-* `HALT` - Stop execution. The remaining bytecode (typically, the routing code for the node) is returned to the invoking function.
-* `INCMP <symbol> <arg>` - Compare registered input to `arg`. If match, it has the same side-effects as `MOVE`. In addition, any consecutive `INCMP` matches will be ignored until `HALT` is called.
-* `LOAD <symbol> <size>` - Execute the code symbol `symbol` and cache the data, constrained to the given `size`. Can be exposed with `MAP` within scope. See "External code" below.
-* `MAP <symbol>` - Expose a code symbol previously loaded by `LOAD` to the rendering client. Roughly corresponds to the `global` directive in Python.
-* `MNEXT <symbol> <selector>` - Define how to display the choice for advancing when browsing menu.
-* `MOUT <symbol> <selector>` - Add menu display entry. Each entry should have a matching `INCMP` whose `arg` matches `choice`. `display` is a descriptive text of the menu item.
-* `MOVE <symbol>` - Create a new execution frame, invalidating all previous `MAP` calls.
-* `MPREV <symbol> <selector>` - Define how to display the choice for returning when browsing menu.
-* `MSINK` - If set, the menu is defined as the browseable content sink. Cannot be used with an active `MAP` of a symbol with `LOAD` size `0`.
-* `RELOAD <symbol>` - Execute a code symbol already loaded by `LOAD` and cache the data, constrained to the previously given `size` for the same symbol. See "External code" below.
-
-
-### External code
-
-`LOAD` is used to execute code symbols in the host environment. It is loaded with a size constraint, and returned data violating this constraint should generate an error.
-
-Any symbol successfully loaded with `LOAD` will be associated with the call stack frame it is loaded. The symbol will be available in the same frame and frames below it. Once the frame goes out of scope (e.g. `BACK` is called in that frame) the symbols should be freed as soon as possible. At this point they are not available to the abandoned scope.
-
-Loaded symbols are not automatically exposed to the rendering client. To expose symbols ot the rendering client the `MAP` opcode must be used.
-
-The associated content of loaded symbols may be refreshed using the `RELOAD` opcode. `RELOAD` only works within the same constraints as `MAP`. However, updated content must be available even if a `MAP` precedes a `RELOAD` within the same frame.
-
-Methods handling `LOAD` symbols have the client input available to them.
-
-
-### External symbol optimizations
-
-Only `LOAD` and `RELOAD` should trigger external code side-effects.
-
-In an effort to prevent leaks from unnecessary external code executions, the following constraints are assumed:
-
-- An explicit `MAP` **must** exist in the scope of any `LOAD`.
-- All symbols declared in `MAP` **must** be used for all template renderings of a specific node.
-
-Any code compiler or checked **should** generate an error on any orphaned `LOAD` or `MAP` symbols as described above.
-
-
-### Signals
-
-Signal may be set when executing of external code symbols, and may be used as a simple exception mechanism.
-
-The signal flag arguments should only set a single flag to be tested. If more than one flag is set, the first flag matched will be used as the trigger.
-
-First 8 flags are reserved and used for internal VM operations.
-
-When a signal is caught, the *bytecode buffer is flushed* before the target symbol code is loaded.
-
-
-### Avoid duplicate menu items
-
-The vm execution should overwrite duplicate `MOUT` directives with the last definition between `HALT` instructions.
-
-The assembler should detect duplicate `INCMP` and `MOUT` (or menu batch code) selectors, and fail to compile. `MSEP` should be included in duplication detection.
-
-
-## Menus
-
-A menu has both a display and a input processing part. They are on either side of a `HALT` instruction.
-
-To assist with menu creation, a few batch operation symbols have been made available for use with the assembly language.
-
-* `DOWN <symbol> <choice> <display>` descend to next frame and move to `symbol`
-* `UP <choice> <display>` return to the previous frame
-* `NEXT <choice> <display>` include pagination advance
-* `PREVIOUS <choice> <display>` include pagination return. If `NEXT` has not been defined this will not be rendered.
-
-
-## Rendering
-
-The fixed-size output is generated using a templating language, and a combination of one or more _max size_ properties, and an optional _sink_ property that will attempt to consume all remaining capacity of the rendered template.
-
-In this example
-
-- `maxOutputSize` is 256 bytes long.
-- `template` is 120 bytes long.
-- param `one` has max size 10 but uses 5.
-- param `two` has max size 20 but uses 12.
-- param `three` is a _sink_.
-- rendered menu is 15 bytes long.
-
-The renderer may use up to `256 - 120 - 5 - 12 - 15 = 104` bytes from the _sink_ when rendering the output.
-
-
-### Menu browsing
-
-A max size can be set for the menu, which will count towards the space available for the _template sink_.
-
-Menus too long for a single screen should be browseable through separate screens. How the browse choice is displayed is defined using the `MSEP` definition. The browse choice counts towards the menu size capacity.
-
-When browsing additional menu pages, the template output should not be included.
-
-
-### Menu defaults
-
-Browsing menu display definitions (`MNEXT`, `MPREV`) as well as size constaints (`MSIZE`) should have sane defaults defined by the assembler if they are missing from the assembly code.
-
-
-### Multi-page support
-
-Multipage outputs, like listings, are handled using the _sink_ output constraints:
-
-- first calculate what the rendered display size is when all symbol results that are _not_ sinks are resolved.
-- split and cache the list data within its semantic context, given the _sink_ limitation after rendering.
-- provide a `next` and `previous` menu item to browse the prepared pagination of the list data.
-
-
-#### Multi-page menu support
-
-Long menus can also be paginated, as long as the rendered template does not have a different _sink_ already defined.
-
-Menu pagination is activated with the `MSINK` instruction. The menu sink is deactivated when the next `HALT` command is encountered.
-
-The menu will be rendered in the same manner as multi-page output, but is always appended to the rendered template.
-
-
-### Languages support
-
-Language for rendering can be set as:
-
-* Configuration variable of the engine
-* By setting the `FLAG_LANG` flag in the return value of a `LOAD` instruction handler.
-
-The language argument in either case **MUST** be a valid `ISO-639` language code string.
-
-Methods for retrieving data and templates are passed the language information through the `context.Context` value under the key `Language`. The value should be cast as a `lang.Language` object.
-
-
-## Virtual machine interface layout
-
-This is the version `0` of the VM.
-
-Currently the following rules apply for encoding in version `0`:
-
-- A code instruction is a _big-endian_ 2-byte value. See `vm/opcodes.go` for valid opcode values.
-- `symbol` value is encoded as _one byte_ of string length, after which the byte-value of the string follows.
-- `size` value is encoded as _one byte_ of numeric length, after which the _big-endian_ byte-value of the integer follows.
-- `signal` value is encoded as _one byte_ of byte length, after which a byte-array representing the defined signal follows.
-
-
-## Reference implementation
-
-This repository provides a `golang` reference implementation for the `vise` concept.
-
-
-### Structure
-
-- `asm`: Assembly parser and compiler.
-- `cache`: Holds and manages all loaded content.
-- `engine`: Outermost interface. Orchestrates execution of bytecode against input.
-- `lang`: Validation and specification of language context.
-- `logging`: Logging interface and build tags for loglevels.
-- `persist`: Interface and reference implementation of `state` and `cache` persistence across asynchronous vm executions.
-- `render`: Renders menu and templates, and enforces output size constraints.
-- `resource`: Retrieves data and bytecode from external symbols, and retrieves templates.
-- `state`: Holds the bytecode buffer, error states and navigation states.
-- `vm`: Defines instructions, and applies transformations according to the instructions.
-
-
-### Template rendering
-
-Template rendering is done using the `text/template` faciilty in the `golang` standard library.
-
-It expects all replacement symbols to be available at time of rendering, and has no tolerance for missing ones.
-
-
-## Runtime engine
-
-The runtime engine:
-
-* Validates client input
-* Runs VM with client input
-* Renders result
-* Restarts execution from top if the vm has nothing more to do.
-
-There are two flavors of the engine:
-
-* `engine.Loop` - class used for continuous, in-memory interaction with the vm (e.g. terminal).
-* `engine.RunPersisted` - method which combines single vm executions with persisted state (e.g. http).
-
-
-### Client identification
-
-The `engine.Config` struct defines a property `SessionId` which is added to the `context.Context` passed through entire engine vm call roundtrip.
-
-This is used to identify the caller, and thus defines a top-level storage key under which data entries should be retrieved.
-
-
-## Bytecode examples
-
-(Minimal, WIP)
-
-```
-000a 03666f6f 05746f666f6f # MOUT tofoo foo - display a menu entry for choice "foo", described by "to foo"
-0008 03666f6f 03626172 # INCMP bar foo - move to node "bar" if input is "FOO"
-0001 0461696565 01 01 # CATCH aiee 1 1 - move to node "aiee" (and immediately halt) if input match flag (1) is not set (1)
-0003 04616263 020104 # LOAD abc 260 - execute code symbol "abc" with a result size limit of 260 (2 byte BE integer, 0x0104)
-0003 04646566 00 # LOAD def 0 - execute code symbol "abc" with no size limit (sink)
-0005 04616263 # MAP abc - make "abc" available for renderer
-0007 # HALT - stop execution (require new input to continue)
-0006 0461313233 # MOVE a123 - move to node "a123" (regardless of input)
-0007 # HALT - stop execution
-```
-
-## Assembly examples
-
-See `testdata/*.vis`
-
-
-## Logging
-
-Loglevels are set at compile-time using the following build tags:
-
-* `lognone`
-* `logerror`
-* `logwarn`
-* `loginfo`
-* `logdebug`
-* `logtrace`
-
-Only use **ONE** of these tags.
-
-The default tag is `lognone` which disables logging completely.
-
-`logging/logging.go:Logger` defines the logging interface. It is faintly inspired by the experimental [slog](https://pkg.go.dev/golang.org/x/exp/slog) package, in that it differentiates explicit context logging.
-
-
-## Development tools
-
-Located in the `dev/` directory.
-
-
-### Test data generation
-
-`go run ./dev/gendata/ <directory>`
-
-Outputs bytecodes and templates for test data scenarios used in `engine` unit tests.
-
-
-### Interactive runner
-
-`go run ./dev/interactive [-d <data_directory>] [--root <root_symbol>] [--session-id <session_id>] [--persist]`
-
-Creates a new interactive session using `engine.DefaultEngine`, starting execution at symbol `root_symbol`
-
-`data_directory` points to a directory where templates and bytecode is to be found (in the same format as generated by `dev/testdata`).
-
-If `data_directory` is not set, current directory will be used.
-
-if `root_symbol` is not set, the symbol `root` will be used.
-
-if `session_id` is set, mutable data will be stored and retrieved keyed by the given identifer (if implemented).
-
-If `persist` is set, the execution state will be persisted across sessions.
-
-
-### Assembler
-
-`go run ./dev/asm <assembly_file>`
-
-Will output bytecode on STDOUT generated from a valid assembly file.
-
-
-### Disassembler
-
-`go run ./dev/disasm/ <binary_file>`
-
-Will list all the instructions on STDOUT from a valid binary file.
-
-
-## Interactive case examples
+## Documentation
-Found in `examples/`.
+Please refer to `doc/build/index.html`.
-Be sure to `make examples` before running them.
+Docs can be rebuilt using `make -B doc`.
-Can be run with `go run ./examples/<case> [...]` except helloworld which is run as `go run ./dev/interactive -d ./examples/helloworld [...]`
+Documentation sources are found in `doc/texinfo/`
-The available options are the same as for the `dev/interactive` tool.
-Contents of the case directory:
+## Examples
-* `*.vis` - assembly code.
-* `*.bin` - bytecode for each node symbol (only available after make).
-* `*.txt.orig` - default contents of a single data entry.
-* `*.txt` - current contents of a single data entry (only available after make).
+Build examples with `make -B examples`