Compare commits
No commits in common. "master" and "bytecode-spec" have entirely different histories.
master
...
bytecode-s
4
.gitignore
vendored
4
.gitignore
vendored
@ -27,7 +27,3 @@ debug.log
|
|||||||
/compile_jvm_output.txt
|
/compile_jvm_output.txt
|
||||||
/compile_metadata_output.txt
|
/compile_metadata_output.txt
|
||||||
test_output*.txt
|
test_output*.txt
|
||||||
/site/src/version-template/lyng-version.js
|
|
||||||
/bugcontents.db
|
|
||||||
/bugs/
|
|
||||||
contents.db
|
|
||||||
|
|||||||
21
AGENTS.md
21
AGENTS.md
@ -1,25 +1,9 @@
|
|||||||
# AI Agent Notes
|
# AI Agent Notes
|
||||||
|
|
||||||
## Canonical AI References
|
|
||||||
- Use `docs/ai_language_reference.md` as the primary, compiler-verified Lyng language reference for code generation.
|
|
||||||
- For generics-heavy code generation, follow `docs/ai_language_reference.md` section `7.1 Generics Runtime Model and Bounds` and `7.2 Differences vs Java / Kotlin / Scala`.
|
|
||||||
- Use `docs/ai_stdlib_reference.md` for default runtime/module APIs and stdlib surface.
|
|
||||||
- Treat `LYNG_AI_SPEC.md` and older docs as secondary if they conflict with the two files above.
|
|
||||||
- Prefer the shortest clear loop: use `for` for straightforward iteration/ranges; use `while` only when loop state/condition is irregular or changes in ways `for` cannot express cleanly.
|
|
||||||
- In Lyng code, slice strings with range indexing (`text[a..<b]`, `text[..<n]`, `text[n..]`) and avoid Java/Kotlin-style `substring(...)`.
|
|
||||||
|
|
||||||
## Lyng-First API Declarations
|
|
||||||
- Use `.lyng` declarations as the single source of truth for Lyng-facing API docs and types (especially module extern declarations).
|
|
||||||
- Prefer defining Lyng entities (enums/classes/type shapes) in `.lyng` files; only define them in Kotlin when there is Kotlin/platform-specific implementation detail that cannot be expressed in Lyng.
|
|
||||||
- Avoid hardcoding Lyng API documentation in Kotlin registrars when it can be declared in `.lyng`; Kotlin-side docs should be fallback/bridge only.
|
|
||||||
- For mixed pluggable modules (Lyng + Kotlin), embed module `.lyng` sources as generated Kotlin string literals, evaluate them into module scope during registration, then attach Kotlin implementations/bindings.
|
|
||||||
- When a change adds or changes Lyng-visible runtime/module behavior, update the corresponding `.lyng` declaration in the same change, including declaration-level docs/comments for new API surface.
|
|
||||||
|
|
||||||
## Kotlin/Wasm generation guardrails
|
## Kotlin/Wasm generation guardrails
|
||||||
- Avoid creating suspend lambdas for compiler runtime statements. Prefer explicit `object : Statement()` with `override suspend fun execute(...)`.
|
- Avoid creating suspend lambdas for compiler runtime statements. Prefer explicit `object : Statement()` with `override suspend fun execute(...)`.
|
||||||
- Do not use `statement { ... }` or other inline suspend lambdas in compiler hot paths (e.g., parsing/var declarations, initializer thunks).
|
- Do not use `statement { ... }` or other inline suspend lambdas in compiler hot paths (e.g., parsing/var declarations, initializer thunks).
|
||||||
- If you need a wrapper for delegated properties, check for `getValue` explicitly and return a concrete `Statement` object when missing; avoid `onNotFoundResult` lambdas.
|
- If you need a wrapper for delegated properties, check for `getValue` explicitly and return a concrete `Statement` object when missing; avoid `onNotFoundResult` lambdas.
|
||||||
- For any code in `commonMain`, verify it is Kotlin Multiplatform compatible before finishing. Do not use JVM-only APIs or Java-backed convenience methods such as `Map.putIfAbsent`; prefer stdlib/common equivalents and run at least the relevant compile/test task that exercises the `commonMain` source set.
|
|
||||||
- If wasmJs browser tests hang, first run `:lynglib:wasmJsNodeTest` and look for wasm compilation errors; hangs usually mean module instantiation failed.
|
- If wasmJs browser tests hang, first run `:lynglib:wasmJsNodeTest` and look for wasm compilation errors; hangs usually mean module instantiation failed.
|
||||||
- Do not increase test timeouts to mask wasm generation errors; fix the invalid IR instead.
|
- Do not increase test timeouts to mask wasm generation errors; fix the invalid IR instead.
|
||||||
|
|
||||||
@ -29,7 +13,6 @@
|
|||||||
- Object members are always allowed even on unknown types; non-Object members require explicit casts. Remove `inspect` from Object and use `toInspectString()` instead.
|
- Object members are always allowed even on unknown types; non-Object members require explicit casts. Remove `inspect` from Object and use `toInspectString()` instead.
|
||||||
- Type expression checks: `x is T` is value instance check; `T1 is T2` is type-subset; `A in T` means `A` is subset of `T`; `==` is structural type equality.
|
- Type expression checks: `x is T` is value instance check; `T1 is T2` is type-subset; `A in T` means `A` is subset of `T`; `==` is structural type equality.
|
||||||
- Type aliases: `type Name = TypeExpr` (generic allowed) expand to their underlying type expressions; no nominal distinctness.
|
- Type aliases: `type Name = TypeExpr` (generic allowed) expand to their underlying type expressions; no nominal distinctness.
|
||||||
- Bounds and variance: `T: A & B` / `T: A | B` for bounds; declaration-site variance with `out` / `in`.
|
|
||||||
- Do not reintroduce bytecode fallback opcodes (e.g., `GET_NAME`, `EVAL_*`, `CALL_FALLBACK`) or runtime name-resolution fallbacks; all symbol resolution must stay compile-time only.
|
- Do not reintroduce bytecode fallback opcodes (e.g., `GET_NAME`, `EVAL_*`, `CALL_FALLBACK`) or runtime name-resolution fallbacks; all symbol resolution must stay compile-time only.
|
||||||
|
|
||||||
## Bytecode frame-first migration plan
|
## Bytecode frame-first migration plan
|
||||||
@ -37,7 +20,3 @@
|
|||||||
- Create closure references only when a capture is detected; use a direct frame+slot reference (foreign slot ref) instead of scope slots.
|
- Create closure references only when a capture is detected; use a direct frame+slot reference (foreign slot ref) instead of scope slots.
|
||||||
- Keep Scope as a lazy reflection facade: resolve name -> slot only on demand for Kotlin interop (no eager name mapping on every call).
|
- Keep Scope as a lazy reflection facade: resolve name -> slot only on demand for Kotlin interop (no eager name mapping on every call).
|
||||||
- Avoid PUSH_SCOPE/POP_SCOPE in bytecode for loops/functions unless dynamic name access or Kotlin reflection is requested.
|
- Avoid PUSH_SCOPE/POP_SCOPE in bytecode for loops/functions unless dynamic name access or Kotlin reflection is requested.
|
||||||
|
|
||||||
## ABI proposal notes
|
|
||||||
- Runtime generic metadata for generic extern classes is tracked in `proposals/extern_generic_runtime_abi.md`.
|
|
||||||
- Keep this design `Obj`-centric: do not assume extern-class values are `ObjInstance`; collection must be enabled on `ObjClass`.
|
|
||||||
|
|||||||
270
CHANGELOG.md
270
CHANGELOG.md
@ -1,152 +1,156 @@
|
|||||||
# Changelog
|
## 1.5.0-SNAPSHOT
|
||||||
|
|
||||||
This file tracks user-visible Lyng language/runtime/tooling changes.
|
### Language Features
|
||||||
|
- Added `return` statement with local and non-local exit support (`return@label`).
|
||||||
|
- Support for `abstract` classes, methods, and variables.
|
||||||
|
- Introduced `interface` as a synonym for `abstract class`.
|
||||||
|
- Multiple Inheritance (MI) completed and enabled by default (C3 MRO).
|
||||||
|
- Class properties with custom accessors (`get`, `set`).
|
||||||
|
- Restricted setter visibility (`private set`, `protected set`).
|
||||||
|
- Late-initialized `val` fields in classes with `Unset` protection.
|
||||||
|
- Named arguments (`name: value`) and named splats (`...Map`).
|
||||||
|
- Assign-if-null operator `?=`.
|
||||||
|
- Refined `protected` visibility rules and `closed` modifier.
|
||||||
|
- Transient attribute `@Transient` for serialization and equality.
|
||||||
|
- Unified Delegation model for `val`, `var`, and `fun`.
|
||||||
|
- Singleton objects (`object`) and object expressions.
|
||||||
|
|
||||||
History note:
|
### Standard Library
|
||||||
- The project had periods where changelog maintenance lagged behind commits.
|
- Added `with(self, block)` for scoped execution.
|
||||||
- Entries below are synchronized and curated for `1.5.x`.
|
- Added `clamp()` function and extension.
|
||||||
- Earlier history may be incomplete and should be cross-checked with git tags/commits when needed.
|
- Improved `Exception` and `StackTraceEntry` reporting.
|
||||||
|
|
||||||
## Unreleased
|
### Tooling and IDE
|
||||||
|
- **CLI**: Added `fmt` as a first-class subcommand for code formatting.
|
||||||
|
- **IDEA Plugin**: Lightweight autocompletion (experimental), improved docs, and Grazie integration.
|
||||||
|
- **Highlighters**: Updated TextMate bundle and website highlighters for new syntax.
|
||||||
|
|
||||||
- No unreleased entries yet.
|
### Detailed Changes:
|
||||||
|
|
||||||
## 1.5.5 (2026-04-23)
|
- Language: Refined `protected` visibility rules
|
||||||
|
- Ancestor classes can now access `protected` members of their descendants, provided the ancestor also defines or inherits a member with the same name (indicating an override of a member known to the ancestor).
|
||||||
|
- This allows patterns where a base class calls a `protected` method that is implemented in a subclass.
|
||||||
|
- Fixed a regression where self-calls to unmangled members sometimes bypassed visibility checks incorrectly; these are now handled by the refined logic.
|
||||||
|
|
||||||
### Concurrency and collections
|
- Language: Added `return` statement
|
||||||
- Added coroutine coordination primitives and helpers for everyday parallel code:
|
- `return [expression]` exits the innermost enclosing callable (function or lambda).
|
||||||
- `Channel` for coroutine-to-coroutine communication
|
- Supports non-local returns using `@label` syntax (e.g., `return@outer 42`).
|
||||||
- `LaunchPool` for bounded-concurrency task execution
|
- Named functions automatically provide their name as a label for non-local returns.
|
||||||
- `Iterable<Deferred>.joinAll()` to await a whole collection of deferreds in input order
|
- Labeled lambdas: lambdas can be explicitly labeled using `@label { ... }`.
|
||||||
- `CompletableDeferred.completeExceptionally(...)` and `Deferred.cancelAndJoin()`
|
- Restriction: `return` is forbidden in shorthand function definitions (e.g., `fun f(x) = return x` is a syntax error).
|
||||||
- Added docs and examples for the new concurrency APIs, including `joinAll()` coverage in iterable and parallelism references.
|
- Control Flow: `return` and `break` are now protected from being caught by user-defined `try-catch` blocks in Lyng.
|
||||||
|
- Documentation: New `docs/return_statement.md` and updated `tutorial.md`.
|
||||||
|
|
||||||
### Database and time APIs
|
- Language: stdlib improvements
|
||||||
- Added the portable `lyng.io.db` SQL contract and the first concrete providers:
|
- Added `with(self, block)` function to `root.lyng` which executes a block with `this` set to the provided object.
|
||||||
- `lyng.io.db.sqlite` on JVM and Linux Native
|
- Language: Abstract Classes and Interfaces
|
||||||
- `lyng.io.db.jdbc` on JVM
|
- Support for `abstract` modifier on classes, methods, and variables.
|
||||||
- Added SQLite/JDBC release hardening:
|
- Introduced `interface` as a synonym for `abstract class`, supporting full state (constructors, fields, `init` blocks) and implementation by parts via MI.
|
||||||
- nested transactions via savepoints
|
- New `closed` modifier (antonym to `open`) to prevent overriding class members.
|
||||||
- detached materialized rows
|
- Refined `override` logic: mandatory keyword when re-declaring members that exist in the ancestor chain (MRO).
|
||||||
- generated-key support through `ExecutionResult.getGeneratedKeys()`
|
- MI Satisfaction: Abstract requirements are automatically satisfied by matching concrete members found later in the C3 MRO chain without requiring explicit proxy methods.
|
||||||
- schema-driven value conversion for `Bool`, `Decimal`, `Date`, `DateTime`, and `Instant`
|
- Integration: Updated highlighters (lynglib, lyngweb, IDEA plugin), IDEA completion, and Grazie grammar checking.
|
||||||
- portable SQLite linker/deployment fixes and documented runtime options
|
- Documentation: Updated `docs/OOP.md` with sections on "Abstract Classes and Members", "Interfaces", and "Overriding and Virtual Dispatch".
|
||||||
- Added `Date` to `lyng.time` and the core runtime as a first-class calendar-date type, plus conversions and arithmetic across `Instant`, `DateTime`, and `Date`.
|
- IDEA plugin: Improved natural language support and spellchecking
|
||||||
|
- Disabled the limited built-in English and Technical dictionaries.
|
||||||
|
- Enforced usage of the platform's standard Natural Languages (Grazie) and Spellchecker components.
|
||||||
|
- Integrated `SpellCheckerManager` for word suggestions and validation, respecting users' personal and project dictionaries.
|
||||||
|
- Added project-specific "learned words" support via `Lyng Formatter` settings and quick-fixes.
|
||||||
|
- Enhanced fallback spellchecker for technical terms and Lyng-specific vocabulary.
|
||||||
|
|
||||||
### Language, stdlib, and tooling
|
- Language: Class properties with accessors
|
||||||
- Added extensions on singleton `object` declarations, including object-scoped indexer overrides for bracket syntax.
|
- Support for `val` (read-only) and `var` (read-write) properties in classes.
|
||||||
- Added backtick string literals and formatter support.
|
- Syntax: `val name [ : Type ] get() { body }` or `var name [ : Type ] get() { body } set(value) { body }`.
|
||||||
- Added `lyng.legacy_digest` for SHA-1 compatibility work, `String.replace`, and `buffer.base64std`.
|
- Laconic Expression Shorthand: `val prop get() = expression` and `var prop get() = read set(v) = write`.
|
||||||
- Improved CLI/runtime behavior with `atExit` shutdown handlers, native release-binary work, and follow-up CLI packaging/import fixes.
|
- Properties are pure accessors and do **not** have automatic backing fields.
|
||||||
- Expanded docs across the tutorial, stdlib references, database docs, networking docs, and release notes.
|
- Validation: `var` properties must have both accessors; `val` must have only a getter.
|
||||||
|
- Integration: Updated TextMate grammar and IntelliJ plugin (highlighting + keywords).
|
||||||
|
- Documentation: New "Properties" section in `docs/OOP.md`.
|
||||||
|
|
||||||
### Runtime/compiler stability and performance
|
- Language: Restricted Setter Visibility
|
||||||
- Extended exact-call and higher-order lambda inlining through the bytecode compiler, including compiled fast paths for simple lambdas, wrappers, captures, and common higher-order helpers.
|
- Support for `private set` and `protected set` modifiers on `var` fields and properties.
|
||||||
- Fixed import caching and class/object bytecode dispatch on JVM.
|
- Allows members to be publicly readable but only writable from within the declaring class or its subclasses.
|
||||||
- Fixed immutable `val` compound assignments so true mutating `*Assign` operations continue to work while fallback reassignments report the correct read-only error.
|
- Enforcement at runtime: throws `AccessException` on unauthorized writes.
|
||||||
- Fixed closure/capture and import regressions across launched loops, singleton/object extensions, aliasing, transitive re-exports, and immutable capture escaping.
|
- Supported only for declarations in class bodies (fields and properties).
|
||||||
- Improved list-fill/list-append fast paths, nullable-let inference, Decimal/Complex interop, and related regression coverage.
|
- Documentation: New "Restricted Setter Visibility" section in `docs/OOP.md`.
|
||||||
|
|
||||||
### Release notes
|
- Language: Late-initialized `val` fields in classes
|
||||||
- Release metadata, homepage samples, docs, and README now point to `1.5.5`.
|
- Support for declaring `val` without an immediate initializer in class bodies.
|
||||||
|
- Compulsory initialization: every late-init `val` must be assigned at least once within the class body or an `init` block.
|
||||||
|
- Write-once enforcement: assigning to a `val` is allowed only if its current value is `Unset`.
|
||||||
|
- Access protection: reading a late-init `val` before it is assigned returns the `Unset` singleton; using `Unset` for most operations throws an `UnsetException`.
|
||||||
|
- Extension properties do not support late-init.
|
||||||
|
- Documentation: New "Late-initialized `val` fields" and "The `Unset` singleton" sections in `docs/OOP.md`.
|
||||||
|
|
||||||
## 1.5.4 (2026-04-03)
|
- Docs: OOP improvements
|
||||||
|
- New page: `docs/scopes_and_closures.md` detailing `ClosureScope` resolution order, recursion‑safe helpers (`chainLookupIgnoreClosure`, `chainLookupWithMembers`, `baseGetIgnoreClosure`), cycle prevention, and capturing lexical environments for callbacks (`snapshotForClosure`).
|
||||||
|
- Updated: `docs/advanced_topics.md` (link to the new page), `docs/parallelism.md` (closures in `launch`/`flow`), `docs/OOP.md` (visibility from closures with preserved `currentClassCtx`), `docs/exceptions_handling.md` (compatibility alias `SymbolNotFound`).
|
||||||
|
- Tutorial: added quick link to Scopes and Closures.
|
||||||
|
|
||||||
### Runtime and compiler stability
|
- IDEA plugin: Lightweight autocompletion (experimental)
|
||||||
- Stabilized the recent `piSpigot` benchmark/compiler work for release.
|
- Global completion: local declarations, in‑scope parameters, imported modules, and stdlib symbols.
|
||||||
- Fixed numeric-mix regressions introduced by overly broad int-coercion in bytecode compilation.
|
- Member completion: after a dot, suggests only members of the inferred receiver type (incl. chained calls like `Path(".." ).lines().` → `Iterator` methods). No global identifiers appear after a dot.
|
||||||
- Restored correct behavior for decimal arithmetic, mixed real/int flows, list literals, list size checks, and national-character script cases.
|
- Inheritance-aware: direct class members first, then inherited (e.g., `List` includes `Collection`/`Iterable` methods).
|
||||||
- Fixed plain-list index fast paths so they no longer bypass subclass behavior such as `ObservableList` hooks and flow notifications.
|
- Heuristics: handles literals (`"…"` → `String`, numbers → `Int/Real`, `[...]` → `List`, `{...}` → `Dict`) and static `Namespace.` members.
|
||||||
- Hardened local numeric compare fast paths to correctly handle primitive-coded frame slots.
|
- Performance: capped results, early prefix filtering, per‑document MiniAst cache, cancellation checks.
|
||||||
|
- Toggle: Settings | Lyng Formatter → "Enable Lyng autocompletion (experimental)" (default ON).
|
||||||
|
- Stabilization: DEBUG completion/Quick Doc logs are OFF by default; behavior aligned between IDE and isolated engine tests.
|
||||||
|
|
||||||
### Performance and examples
|
- Language: Named arguments and named splats
|
||||||
- Added `piSpigot` benchmark/example coverage:
|
- New call-site syntax for named arguments using colon: `name: value`.
|
||||||
- `examples/pi-test.lyng`
|
- Positional arguments must come before named; positionals after a named argument inside parentheses are rejected.
|
||||||
- `examples/pi-bench.lyng`
|
- Trailing-lambda interaction: if the last parameter is already assigned by name (or via a named splat), a trailing `{ ... }` block is illegal.
|
||||||
- JVM benchmark test for release-baseline verification
|
- Named splats: `...` can now expand a Map into named arguments.
|
||||||
- Kept the safe list/index/runtime wins that improve the optimized `piSpigot` path without reintroducing type-unsound coercions.
|
- Only string keys are allowed; non-string keys raise a clear error.
|
||||||
- Changed the default `RVAL_FASTPATH` setting off on JVM/Android and in the benchmark preset after verification that it no longer helps the stabilized `piSpigot` workload.
|
- Duplicate assignment across named args and named splats is an error.
|
||||||
|
- Ellipsis (variadic) parameters remain positional-only and cannot be named.
|
||||||
|
- Rationale: `=` is assignment and an expression in Lyng; `:` at call sites avoids ambiguity. Declarations keep `name: Type`; call-site casts continue to use `as` / `as?`.
|
||||||
|
- Documentation updated: proposals and declaring-arguments sections now cover named args/splats and error cases.
|
||||||
|
- Tests added covering success cases and errors for named args/splats and trailing-lambda interactions.
|
||||||
|
|
||||||
### Release notes
|
- Tooling: Highlighters and TextMate bundle updated for named args
|
||||||
- Full JVM and wasm test gates pass on the release tree.
|
- Website/editor highlighter (lyngweb + site) works with `name: value` and `...Map("k" => v)`; added JS tests covering punctuation/operator spans for `:` and `...`.
|
||||||
- Benchmark findings and remaining post-release optimization targets are documented in `notes/pi_spigot_benchmark_baseline_2026-04-03.md`.
|
- TextMate grammar updated to recognize named call arguments: `name: value` after `(` or `,` with `name` highlighted as `variable.parameter.named.lyng` and `:` as punctuation; excludes `::`.
|
||||||
|
- TextMate bundle version bumped to 0.0.3; README updated with details and guidance.
|
||||||
|
|
||||||
## 1.5.1 (2026-03-25)
|
- Multiple Inheritance (MI) completed and enabled by default:
|
||||||
|
- Active C3 Method Resolution Order (MRO) for deterministic, monotonic lookup across complex hierarchies and diamonds.
|
||||||
|
- Qualified dispatch:
|
||||||
|
- `this@Type.member(...)` inside class bodies starts lookup at the specified ancestor.
|
||||||
|
- Cast-based disambiguation: `(expr as Type).member(...)`, `(expr as? Type)?.member(...)` (works with existing safe-call `?.`).
|
||||||
|
- Field inheritance (`val`/`var`) under MI:
|
||||||
|
- Instance storage is disambiguated per declaring class; unqualified read/write resolves to the first match in MRO.
|
||||||
|
- Qualified read/write targets the chosen ancestor’s storage.
|
||||||
|
- Constructors and initialization:
|
||||||
|
- Direct bases are initialized left-to-right; each ancestor is initialized at most once (diamond-safe de-duplication).
|
||||||
|
- Header-specified constructor arguments are passed to direct bases.
|
||||||
|
- Visibility enforcement under MI:
|
||||||
|
- `private` visible only inside the declaring class body.
|
||||||
|
- `protected` visible inside the declaring class and its transitive subclasses. Additionally, ancestor classes can access protected members of their descendants if it's an override of a member known to the ancestor. Unrelated contexts cannot access it (qualification/casts do not bypass).
|
||||||
|
- Diagnostics improvements:
|
||||||
|
- Missing member/field messages include receiver class and linearization order; hints for `this@Type` or casts when helpful.
|
||||||
|
- Invalid `this@Type` reports that the qualifier is not an ancestor and shows the receiver lineage.
|
||||||
|
- `as`/`as?` cast errors include actual and target type names.
|
||||||
|
|
||||||
### Language
|
- Documentation updated (docs/OOP.md and tutorial quick-start) to reflect MI with active C3 MRO.
|
||||||
- Added string interpolation:
|
|
||||||
- `"$name"` identifier interpolation.
|
|
||||||
- `"${expr}"` expression interpolation.
|
|
||||||
- Added literal-dollar forms in strings:
|
|
||||||
- `"\$"` -> `$`
|
|
||||||
- `"$$"` -> `$`
|
|
||||||
- `\\$x` is parsed as backslash + interpolation of `x`.
|
|
||||||
- Added per-file interpolation opt-out via leading directive comment:
|
|
||||||
- `// feature: interpolation: off`
|
|
||||||
|
|
||||||
### Docs and AI references
|
- CLI: Added `fmt` as a first-class Clikt subcommand.
|
||||||
- Updated compiler-accurate AI language docs:
|
- Default behavior: formats files to stdout (no in-place edits by default).
|
||||||
- interpolation syntax and escaping
|
- Options:
|
||||||
- per-file feature switch behavior
|
- `--check`: check only; print files that would change; exit with code 2 if any changes are needed.
|
||||||
- Refreshed tutorial examples and doctests to reflect new interpolation semantics.
|
- `-i, --in-place`: write formatted result back to files.
|
||||||
- Added/reworked current proposal/reference materials for Lyng common-platform guidance.
|
- `--spacing`: apply spacing normalization.
|
||||||
|
- `--wrap`, `--wrapping`: enable line wrapping.
|
||||||
|
- Mutually exclusive: `--check` and `--in-place` together now produce an error and exit with code 1.
|
||||||
|
- Multi-file stdout prints headers `--- <path> ---` per file.
|
||||||
|
- `lyng --help` shows `fmt`; `lyng fmt --help` displays dedicated help.
|
||||||
|
- Fix: Property accessors (`get`, `set`, `private set`, `protected set`) are now correctly indented relative to the property declaration.
|
||||||
|
- Fix: Indentation now correctly carries over into blocks that start on extra‑indented lines (e.g., nested `if` statements or property accessor bodies).
|
||||||
|
- Fix: Formatting Markdown files no longer deletes content in `.lyng` code fences and works correctly with injected files (resolves clobbering, `StringIndexOutOfBoundsException`, and `nonempty text is not covered by block` errors).
|
||||||
|
|
||||||
### Compatibility notes
|
- CLI: Preserved legacy script invocation fast-paths:
|
||||||
- Interpolation is enabled by default for normal string literals.
|
- `lyng script.lyng [args...]` executes the script directly.
|
||||||
- Existing code that intentionally used `$name` as literal text should use `\$name`, `$$name`, or the file directive `// feature: interpolation: off`.
|
- `lyng -- -file.lyng [args...]` executes a script whose name begins with `-`.
|
||||||
|
|
||||||
## 1.5.0 (2026-03-22)
|
- CLI: Fixed a regression where the root help banner could print before subcommands.
|
||||||
|
- Root command no longer prints help when a subcommand (e.g., `fmt`) is invoked.
|
||||||
### Major runtime/compiler direction
|
|
||||||
- Completed migration to bytecode-first/bytecode-only execution paths.
|
|
||||||
- Removed interpreter fallback behavior in core execution hot paths.
|
|
||||||
- Continued frame-slot-first local/capture model improvements and related diagnostics.
|
|
||||||
|
|
||||||
### Language features and semantics
|
|
||||||
- Added/finished `return` semantics including labeled non-local forms (`return@label`).
|
|
||||||
- Added abstract classes/members and `interface` support (as abstract-class-style construct).
|
|
||||||
- Completed and enabled multiple inheritance with C3 MRO by default.
|
|
||||||
- Added class properties with accessors (`get`/`set`) and restricted setter visibility (`private set`, `protected set`).
|
|
||||||
- Added late-initialized class `val` support with `Unset` protection rules.
|
|
||||||
- Added named arguments (`name: value`) and named splats (`...map`) with stricter validation.
|
|
||||||
- Added assign-if-null operator `?=`.
|
|
||||||
- Improved nullable/type-checking behavior (including `T is nullable` and related type checks).
|
|
||||||
- Added variadic function types (`...` in function type declarations) and tighter lambda type checks.
|
|
||||||
|
|
||||||
### Type system and collections
|
|
||||||
- Added immutable collections hierarchy (`ImmutableList`, `ImmutableSet`, `ImmutableMap`).
|
|
||||||
- Improved generic runtime binding/checking for explicit type arguments and bounds.
|
|
||||||
- Added smarter type-aware collection ops (`+=`, `-=`) and stronger declared-member type checks.
|
|
||||||
|
|
||||||
### Extern/Kotlin bridge
|
|
||||||
- Tightened extern declaration rules:
|
|
||||||
- explicit extern members are required for extern class/object declarations.
|
|
||||||
- Improved extern generic class behavior and diagnostics.
|
|
||||||
- Extended bridge APIs for binding global functions/variables and object/member interop scenarios.
|
|
||||||
|
|
||||||
### Standard library and modules
|
|
||||||
- Added `lyng.observable` improvements (`ObservableList` hooks/events).
|
|
||||||
- Added `Random` stdlib API used by updated samples.
|
|
||||||
- Added/extended `lyngio.console` support and CLI integration for console interaction.
|
|
||||||
- Migrated time APIs to `kotlin.time` (`Instant` migration and related docs/tests).
|
|
||||||
|
|
||||||
### CLI, IDE, and docs/tooling
|
|
||||||
- CLI:
|
|
||||||
- added first-class `fmt` command
|
|
||||||
- preserved direct script fast-path invocation
|
|
||||||
- improved command help/dispatch behavior
|
|
||||||
- IntelliJ plugin:
|
|
||||||
- improved lightweight completion and documentation/inspection behavior
|
|
||||||
- continued highlighter and Grazie/spellchecking integration work
|
|
||||||
- Docs:
|
|
||||||
- substantial updates across tutorial/OOP/type/runtime references
|
|
||||||
- expanded bytecode and advanced topics coverage
|
|
||||||
|
|
||||||
## Migration checklist for 1.5.x
|
|
||||||
|
|
||||||
- If you rely on literal `$...` strings:
|
|
||||||
- replace with `\$...` or `$$...`, or
|
|
||||||
- add `// feature: interpolation: off` at file top.
|
|
||||||
- Review any code relying on interpreter-era fallback behavior; 1.5.x assumes bytecode-first execution.
|
|
||||||
- For extern declarations, ensure members are explicitly declared where required.
|
|
||||||
- For named arguments/splats, verify call sites follow stricter ordering/duplication rules.
|
|
||||||
|
|||||||
@ -4,7 +4,6 @@ High-density specification for LLMs. Reference this for all Lyng code generation
|
|||||||
|
|
||||||
## 1. Core Philosophy & Syntax
|
## 1. Core Philosophy & Syntax
|
||||||
- **Everything is an Expression**: Blocks, `if`, `when`, `for`, `while`, `do-while` return their last expression (or `void`).
|
- **Everything is an Expression**: Blocks, `if`, `when`, `for`, `while`, `do-while` return their last expression (or `void`).
|
||||||
- **Static Types + Inference**: Every declaration has a compile-time type (explicit or inferred). Types are Kotlin‑style: non‑null by default, nullable with `?`.
|
|
||||||
- **Loops with `else`**: `for`, `while`, and `do-while` support an optional `else` block.
|
- **Loops with `else`**: `for`, `while`, and `do-while` support an optional `else` block.
|
||||||
- `else` executes **only if** the loop finishes normally (without a `break`).
|
- `else` executes **only if** the loop finishes normally (without a `break`).
|
||||||
- `break <value>` exits the loop and sets its return value.
|
- `break <value>` exits the loop and sets its return value.
|
||||||
@ -14,7 +13,6 @@ High-density specification for LLMs. Reference this for all Lyng code generation
|
|||||||
3. Result of the last iteration (if loop finished normally and no `else`).
|
3. Result of the last iteration (if loop finished normally and no `else`).
|
||||||
4. `void` (if loop body never executed and no `else`).
|
4. `void` (if loop body never executed and no `else`).
|
||||||
- **Implicit Coroutines**: All functions are coroutines. No `async/await`. Use `launch { ... }` (returns `Deferred`) or `flow { ... }`.
|
- **Implicit Coroutines**: All functions are coroutines. No `async/await`. Use `launch { ... }` (returns `Deferred`) or `flow { ... }`.
|
||||||
- **Functions**: Use `fun` or the short form `fn`. Function declarations are expressions returning a callable.
|
|
||||||
- **Variables**: `val` (read-only), `var` (mutable). Supports late-init `val` in classes (must be assigned in `init` or body).
|
- **Variables**: `val` (read-only), `var` (mutable). Supports late-init `val` in classes (must be assigned in `init` or body).
|
||||||
- **Serialization**: Use `@Transient` attribute before `val`/`var` or constructor parameters to exclude them from Lynon/JSON serialization. Transient fields are also ignored during `==` structural equality checks.
|
- **Serialization**: Use `@Transient` attribute before `val`/`var` or constructor parameters to exclude them from Lynon/JSON serialization. Transient fields are also ignored during `==` structural equality checks.
|
||||||
- **Null Safety**: `?` (nullable type), `?.` (safe access), `?( )` (safe invoke), `?{ }` (safe block invoke), `?[ ]` (safe index), `?:` or `??` (elvis), `?=` (assign-if-null).
|
- **Null Safety**: `?` (nullable type), `?.` (safe access), `?( )` (safe invoke), `?{ }` (safe block invoke), `?[ ]` (safe index), `?:` or `??` (elvis), `?=` (assign-if-null).
|
||||||
@ -46,21 +44,9 @@ High-density specification for LLMs. Reference this for all Lyng code generation
|
|||||||
- **Root Type**: Everything is an `Object` (root of the hierarchy).
|
- **Root Type**: Everything is an `Object` (root of the hierarchy).
|
||||||
- **Nullability**: Non-null by default (`T`), nullable with `T?`, `!!` asserts non-null.
|
- **Nullability**: Non-null by default (`T`), nullable with `T?`, `!!` asserts non-null.
|
||||||
- **Untyped params**: `fun foo(x)` -> `x: Object`, `fun foo(x?)` -> `x: Object?`.
|
- **Untyped params**: `fun foo(x)` -> `x: Object`, `fun foo(x?)` -> `x: Object?`.
|
||||||
- **Untyped vars**: `var x` is `Unset` until first assignment locks the type (including nullability).
|
- **Untyped vars**: `var x` is `Unset` until first assignment locks the type.
|
||||||
- `val x = null` -> type `Null`; `var x = null` -> type `Object?`.
|
- **Inference**: List/map literals infer union element types; empty list is `List<Object>`, empty map is `{:}`.
|
||||||
- **Inference**:
|
- **Generics**: Bounds with `T: A & B` or `T: A | B`; variance uses `out`/`in`.
|
||||||
- List literals infer union element types; empty list defaults to `List<Object>` unless constrained.
|
|
||||||
- Map literals infer key/value types; empty map defaults to `Map<Object, Object>` unless constrained.
|
|
||||||
- Mixed numeric ops promote `Int` + `Real` to `Real`.
|
|
||||||
- **Type aliases**: `type Name = TypeExpr` (generic allowed). Aliases expand to their underlying type expressions (no nominal distinctness).
|
|
||||||
- **Generics**: Bounds with `T: A & B` or `T: A | B`; variance uses `out`/`in` (declaration‑site only).
|
|
||||||
- **Casts**: `as` is a runtime-checked cast; `as?` is safe-cast returning `null`. If the value is nullable, `as T` implies `!!`.
|
|
||||||
|
|
||||||
## 2.2 Type Expressions and Checks
|
|
||||||
- **Value checks**: `x is T` (runtime instance check).
|
|
||||||
- **Type checks**: `T1 is T2` and `A in T` are subset checks between type expressions (compile-time where possible).
|
|
||||||
- **Type equality**: `T1 == T2` is structural (unions/intersections are order‑insensitive).
|
|
||||||
- **Compile-time enforcement**: Bounds are checked at call sites; runtime checks only appear when the compile‑time type is too general.
|
|
||||||
|
|
||||||
## 3. Delegation (`by`)
|
## 3. Delegation (`by`)
|
||||||
Unified model for `val`, `var`, and `fun`.
|
Unified model for `val`, `var`, and `fun`.
|
||||||
|
|||||||
86
README.md
86
README.md
@ -32,9 +32,9 @@ class A {
|
|||||||
enum E* { One, Two }
|
enum E* { One, Two }
|
||||||
}
|
}
|
||||||
val ab = A.B()
|
val ab = A.B()
|
||||||
assertEquals(null, ab.x)
|
assertEquals(ab.x, null)
|
||||||
assertEquals("bar", A.Inner.foo)
|
assertEquals(A.Inner.foo, "bar")
|
||||||
assertEquals(A.E.One, A.One)
|
assertEquals(A.One, A.E.One)
|
||||||
```
|
```
|
||||||
|
|
||||||
- extremely simple Kotlin integration on any platform (JVM, JS, WasmJS, Lunux, MacOS, iOS, Windows)
|
- extremely simple Kotlin integration on any platform (JVM, JS, WasmJS, Lunux, MacOS, iOS, Windows)
|
||||||
@ -48,12 +48,9 @@ assertEquals(A.E.One, A.One)
|
|||||||
|
|
||||||
- [Language home](https://lynglang.com)
|
- [Language home](https://lynglang.com)
|
||||||
- [introduction and tutorial](docs/tutorial.md) - start here please
|
- [introduction and tutorial](docs/tutorial.md) - start here please
|
||||||
- [Latest release notes (1.5.5)](docs/whats_new.md)
|
|
||||||
- [What's New in 1.5](docs/whats_new_1_5.md)
|
- [What's New in 1.5](docs/whats_new_1_5.md)
|
||||||
- [Testing and Assertions](docs/Testing.md)
|
- [Testing and Assertions](docs/Testing.md)
|
||||||
- [Filesystem and Processes (lyngio)](docs/lyngio.md)
|
- [Filesystem and Processes (lyngio)](docs/lyngio.md)
|
||||||
- [SQL Databases (lyng.io.db)](docs/lyng.io.db.md)
|
|
||||||
- [Time and Calendar Types](docs/time.md)
|
|
||||||
- [Return Statement](docs/return_statement.md)
|
- [Return Statement](docs/return_statement.md)
|
||||||
- [Efficient Iterables in Kotlin Interop](docs/EfficientIterables.md)
|
- [Efficient Iterables in Kotlin Interop](docs/EfficientIterables.md)
|
||||||
- [Samples directory](docs/samples)
|
- [Samples directory](docs/samples)
|
||||||
@ -66,7 +63,8 @@ assertEquals(A.E.One, A.One)
|
|||||||
### Add dependency to your project
|
### Add dependency to your project
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val lyngVersion = "1.5.5"
|
// update to current please:
|
||||||
|
val lyngVersion = "1.5.0-SNAPSHOT"
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
// ...
|
// ...
|
||||||
@ -95,49 +93,42 @@ import net.sergeych.lyng.*
|
|||||||
// we need a coroutine to start, as Lyng
|
// we need a coroutine to start, as Lyng
|
||||||
// is a coroutine based language, async topdown
|
// is a coroutine based language, async topdown
|
||||||
runBlocking {
|
runBlocking {
|
||||||
val session = EvalSession()
|
assert(5 == eval(""" 3*3 - 4 """).toInt())
|
||||||
assert(5 == session.eval(""" 3*3 - 4 """).toInt())
|
eval(""" println("Hello, Lyng!") """)
|
||||||
session.eval(""" println("Hello, Lyng!") """)
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
### Exchanging information
|
### Exchanging information
|
||||||
|
|
||||||
The preferred host runtime is `EvalSession`. It owns the script scope and any coroutines
|
Script is executed over some `Scope`. Create instance,
|
||||||
started with `launch { ... }`. Create a session, grab its scope when you need low-level
|
add your specific vars and functions to it, and call:
|
||||||
binding APIs, then execute scripts through the session:
|
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
import net.sergeych.lyng.*
|
import net.sergeych.lyng.*
|
||||||
|
|
||||||
runBlocking {
|
// simple function
|
||||||
val session = EvalSession()
|
val scope = Script.newScope().apply {
|
||||||
val scope = session.getScope().apply {
|
addFn("sumOf") {
|
||||||
// simple function
|
var sum = 0.0
|
||||||
addFn("sumOf") {
|
for (a in args) sum += a.toDouble()
|
||||||
var sum = 0.0
|
ObjReal(sum)
|
||||||
for (a in args) sum += a.toDouble()
|
|
||||||
ObjReal(sum)
|
|
||||||
}
|
|
||||||
addConst("LIGHT_SPEED", ObjReal(299_792_458.0))
|
|
||||||
|
|
||||||
// callback back to kotlin to some suspend fn, for example::
|
|
||||||
// suspend fun doSomeWork(text: String): Int
|
|
||||||
addFn("doSomeWork") {
|
|
||||||
// this _is_ a suspend lambda, we can call suspend function,
|
|
||||||
// and it won't consume the thread.
|
|
||||||
// note that in kotlin handler, `args` is a list of `Obj` arguments
|
|
||||||
// and return value from this lambda should be Obj too:
|
|
||||||
doSomeWork(args[0]).toObj()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
addConst("LIGHT_SPEED", ObjReal(299_792_458.0))
|
||||||
|
|
||||||
// execute through the session:
|
// callback back to kotlin to some suspend fn, for example::
|
||||||
session.eval("sumOf(1,2,3)") // <- 6
|
// suspend fun doSomeWork(text: String): Int
|
||||||
|
addFn("doSomeWork") {
|
||||||
|
// this _is_ a suspend lambda, we can call suspend function,
|
||||||
|
// and it won't consume the thread.
|
||||||
|
// note that in kotlin handler, `args` is a list of `Obj` arguments
|
||||||
|
// and return value from this lambda should be Obj too:
|
||||||
|
doSomeWork(args[0]).toObj()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
// adding constant:
|
||||||
|
scope.eval("sumOf(1,2,3)") // <- 6
|
||||||
```
|
```
|
||||||
Note that the session reuses one scope, so state persists across `session.eval(...)` calls.
|
Note that the scope stores all changes in it so you can make calls on a single scope to preserve state between calls.
|
||||||
Use raw `Scope.eval(...)` only when you intentionally want low-level control without session-owned coroutine lifecycle.
|
|
||||||
|
|
||||||
## IntelliJ IDEA plugin: Lightweight autocompletion (experimental)
|
## IntelliJ IDEA plugin: Lightweight autocompletion (experimental)
|
||||||
|
|
||||||
@ -186,7 +177,8 @@ Designed to add scripting to kotlin multiplatform application in easy and effici
|
|||||||
|
|
||||||
# Language Roadmap
|
# Language Roadmap
|
||||||
|
|
||||||
The current stable release is **v1.5.5**: the 1.5 cycle now includes the database/date/concurrency additions as well as the latest compiler/runtime stabilization work, and the language, tooling, and site are aligned around this release.
|
We are now at **v1.5.0-SNAPSHOT** (stable development cycle): basic optimization performed, battery included: standard library is 90% here, initial
|
||||||
|
support in HTML, popular editors, and IDEA; tools to syntax highlight and format code are ready. It was released closed to schedule.
|
||||||
|
|
||||||
Ready features:
|
Ready features:
|
||||||
|
|
||||||
@ -223,21 +215,23 @@ Ready features:
|
|||||||
- [x] assign-if-null operator `?=`
|
- [x] assign-if-null operator `?=`
|
||||||
- [x] user-defined exception classes
|
- [x] user-defined exception classes
|
||||||
|
|
||||||
All of this is documented on the [language site](https://lynglang.com) and locally in [docs/tutorial.md](docs/tutorial.md). The site reflects the current release, while development snapshots continue in the private Maven repository.
|
All of this is documented in the [language site](https://lynglang.com) and locally [docs/language.md](docs/tutorial.md). the current nightly builds published on the site and in the private maven repository.
|
||||||
|
|
||||||
## plan: towards v2.0 Next Generation
|
## plan: towards v2.0 Next Generation
|
||||||
|
|
||||||
- [x] site with integrated interpreter to give a try
|
- [x] site with integrated interpreter to give a try
|
||||||
- [x] kotlin part public API good docs, integration focused
|
- [x] kotlin part public API good docs, integration focused
|
||||||
- [x] type specifications
|
- [ ] type specifications
|
||||||
- [x] Textmate Bundle
|
- [x] Textmate Bundle
|
||||||
- [x] IDEA plugin
|
- [x] IDEA plugin
|
||||||
- [x] source docs and maybe lyng.md to a standard
|
- [ ] source docs and maybe lyng.md to a standard
|
||||||
|
- [ ] metadata first class access from lyng
|
||||||
- [x] aggressive optimizations
|
- [x] aggressive optimizations
|
||||||
|
- [ ] compile to JVM bytecode optimization
|
||||||
|
|
||||||
## After 1.5 "Ideal scripting"
|
## After 1.5 "Ideal scripting"
|
||||||
|
|
||||||
* __we are here now ;)__
|
Estimated summer 2026
|
||||||
|
|
||||||
- propose your feature!
|
- propose your feature!
|
||||||
|
|
||||||
@ -245,12 +239,8 @@ All of this is documented on the [language site](https://lynglang.com) and local
|
|||||||
|
|
||||||
@-links are for contacting authors on [project home](https://gitea.sergeych.net/SergeychWorks/lyng): this simplest s to open issue for the person you need to convey any information about this project.
|
@-links are for contacting authors on [project home](https://gitea.sergeych.net/SergeychWorks/lyng): this simplest s to open issue for the person you need to convey any information about this project.
|
||||||
|
|
||||||
<img src="https://www.gravatar.com/avatar/7e3a56ff8a090fc9ffbd1909dea94904?s=32&d=identicon" alt="Sergey Chernov" width="32" height="32" style="vertical-align: middle; margin-right: 0.5em;" /> <b>Sergey Chernov</b> @sergeych, real.sergeych@gmail.com: Initial idea and architecture, language concept, design, implementation.
|
__Sergey Chernov__ @sergeych: Initial idea and architecture, language concept, design, implementation.
|
||||||
|
|
||||||
<br/>
|
|
||||||
|
|
||||||
<img src="https://www.gravatar.com/avatar/53a90bca30c85a81db8f0c0d8dea43a1?s=32&d=identicon" alt="Yulia Nezhinskaya" width="32" height="32" style="vertical-align: middle; margin-right: 0.5em;" /> <b>Yulia Nezhinskaya</b> @AlterEgoJuliaN, neleka88@gmail.com: System analysis, math and feature design.
|
|
||||||
|
|
||||||
|
|
||||||
|
__Yulia Nezhinskaya__ @AlterEgoJuliaN: System analysis, math and features design.
|
||||||
|
|
||||||
[parallelism]: docs/parallelism.md
|
[parallelism]: docs/parallelism.md
|
||||||
|
|||||||
@ -20,12 +20,13 @@
|
|||||||
set -e
|
set -e
|
||||||
echo "publishing all artifacts"
|
echo "publishing all artifacts"
|
||||||
echo
|
echo
|
||||||
./gradlew publishToMavenLocal site:jsBrowserDistribution publish buildInstallablePlugin :lyng:linkReleaseExecutableLinuxX64 :lyng:installJvmDist --parallel --no-configuration-cache
|
./gradlew publishToMavenLocal
|
||||||
|
./gradlew publish
|
||||||
|
|
||||||
#echo
|
echo
|
||||||
#echo "Creating plugin"
|
echo "Creating plugin"
|
||||||
#echo
|
echo
|
||||||
#./gradlew buildInstallablePlugin
|
./gradlew buildInstallablePlugin
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "building CLI tools"
|
echo "building CLI tools"
|
||||||
|
|||||||
@ -18,12 +18,10 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
upload_only=false
|
upload_only=false
|
||||||
target=vps # default: new server; use --old for d.lynglang.com
|
|
||||||
for arg in "$@"; do
|
for arg in "$@"; do
|
||||||
if [[ "$arg" == "-u" || "$arg" == "--upload-only" ]]; then
|
if [[ "$arg" == "-u" || "$arg" == "--upload-only" ]]; then
|
||||||
upload_only=true
|
upload_only=true
|
||||||
elif [[ "$arg" == "--old" ]]; then
|
break
|
||||||
target=com
|
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
@ -90,20 +88,19 @@ function updateIdeaPluginDownloadLink() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# target settings (-t com | -t vps)
|
# default target settings
|
||||||
case "$target" in
|
case "com" in
|
||||||
com)
|
com)
|
||||||
SSH_HOST=sergeych@d.lynglang.com
|
SSH_HOST=sergeych@d.lynglang.com # host to deploy to
|
||||||
SSH_PORT=22
|
SSH_PORT=22 # ssh port on it
|
||||||
ROOT=/bigstore/sergeych_pub/lyng
|
ROOT=/bigstore/sergeych_pub/lyng # directory to rsync to
|
||||||
;;
|
|
||||||
vps)
|
|
||||||
SSH_HOST=sergeych@94.130.36.94
|
|
||||||
SSH_PORT=22
|
|
||||||
ROOT=/var/www/lynglang
|
|
||||||
;;
|
;;
|
||||||
|
# com)
|
||||||
|
# SSH_HOST=vvk@front-01.neurodatalab.com
|
||||||
|
# ROOT=/home/vvk
|
||||||
|
# ;;
|
||||||
*)
|
*)
|
||||||
echo "*** ERROR: unknown target '$target' (use -t com | -t vps)"
|
echo "*** ERROR: target not specified (use deploy com | dev)"
|
||||||
echo "*** stop"
|
echo "*** stop"
|
||||||
exit 101
|
exit 101
|
||||||
esac
|
esac
|
||||||
|
|||||||
@ -1,7 +1,7 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
#
|
#
|
||||||
# Copyright 2026 Sergey S. Chernov real.sergeych@gmail.com
|
# Copyright 2025 Sergey S. Chernov real.sergeych@gmail.com
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# you may not use this file except in compliance with the License.
|
# you may not use this file except in compliance with the License.
|
||||||
@ -19,15 +19,13 @@
|
|||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
archive=./lyng/build/distributions/lyng-jvm.zip
|
root=./lyng/build/install/lyng-jvm/
|
||||||
install_root="$HOME/bin/jlyng-jvm"
|
|
||||||
launcher="$install_root/lyng-jvm/bin/lyng"
|
|
||||||
|
|
||||||
./gradlew :lyng:jvmDistZip
|
./gradlew :lyng:installJvmDist
|
||||||
mkdir -p ./distributables
|
#strip $file
|
||||||
cp "$archive" ./distributables/lyng-jvm.zip
|
#upx $file
|
||||||
rm -rf "$install_root" || true
|
rm -rf ~/bin/jlyng-jvm || true
|
||||||
rm "$HOME/bin/jlyng" 2>/dev/null || true
|
rm ~/bin/jlyng 2>/dev/null || true
|
||||||
mkdir -p "$install_root"
|
mkdir -p ~/bin/jlyng-jvm
|
||||||
unzip -q ./distributables/lyng-jvm.zip -d "$install_root"
|
cp -R $root ~/bin/jlyng-jvm
|
||||||
ln -s "$launcher" "$HOME/bin/jlyng"
|
ln -s ~/bin/jlyng-jvm/lyng-jvm/bin/lyng ~/bin/jlyng
|
||||||
|
|||||||
@ -35,27 +35,3 @@ tasks.register<Exec>("generateDocs") {
|
|||||||
description = "Generates a single-file documentation HTML using bin/generate_docs.sh"
|
description = "Generates a single-file documentation HTML using bin/generate_docs.sh"
|
||||||
commandLine("./bin/generate_docs.sh")
|
commandLine("./bin/generate_docs.sh")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sample generator task for .lyng.d definition files (not wired into build).
|
|
||||||
// Usage: ./gradlew generateLyngDefsSample
|
|
||||||
tasks.register("generateLyngDefsSample") {
|
|
||||||
group = "lyng"
|
|
||||||
description = "Generate a sample .lyng.d file under build/generated/lyng/defs"
|
|
||||||
outputs.dir(layout.buildDirectory.dir("generated/lyng/defs"))
|
|
||||||
doLast {
|
|
||||||
val outDir = layout.buildDirectory.dir("generated/lyng/defs").get().asFile
|
|
||||||
outDir.mkdirs()
|
|
||||||
val outFile = outDir.resolve("sample.lyng.d")
|
|
||||||
outFile.writeText(
|
|
||||||
"""
|
|
||||||
/** Generated API */
|
|
||||||
extern fun ping(): Int
|
|
||||||
|
|
||||||
/** Generated class */
|
|
||||||
class Generated(val name: String) {
|
|
||||||
fun greet(): String = "hi " + name
|
|
||||||
}
|
|
||||||
""".trimIndent()
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
7
bytecode_migration_plan.md
Normal file
7
bytecode_migration_plan.md
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
# Bytecode Migration Plan (Archived)
|
||||||
|
|
||||||
|
Status: completed.
|
||||||
|
|
||||||
|
Historical reference:
|
||||||
|
- `notes/archive/bytecode_migration_plan.md` (full plan)
|
||||||
|
- `notes/archive/bytecode_migration_plan_completed.md` (summary)
|
||||||
@ -1,15 +1,8 @@
|
|||||||
# Array
|
# Array
|
||||||
|
|
||||||
It's an interface if the [Collection] that provides indexing access, like `array[3] = 0`.
|
It's an interface if the [Collection] that provides indexing access, like `array[3] = 0`.
|
||||||
Array therefore implements [Iterable] too. Well known implementations of `Array` are
|
Array therefore implements [Iterable] too. The well known implementatino of the `Array` is
|
||||||
[List] and [ImmutableList].
|
[List].
|
||||||
|
|
||||||
The language-level bracket syntax supports one or more selectors:
|
|
||||||
|
|
||||||
- `value[i]`
|
|
||||||
- `value[i, j]`
|
|
||||||
|
|
||||||
Concrete array-like types decide what selectors they accept. Built-in list-like arrays use one selector at a time; custom types such as matrices may interpret multiple selectors.
|
|
||||||
|
|
||||||
Array adds the following methods:
|
Array adds the following methods:
|
||||||
|
|
||||||
@ -42,4 +35,3 @@ To pre-sort and array use `Iterable.sorted*` or in-place `List.sort*` families,
|
|||||||
[Collection]: Collection.md
|
[Collection]: Collection.md
|
||||||
[Iterable]: Iterable.md
|
[Iterable]: Iterable.md
|
||||||
[List]: List.md
|
[List]: List.md
|
||||||
[ImmutableList]: ImmutableList.md
|
|
||||||
|
|||||||
@ -120,18 +120,17 @@ which is used in `toString`) and hex encoding:
|
|||||||
|
|
||||||
## Members
|
## Members
|
||||||
|
|
||||||
| name | meaning | type |
|
| name | meaning | type |
|
||||||
|----------------------------|------------------------------------------------|---------------|
|
|----------------------------|-----------------------------------------|---------------|
|
||||||
| `size` | size | Int |
|
| `size` | size | Int |
|
||||||
| `decodeUtf8` | decode to String using UTF8 rules | Any |
|
| `decodeUtf8` | decode to String using UTF8 rules | Any |
|
||||||
| `+` | buffer concatenation | Any |
|
| `+` | buffer concatenation | Any |
|
||||||
| `toMutable()` | create a mutable copy | MutableBuffer |
|
| `toMutable()` | create a mutable copy | MutableBuffer |
|
||||||
| `hex` | encode to hex strign | String |
|
| `hex` | encode to hex strign | String |
|
||||||
| `Buffer.decodeHex(hexStr) | decode hex string | Buffer |
|
| `Buffer.decodeHex(hexStr) | decode hex string | Buffer |
|
||||||
| `base64` | encode to base64 (url flavor) (2) | String |
|
| `base64` | encode to base64 (url flavor) (2) | String |
|
||||||
| `base64std` | encode to base64 (default vocabulary, filling) | String |
|
| `Buffer.decodeBase64(str)` | decode base64 to new Buffer (2) | Buffer |
|
||||||
| `Buffer.decodeBase64(str)` | decode base64 to new Buffer (2) | Buffer |
|
| `toBitInput()` | create bit input from a byte buffer (3) | |
|
||||||
| `toBitInput()` | create bit input from a byte buffer (3) | |
|
|
||||||
|
|
||||||
(1)
|
(1)
|
||||||
: optimized implementation that override `Iterable` one
|
: optimized implementation that override `Iterable` one
|
||||||
|
|||||||
211
docs/Channel.md
211
docs/Channel.md
@ -1,211 +0,0 @@
|
|||||||
# Channel
|
|
||||||
|
|
||||||
A `Channel` is a **hot, bidirectional pipe** for passing values between concurrently running coroutines.
|
|
||||||
Unlike a [Flow], which is cold and replayed on every collection, a `Channel` is stateful: each value
|
|
||||||
sent is consumed by exactly one receiver.
|
|
||||||
|
|
||||||
Channels model the classic _producer / consumer_ pattern and are the right tool when:
|
|
||||||
|
|
||||||
- two or more coroutines need to exchange individual values at their own pace;
|
|
||||||
- you want back-pressure (rendezvous) or explicit buffering control;
|
|
||||||
- you need a push-based, hot data source (opposite of the pull-based, cold [Flow]).
|
|
||||||
|
|
||||||
## Constructors
|
|
||||||
|
|
||||||
```
|
|
||||||
Channel() // rendezvous — sender and receiver must meet
|
|
||||||
Channel(n: Int) // buffered — sender may run n items ahead of the receiver
|
|
||||||
Channel(Channel.UNLIMITED) // no limit on buffered items
|
|
||||||
```
|
|
||||||
|
|
||||||
**Rendezvous** (`Channel()`, capacity 0): `send` suspends until a matching `receive` is ready,
|
|
||||||
and vice-versa. This gives the tightest synchronisation and the smallest memory footprint.
|
|
||||||
|
|
||||||
**Buffered** (`Channel(n)`): `send` only suspends when the internal buffer is full. Allows the
|
|
||||||
producer to get up to _n_ items ahead of the consumer.
|
|
||||||
|
|
||||||
**Unlimited** (`Channel(Channel.UNLIMITED)`): `send` never suspends. Useful when the producer is
|
|
||||||
bursty and you do not want it blocked, but be careful not to grow the buffer without bound.
|
|
||||||
|
|
||||||
## Sending and receiving
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ch = Channel() // rendezvous channel
|
|
||||||
|
|
||||||
val producer = launch {
|
|
||||||
ch.send("hello") // suspends until the receiver is ready
|
|
||||||
ch.send("world")
|
|
||||||
ch.close() // signal: no more values
|
|
||||||
}
|
|
||||||
|
|
||||||
val a = ch.receive() // suspends until "hello" arrives
|
|
||||||
val b = ch.receive() // suspends until "world" arrives
|
|
||||||
val c = ch.receive() // channel is closed and drained → null
|
|
||||||
assertEquals("hello", a)
|
|
||||||
assertEquals("world", b)
|
|
||||||
assertEquals(null, c)
|
|
||||||
```
|
|
||||||
|
|
||||||
`receive()` returns `null` when the channel is both **closed** _and_ **fully drained** — that is
|
|
||||||
the idiomatic loop termination condition:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ch = Channel(4)
|
|
||||||
|
|
||||||
launch {
|
|
||||||
for (i in 1..5) ch.send(i)
|
|
||||||
ch.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
var item = ch.receive()
|
|
||||||
while (item != null) {
|
|
||||||
println(item)
|
|
||||||
item = ch.receive()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Non-suspending poll
|
|
||||||
|
|
||||||
`tryReceive()` never suspends. It returns the next buffered value, or `null` if the buffer is
|
|
||||||
empty or the channel is closed.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ch = Channel(8)
|
|
||||||
ch.send(42)
|
|
||||||
println(ch.tryReceive()) // 42
|
|
||||||
println(ch.tryReceive()) // null — nothing buffered right now
|
|
||||||
```
|
|
||||||
|
|
||||||
Use `tryReceive` for _polling_ patterns where blocking would be unacceptable, for example when
|
|
||||||
combining channel checks with other work inside a coroutine loop.
|
|
||||||
|
|
||||||
## Closing a channel
|
|
||||||
|
|
||||||
`close()` marks the channel so that no further `send` calls are accepted. Any items already in the
|
|
||||||
buffer can still be received. Once the buffer is drained, `receive()` returns `null` and
|
|
||||||
`isClosedForReceive` becomes `true`.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ch = Channel(2)
|
|
||||||
ch.send(1)
|
|
||||||
ch.send(2)
|
|
||||||
ch.close()
|
|
||||||
|
|
||||||
assert(ch.isClosedForSend)
|
|
||||||
assert(!ch.isClosedForReceive) // still has 2 buffered items
|
|
||||||
|
|
||||||
ch.receive() // 1
|
|
||||||
ch.receive() // 2
|
|
||||||
assert(ch.isClosedForReceive) // drained
|
|
||||||
```
|
|
||||||
|
|
||||||
Calling `send` after `close()` throws `IllegalStateException`.
|
|
||||||
|
|
||||||
## Properties
|
|
||||||
|
|
||||||
| property | type | description |
|
|
||||||
|---------------------|--------|----------------------------------------------------------|
|
|
||||||
| `isClosedForSend` | `Bool` | `true` after `close()` is called |
|
|
||||||
| `isClosedForReceive`| `Bool` | `true` when closed _and_ every buffered item is consumed |
|
|
||||||
|
|
||||||
## Methods
|
|
||||||
|
|
||||||
| method | suspends | description |
|
|
||||||
|-----------------|----------|----------------------------------------------------------------------------------|
|
|
||||||
| `send(value)` | yes | send a value; suspends when buffer full (rendezvous: always until partner ready) |
|
|
||||||
| `receive()` | yes | receive next value; suspends when empty; returns `null` when closed + drained |
|
|
||||||
| `tryReceive()` | no | return next buffered value or `null`; never suspends |
|
|
||||||
| `close()` | no | signal end of production; existing buffer items are still receivable |
|
|
||||||
|
|
||||||
## Static constants
|
|
||||||
|
|
||||||
| constant | value | description |
|
|
||||||
|---------------------|------------------|-------------------------------------|
|
|
||||||
| `Channel.UNLIMITED` | `Int.MAX_VALUE` | capacity for an unlimited-buffer channel |
|
|
||||||
|
|
||||||
## Common patterns
|
|
||||||
|
|
||||||
### Producer / consumer
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ch = Channel()
|
|
||||||
val results = []
|
|
||||||
val mu = Mutex()
|
|
||||||
|
|
||||||
val consumer = launch {
|
|
||||||
var item = ch.receive()
|
|
||||||
while (item != null) {
|
|
||||||
mu.withLock { results += item }
|
|
||||||
item = ch.receive()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
launch {
|
|
||||||
for (i in 1..5) ch.send("msg:$i")
|
|
||||||
ch.close()
|
|
||||||
}.await()
|
|
||||||
|
|
||||||
consumer.await()
|
|
||||||
println(results)
|
|
||||||
```
|
|
||||||
|
|
||||||
### Fan-out: one channel, many consumers
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ch = Channel(16)
|
|
||||||
|
|
||||||
// multiple consumers
|
|
||||||
val workers = (1..4).map { id ->
|
|
||||||
launch {
|
|
||||||
var task = ch.receive()
|
|
||||||
while (task != null) {
|
|
||||||
println("worker $id handles $task")
|
|
||||||
task = ch.receive()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// single producer
|
|
||||||
for (i in 1..20) ch.send(i)
|
|
||||||
ch.close()
|
|
||||||
|
|
||||||
workers.forEach { it.await() }
|
|
||||||
```
|
|
||||||
|
|
||||||
### Ping-pong between two coroutines
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val ping = Channel()
|
|
||||||
val pong = Channel()
|
|
||||||
|
|
||||||
launch {
|
|
||||||
repeat(3) {
|
|
||||||
val msg = ping.receive()
|
|
||||||
println("got: $msg → sending pong")
|
|
||||||
pong.send("pong")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
repeat(3) {
|
|
||||||
ping.send("ping")
|
|
||||||
println(pong.receive())
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Channel vs Flow
|
|
||||||
|
|
||||||
| | [Flow] | Channel |
|
|
||||||
|---|---|---|
|
|
||||||
| **temperature** | cold (lazy) | hot (eager) |
|
|
||||||
| **replay** | every collector gets a fresh run | each item is consumed once |
|
|
||||||
| **consumers** | any number; each gets all items | one receiver per item |
|
|
||||||
| **back-pressure** | built-in via rendezvous | configurable (rendezvous / buffered / unlimited) |
|
|
||||||
| **typical use** | transform pipelines, sequences | producer–consumer, fan-out |
|
|
||||||
|
|
||||||
## See also
|
|
||||||
|
|
||||||
- [parallelism] — `launch`, `Deferred`, `Mutex`, `Flow`, and the full concurrency picture
|
|
||||||
- [Flow] — cold async sequences
|
|
||||||
|
|
||||||
[Flow]: parallelism.md#flow
|
|
||||||
[parallelism]: parallelism.md
|
|
||||||
@ -6,13 +6,6 @@ Is a [Iterable] with known `size`, a finite [Iterable]:
|
|||||||
val size
|
val size
|
||||||
}
|
}
|
||||||
|
|
||||||
`Collection` is a read/traversal contract shared by mutable and immutable collections.
|
|
||||||
Concrete collection classes:
|
|
||||||
|
|
||||||
- Mutable: [List], [Set], [Map]
|
|
||||||
- Immutable: [ImmutableList], [ImmutableSet], [ImmutableMap]
|
|
||||||
- Observable mutable lists (opt-in module): [ObservableList]
|
|
||||||
|
|
||||||
| name | description |
|
| name | description |
|
||||||
|------------------------|------------------------------------------------------|
|
|------------------------|------------------------------------------------------|
|
||||||
|
|
||||||
@ -23,9 +16,4 @@ See [List], [Set], [Iterable] and [Efficient Iterables in Kotlin Interop](Effici
|
|||||||
|
|
||||||
[Iterable]: Iterable.md
|
[Iterable]: Iterable.md
|
||||||
[List]: List.md
|
[List]: List.md
|
||||||
[Set]: Set.md
|
[Set]: Set.md
|
||||||
[Map]: Map.md
|
|
||||||
[ImmutableList]: ImmutableList.md
|
|
||||||
[ImmutableSet]: ImmutableSet.md
|
|
||||||
[ImmutableMap]: ImmutableMap.md
|
|
||||||
[ObservableList]: ObservableList.md
|
|
||||||
@ -1,82 +0,0 @@
|
|||||||
# Complex Numbers (`lyng.complex`)
|
|
||||||
|
|
||||||
`lyng.complex` adds a pure-Lyng `Complex` type backed by `Real` components.
|
|
||||||
|
|
||||||
Import it when you want ordinary complex arithmetic:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.complex
|
|
||||||
```
|
|
||||||
|
|
||||||
## Construction
|
|
||||||
|
|
||||||
Use any of these:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.complex
|
|
||||||
|
|
||||||
val a = Complex(1.0, 2.0)
|
|
||||||
val b = complex(1.0, 2.0)
|
|
||||||
val c = 2.i
|
|
||||||
val d = 3.re
|
|
||||||
|
|
||||||
assertEquals(Complex(1.0, 2.0), 1 + 2.i)
|
|
||||||
```
|
|
||||||
|
|
||||||
Convenience extensions:
|
|
||||||
|
|
||||||
- `Int.re`, `Real.re`: embed a real value into the complex plane
|
|
||||||
- `Int.i`, `Real.i`: create a pure imaginary value
|
|
||||||
- `cis(angle)`: shorthand for `cos(angle) + i sin(angle)`
|
|
||||||
|
|
||||||
## Core Operations
|
|
||||||
|
|
||||||
`Complex` supports:
|
|
||||||
|
|
||||||
- `+`
|
|
||||||
- `-`
|
|
||||||
- `*`
|
|
||||||
- `/`
|
|
||||||
- unary `-`
|
|
||||||
- `conjugate`
|
|
||||||
- `magnitude`
|
|
||||||
- `phase`
|
|
||||||
|
|
||||||
Mixed arithmetic with `Int` and `Real` is enabled through `lyng.operators`, so both sides work naturally:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.complex
|
|
||||||
|
|
||||||
assertEquals(Complex(1.0, 2.0), 1 + 2.i)
|
|
||||||
assertEquals(Complex(1.5, 2.0), 1.5 + 2.i)
|
|
||||||
assertEquals(Complex(2.0, 2.0), 2.i + 2)
|
|
||||||
```
|
|
||||||
|
|
||||||
Mixed equality with built-in numeric types is intentionally not promised yet. Keep equality checks in the `Complex` domain for now.
|
|
||||||
|
|
||||||
## Transcendental Functions
|
|
||||||
|
|
||||||
For now, use member-style calls:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.complex
|
|
||||||
|
|
||||||
val z = 1 + π.i
|
|
||||||
val w = z.exp()
|
|
||||||
val s = z.sin()
|
|
||||||
val r = z.sqrt()
|
|
||||||
```
|
|
||||||
|
|
||||||
This is deliberate. Lyng already has built-in top-level real-valued functions such as `exp(x)` and `sin(x)`, and imported modules do not currently replace those root bindings. So plain `exp(z)` is not yet the right extension mechanism for complex math.
|
|
||||||
|
|
||||||
## Design Scope
|
|
||||||
|
|
||||||
This module intentionally uses `Complex` with `Real` parts, not `Complex<T>`.
|
|
||||||
|
|
||||||
Reasons:
|
|
||||||
|
|
||||||
- the existing math runtime is `Real`-centric
|
|
||||||
- the operator interop registry works with concrete runtime classes
|
|
||||||
- transcendental functions (`exp`, `sin`, `ln`, `sqrt`) are defined over the `Real` math backend here
|
|
||||||
|
|
||||||
If Lyng later gets a more general numeric-trait or callable-overload registry, a generic algebraic `Complex<T>` can be revisited on firmer ground.
|
|
||||||
325
docs/Decimal.md
325
docs/Decimal.md
@ -1,325 +0,0 @@
|
|||||||
# Decimal (`lyng.decimal`)
|
|
||||||
|
|
||||||
`lyng.decimal` adds an arbitrary-precision decimal type to Lyng as a normal library module.
|
|
||||||
|
|
||||||
Import it when you need decimal arithmetic that should not inherit `Real`'s binary floating-point behavior:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
```
|
|
||||||
|
|
||||||
## What `Decimal` Is For
|
|
||||||
|
|
||||||
Use `Decimal` when values are fundamentally decimal:
|
|
||||||
|
|
||||||
- money
|
|
||||||
- human-entered quantities
|
|
||||||
- exact decimal text
|
|
||||||
- predictable decimal rounding
|
|
||||||
- user-facing formatting and tests
|
|
||||||
|
|
||||||
Do not use it just because a number has a fractional part. `Real` is still the right type for ordinary double-precision numeric work.
|
|
||||||
|
|
||||||
## Creating Decimal Values
|
|
||||||
|
|
||||||
There are three supported conversions:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
val a = 1.d
|
|
||||||
val b = 2.2.d
|
|
||||||
val c = "2.2".d
|
|
||||||
|
|
||||||
assertEquals("1", a.toStringExpanded())
|
|
||||||
assertEquals("2.2", b.toStringExpanded())
|
|
||||||
assertEquals("2.2", c.toStringExpanded())
|
|
||||||
```
|
|
||||||
|
|
||||||
The three forms mean different things:
|
|
||||||
|
|
||||||
- `1.d`: convert `Int -> Decimal`
|
|
||||||
- `2.2.d`: convert `Real -> Decimal`
|
|
||||||
- `"2.2".d`: parse exact decimal text
|
|
||||||
|
|
||||||
That distinction is intentional.
|
|
||||||
|
|
||||||
### `Real.d` vs `"..." .d`
|
|
||||||
|
|
||||||
`Real.d` preserves the current `Real` value. It does not pretend the source code was exact decimal text.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("0.30000000000000004", (0.1 + 0.2).d.toStringExpanded())
|
|
||||||
assertEquals("0.3", "0.3".d.toStringExpanded())
|
|
||||||
```
|
|
||||||
|
|
||||||
This follows the "minimal confusion" rule:
|
|
||||||
|
|
||||||
- if you start from a `Real`, you get a decimal representation of that `Real`
|
|
||||||
- if you want exact decimal source text, use a `String`
|
|
||||||
|
|
||||||
## Factory Functions
|
|
||||||
|
|
||||||
The explicit factory methods are:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
Decimal.fromInt(10)
|
|
||||||
Decimal.fromReal(2.5)
|
|
||||||
Decimal.fromString("12.34")
|
|
||||||
```
|
|
||||||
|
|
||||||
These are equivalent to the conversion-property forms, but sometimes clearer in APIs or generated code.
|
|
||||||
|
|
||||||
## From Kotlin
|
|
||||||
|
|
||||||
If you already have an ionspin `BigDecimal` on the host side, the simplest supported way to create a Lyng `Decimal` is:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import com.ionspin.kotlin.bignum.decimal.BigDecimal
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.asFacade
|
|
||||||
import net.sergeych.lyng.newDecimal
|
|
||||||
|
|
||||||
val scope = EvalSession().getScope()
|
|
||||||
val decimal = scope.asFacade().newDecimal(BigDecimal.parseStringWithMode("12.34"))
|
|
||||||
```
|
|
||||||
|
|
||||||
Notes:
|
|
||||||
|
|
||||||
- `newDecimal(...)` loads `lyng.decimal` if needed
|
|
||||||
- it returns a real Lyng `Decimal` object instance
|
|
||||||
- this is the preferred Kotlin-side construction path when you already hold a host `BigDecimal`
|
|
||||||
|
|
||||||
## Core Operations
|
|
||||||
|
|
||||||
`Decimal` supports:
|
|
||||||
|
|
||||||
- `+`
|
|
||||||
- `-`
|
|
||||||
- `*`
|
|
||||||
- `/`
|
|
||||||
- `%`
|
|
||||||
- unary `-`
|
|
||||||
- comparison operators
|
|
||||||
- equality operators
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("3.75", ("1.5".d + "2.25".d).toStringExpanded())
|
|
||||||
assertEquals("1.25", ("2.5".d - "1.25".d).toStringExpanded())
|
|
||||||
assertEquals("3.0", ("1.5".d * 2.d).toStringExpanded())
|
|
||||||
assertEquals("0.5", (1.d / 2.d).toStringExpanded())
|
|
||||||
assert("2.0".d > "1.5".d)
|
|
||||||
assert("2.0".d == 2.d)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Interoperability With `Int` and `Real`
|
|
||||||
|
|
||||||
The decimal module registers mixed-operand operator bridges so both sides read naturally:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals(3.d, 1 + 2.d)
|
|
||||||
assertEquals(3.d, 2.d + 1)
|
|
||||||
assertEquals(1.5.d, 1.d + 0.5)
|
|
||||||
assertEquals(1.5.d, 0.5 + 1.d)
|
|
||||||
assert(2 == 2.d)
|
|
||||||
assert(3 > 2.d)
|
|
||||||
```
|
|
||||||
|
|
||||||
Without this registration mechanism, only the cases directly implemented on the left-hand class would work. The bridge fills the gap for expressions such as `Int + Decimal` and `Real + Decimal`.
|
|
||||||
|
|
||||||
See [OperatorInterop.md](OperatorInterop.md) for the generic mechanism behind that.
|
|
||||||
|
|
||||||
## String Representation
|
|
||||||
|
|
||||||
Use `toStringExpanded()` when you want plain decimal output without scientific notation:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("12.34", "12.34".d.toStringExpanded())
|
|
||||||
```
|
|
||||||
|
|
||||||
This is the recommended representation for:
|
|
||||||
|
|
||||||
- tests
|
|
||||||
- user-visible diagnostics
|
|
||||||
- decimal formatting checks
|
|
||||||
|
|
||||||
## Conversions Back To Built-ins
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals(2, "2.9".d.toInt())
|
|
||||||
assertEquals(2.9, "2.9".d.toReal())
|
|
||||||
```
|
|
||||||
|
|
||||||
Use `toReal()` only when you are willing to return to binary floating-point semantics.
|
|
||||||
|
|
||||||
## Non-Finite Checks
|
|
||||||
|
|
||||||
`Decimal` values are always finite, so these helpers exist for API symmetry with `Real` and always return `false`:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals(false, "2.9".d.isInfinite())
|
|
||||||
assertEquals(false, "2.9".d.isNaN())
|
|
||||||
```
|
|
||||||
|
|
||||||
## Division Context
|
|
||||||
|
|
||||||
Division is the operation where precision and rounding matter most.
|
|
||||||
|
|
||||||
By default, decimal division uses:
|
|
||||||
|
|
||||||
- precision: `34` significant digits
|
|
||||||
- rounding: `HalfEven`
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("0.3333333333333333333333333333333333", (1.d / 3.d).toStringExpanded())
|
|
||||||
assertEquals("0.6666666666666666666666666666666667", ("2".d / 3.d).toStringExpanded())
|
|
||||||
```
|
|
||||||
|
|
||||||
## `withDecimalContext(...)`
|
|
||||||
|
|
||||||
Use `withDecimalContext(...)` to override decimal division rules inside a block:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals(
|
|
||||||
"0.3333333333",
|
|
||||||
withDecimalContext(10) { (1.d / 3.d).toStringExpanded() }
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
You can also pass an explicit context object:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
val ctx = DecimalContext(6, DecimalRounding.HalfAwayFromZero)
|
|
||||||
assertEquals("0.666667", withDecimalContext(ctx) { ("2".d / 3.d).toStringExpanded() })
|
|
||||||
```
|
|
||||||
|
|
||||||
The context is dynamic and local to the block. After the block exits, the previous context is restored.
|
|
||||||
|
|
||||||
## Rounding Modes
|
|
||||||
|
|
||||||
Available rounding modes:
|
|
||||||
|
|
||||||
- `HalfEven`
|
|
||||||
- `HalfAwayFromZero`
|
|
||||||
- `HalfTowardsZero`
|
|
||||||
- `Ceiling`
|
|
||||||
- `Floor`
|
|
||||||
- `AwayFromZero`
|
|
||||||
- `TowardsZero`
|
|
||||||
|
|
||||||
Tie example at precision `2`:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("0.12", withDecimalContext(2, DecimalRounding.HalfEven) { (1.d / 8.d).toStringExpanded() })
|
|
||||||
assertEquals("0.13", withDecimalContext(2, DecimalRounding.HalfAwayFromZero) { (1.d / 8.d).toStringExpanded() })
|
|
||||||
assertEquals("0.12", withDecimalContext(2, DecimalRounding.HalfTowardsZero) { (1.d / 8.d).toStringExpanded() })
|
|
||||||
assertEquals("0.13", withDecimalContext(2, DecimalRounding.Ceiling) { (1.d / 8.d).toStringExpanded() })
|
|
||||||
assertEquals("0.12", withDecimalContext(2, DecimalRounding.Floor) { (1.d / 8.d).toStringExpanded() })
|
|
||||||
```
|
|
||||||
|
|
||||||
Negative values follow the same named policy in the obvious direction:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("-0.13", withDecimalContext(2, DecimalRounding.HalfAwayFromZero) { (-1.d / 8.d).toStringExpanded() })
|
|
||||||
assertEquals("-0.12", withDecimalContext(2, DecimalRounding.HalfTowardsZero) { (-1.d / 8.d).toStringExpanded() })
|
|
||||||
```
|
|
||||||
|
|
||||||
## Recommended Usage Rules
|
|
||||||
|
|
||||||
## Decimal With Stdlib Math Functions
|
|
||||||
|
|
||||||
Core math helpers such as `abs`, `floor`, `ceil`, `round`, `sin`, `exp`, `ln`, `sqrt`, `log10`, `log2`, and `pow`
|
|
||||||
now also accept `Decimal`.
|
|
||||||
|
|
||||||
Current behavior is intentionally split:
|
|
||||||
|
|
||||||
- exact decimal implementation:
|
|
||||||
- `abs(x)`
|
|
||||||
- `floor(x)`
|
|
||||||
- `ceil(x)`
|
|
||||||
- `round(x)`
|
|
||||||
- `pow(x, y)` when `x` is `Decimal` and `y` is an integral exponent
|
|
||||||
- temporary bridge through `Real`:
|
|
||||||
- `sin`, `cos`, `tan`
|
|
||||||
- `asin`, `acos`, `atan`
|
|
||||||
- `sinh`, `cosh`, `tanh`
|
|
||||||
- `asinh`, `acosh`, `atanh`
|
|
||||||
- `exp`, `ln`, `log10`, `log2`
|
|
||||||
- `sqrt`
|
|
||||||
- `pow` for the remaining non-integral decimal exponent cases
|
|
||||||
|
|
||||||
The temporary bridge is:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
Decimal -> Real -> host math -> Decimal
|
|
||||||
```
|
|
||||||
|
|
||||||
This is a compatibility step, not the long-term design. Native decimal implementations will replace these bridge-based
|
|
||||||
paths over time.
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("2.5", (abs("-2.5".d) as Decimal).toStringExpanded())
|
|
||||||
assertEquals("2", (floor("2.9".d) as Decimal).toStringExpanded())
|
|
||||||
|
|
||||||
// Temporary Real bridge:
|
|
||||||
assertEquals((exp(1.25) as Real).d.toStringExpanded(), (exp("1.25".d) as Decimal).toStringExpanded())
|
|
||||||
assertEquals((sqrt(2.0) as Real).d.toStringExpanded(), (sqrt("2".d) as Decimal).toStringExpanded())
|
|
||||||
```
|
|
||||||
|
|
||||||
If you care about exact decimal source text:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
"12.34".d
|
|
||||||
```
|
|
||||||
|
|
||||||
If you intentionally convert an existing binary floating-point value:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
someReal.d
|
|
||||||
```
|
|
||||||
|
|
||||||
If you want local control over division:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
withDecimalContext(precision, rounding) { ... }
|
|
||||||
```
|
|
||||||
|
|
||||||
If you want custom mixed operators for your own type, follow the same pattern as Decimal and use the operator registry:
|
|
||||||
|
|
||||||
- define the operators on your own class
|
|
||||||
- choose a common class
|
|
||||||
- register mixed operand bridges
|
|
||||||
|
|
||||||
See [OperatorInterop.md](OperatorInterop.md).
|
|
||||||
@ -1,37 +0,0 @@
|
|||||||
# ImmutableList built-in class
|
|
||||||
|
|
||||||
`ImmutableList` is an immutable, indexable list value.
|
|
||||||
It implements [Array], therefore [Collection] and [Iterable].
|
|
||||||
|
|
||||||
Use it when API contracts require a list that cannot be mutated through aliases.
|
|
||||||
|
|
||||||
## Creating
|
|
||||||
|
|
||||||
val a = ImmutableList(1,2,3)
|
|
||||||
val b = [1,2,3].toImmutable()
|
|
||||||
val c = (1..3).toImmutableList()
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Converting
|
|
||||||
|
|
||||||
val i = ImmutableList(1,2,3)
|
|
||||||
val m = i.toMutable()
|
|
||||||
m += 4
|
|
||||||
assertEquals( ImmutableList(1,2,3), i )
|
|
||||||
assertEquals( [1,2,3,4], m )
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Members
|
|
||||||
|
|
||||||
| name | meaning |
|
|
||||||
|---------------|-----------------------------------------|
|
|
||||||
| `size` | number of elements |
|
|
||||||
| `[index]` | element access by index |
|
|
||||||
| `[Range]` | immutable slice |
|
|
||||||
| `+` | append element(s), returns new immutable list |
|
|
||||||
| `-` | remove element(s), returns new immutable list |
|
|
||||||
| `toMutable()` | create mutable copy |
|
|
||||||
|
|
||||||
[Array]: Array.md
|
|
||||||
[Collection]: Collection.md
|
|
||||||
[Iterable]: Iterable.md
|
|
||||||
@ -1,36 +0,0 @@
|
|||||||
# ImmutableMap built-in class
|
|
||||||
|
|
||||||
`ImmutableMap` is an immutable map of key-value pairs.
|
|
||||||
It implements [Collection] and [Iterable] of [MapEntry].
|
|
||||||
|
|
||||||
## Creating
|
|
||||||
|
|
||||||
val a = ImmutableMap("a" => 1, "b" => 2)
|
|
||||||
val b = Map("a" => 1, "b" => 2).toImmutable()
|
|
||||||
val c = ["a" => 1, "b" => 2].toImmutableMap
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Converting
|
|
||||||
|
|
||||||
val i = ImmutableMap("a" => 1)
|
|
||||||
val m = i.toMutable()
|
|
||||||
m["a"] = 2
|
|
||||||
assertEquals( 1, i["a"] )
|
|
||||||
assertEquals( 2, m["a"] )
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Members
|
|
||||||
|
|
||||||
| name | meaning |
|
|
||||||
|-----------------|------------------------------------------|
|
|
||||||
| `size` | number of entries |
|
|
||||||
| `[key]` | get value by key, or `null` if absent |
|
|
||||||
| `getOrNull(key)`| same as `[key]` |
|
|
||||||
| `keys` | list of keys |
|
|
||||||
| `values` | list of values |
|
|
||||||
| `+` | merge (rightmost wins), returns new immutable map |
|
|
||||||
| `toMutable()` | create mutable copy |
|
|
||||||
|
|
||||||
[Collection]: Collection.md
|
|
||||||
[Iterable]: Iterable.md
|
|
||||||
[MapEntry]: Map.md
|
|
||||||
@ -1,34 +0,0 @@
|
|||||||
# ImmutableSet built-in class
|
|
||||||
|
|
||||||
`ImmutableSet` is an immutable set of unique elements.
|
|
||||||
It implements [Collection] and [Iterable].
|
|
||||||
|
|
||||||
## Creating
|
|
||||||
|
|
||||||
val a = ImmutableSet(1,2,3)
|
|
||||||
val b = Set(1,2,3).toImmutable()
|
|
||||||
val c = [1,2,3].toImmutableSet
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Converting
|
|
||||||
|
|
||||||
val i = ImmutableSet(1,2,3)
|
|
||||||
val m = i.toMutable()
|
|
||||||
m += 4
|
|
||||||
assertEquals( ImmutableSet(1,2,3), i )
|
|
||||||
assertEquals( Set(1,2,3,4), m )
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Members
|
|
||||||
|
|
||||||
| name | meaning |
|
|
||||||
|---------------|-----------------------------------------------------|
|
|
||||||
| `size` | number of elements |
|
|
||||||
| `contains(x)` | membership test |
|
|
||||||
| `+`, `union` | union, returns new immutable set |
|
|
||||||
| `-`, `subtract` | subtraction, returns new immutable set |
|
|
||||||
| `*`, `intersect` | intersection, returns new immutable set |
|
|
||||||
| `toMutable()` | create mutable copy |
|
|
||||||
|
|
||||||
[Collection]: Collection.md
|
|
||||||
[Iterable]: Iterable.md
|
|
||||||
@ -55,23 +55,6 @@ Here is the sample:
|
|||||||
assertEquals( (1..3).joinToString { it * 10 }, "10 20 30")
|
assertEquals( (1..3).joinToString { it * 10 }, "10 20 30")
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
## joinAll
|
|
||||||
|
|
||||||
`joinAll()` is an `Iterable<Deferred>` helper that awaits every deferred in iteration order and returns a `List`
|
|
||||||
with the collected results.
|
|
||||||
|
|
||||||
val jobs = (1..4).map { n ->
|
|
||||||
launch { n * n }
|
|
||||||
}
|
|
||||||
assertEquals([1, 4, 9, 16], jobs.joinAll())
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Notes:
|
|
||||||
|
|
||||||
- it does not start any task by itself; it only awaits the deferreds already present in the iterable.
|
|
||||||
- awaiting happens in iteration order, so the result list keeps the same order as the input iterable.
|
|
||||||
- if any deferred fails or was cancelled, that `await()` error is propagated from `joinAll()`.
|
|
||||||
|
|
||||||
## `sum` and `sumOf`
|
## `sum` and `sumOf`
|
||||||
|
|
||||||
These, again, does the thing:
|
These, again, does the thing:
|
||||||
@ -125,8 +108,8 @@ You can also use flow variations that return a cold `Flow` instead of a `List`,
|
|||||||
Find the minimum or maximum value of a function applied to each element:
|
Find the minimum or maximum value of a function applied to each element:
|
||||||
|
|
||||||
val source = ["abc", "de", "fghi"]
|
val source = ["abc", "de", "fghi"]
|
||||||
assertEquals(2, source.minOf { (it as String).length })
|
assertEquals(2, source.minOf { it.length })
|
||||||
assertEquals(4, source.maxOf { (it as String).length })
|
assertEquals(4, source.maxOf { it.length })
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
## flatten and flatMap
|
## flatten and flatMap
|
||||||
@ -164,15 +147,12 @@ Search for the first element that satisfies the given predicate:
|
|||||||
| fun/method | description |
|
| fun/method | description |
|
||||||
|------------------------|---------------------------------------------------------------------------------|
|
|------------------------|---------------------------------------------------------------------------------|
|
||||||
| toList() | create a list from iterable |
|
| toList() | create a list from iterable |
|
||||||
| toImmutableList() | create an immutable list from iterable |
|
|
||||||
| toSet() | create a set from iterable |
|
| toSet() | create a set from iterable |
|
||||||
| toImmutableSet | create an immutable set from iterable |
|
|
||||||
| contains(i) | check that iterable contains `i` |
|
| contains(i) | check that iterable contains `i` |
|
||||||
| `i in iterable` | same as `contains(i)` |
|
| `i in iterable` | same as `contains(i)` |
|
||||||
| isEmpty() | check iterable is empty |
|
| isEmpty() | check iterable is empty |
|
||||||
| forEach(f) | call f for each element |
|
| forEach(f) | call f for each element |
|
||||||
| toMap() | create a map from list of key-value pairs (arrays of 2 items or like) |
|
| toMap() | create a map from list of key-value pairs (arrays of 2 items or like) |
|
||||||
| toImmutableMap | create an immutable map from list of key-value pairs |
|
|
||||||
| any(p) | true if any element matches predicate `p` |
|
| any(p) | true if any element matches predicate `p` |
|
||||||
| all(p) | true if all elements match predicate `p` |
|
| all(p) | true if all elements match predicate `p` |
|
||||||
| map(f) | create a list of values returned by `f` called for each element of the iterable |
|
| map(f) | create a list of values returned by `f` called for each element of the iterable |
|
||||||
@ -201,7 +181,6 @@ Search for the first element that satisfies the given predicate:
|
|||||||
| sortedWith(comparator) | sort using a comparator that compares elements (1) |
|
| sortedWith(comparator) | sort using a comparator that compares elements (1) |
|
||||||
| sortedBy(predicate) | sort by comparing results of the predicate function |
|
| sortedBy(predicate) | sort by comparing results of the predicate function |
|
||||||
| joinToString(s,t) | convert iterable to string, see (2) |
|
| joinToString(s,t) | convert iterable to string, see (2) |
|
||||||
| joinAll() | for `Iterable<Deferred>`, await all items in order and collect results to [List] |
|
|
||||||
| reversed() | create a list containing items from this in reverse order |
|
| reversed() | create a list containing items from this in reverse order |
|
||||||
| shuffled() | create a list of shuffled elements |
|
| shuffled() | create a list of shuffled elements |
|
||||||
|
|
||||||
@ -227,20 +206,16 @@ For high-performance Kotlin-side interop and custom iterable implementation deta
|
|||||||
|
|
||||||
## Implemented in classes:
|
## Implemented in classes:
|
||||||
|
|
||||||
- [List], [ImmutableList], [Range], [Buffer](Buffer.md), [BitBuffer], [Buffer], [Set], [ImmutableSet], [Map], [ImmutableMap], [RingBuffer]
|
- [List], [Range], [Buffer](Buffer.md), [BitBuffer], [Buffer], [Set], [RingBuffer]
|
||||||
|
|
||||||
[Collection]: Collection.md
|
[Collection]: Collection.md
|
||||||
|
|
||||||
[List]: List.md
|
[List]: List.md
|
||||||
[ImmutableList]: ImmutableList.md
|
|
||||||
|
|
||||||
[Flow]: parallelism.md#flow
|
[Flow]: parallelism.md#flow
|
||||||
|
|
||||||
[Range]: Range.md
|
[Range]: Range.md
|
||||||
|
|
||||||
[Set]: Set.md
|
[Set]: Set.md
|
||||||
[ImmutableSet]: ImmutableSet.md
|
|
||||||
[Map]: Map.md
|
|
||||||
[ImmutableMap]: ImmutableMap.md
|
|
||||||
|
|
||||||
[RingBuffer]: RingBuffer.md
|
[RingBuffer]: RingBuffer.md
|
||||||
@ -1,110 +0,0 @@
|
|||||||
# LaunchPool
|
|
||||||
|
|
||||||
`LaunchPool` is a bounded-concurrency task pool: you submit lambdas with `launch`, and the pool runs them using a fixed number of worker coroutines.
|
|
||||||
|
|
||||||
## Constructor
|
|
||||||
|
|
||||||
```
|
|
||||||
LaunchPool(maxWorkers, maxQueueSize = Channel.UNLIMITED)
|
|
||||||
```
|
|
||||||
|
|
||||||
| Parameter | Description |
|
|
||||||
|-----------|-------------|
|
|
||||||
| `maxWorkers` | Maximum number of tasks that run in parallel. |
|
|
||||||
| `maxQueueSize` | Maximum number of tasks that may wait in the queue. When the queue is full, `launch` suspends the caller until space becomes available. Defaults to `Channel.UNLIMITED` (no bound). |
|
|
||||||
|
|
||||||
## Methods
|
|
||||||
|
|
||||||
### `launch(lambda): Deferred`
|
|
||||||
|
|
||||||
Schedules `lambda` for execution and returns a `Deferred` for its result.
|
|
||||||
|
|
||||||
- Suspends if the queue is full (`maxQueueSize` reached).
|
|
||||||
- Throws `IllegalStateException` if the pool is already closed or cancelled.
|
|
||||||
- Any exception thrown by `lambda` is captured in the returned `Deferred` and **does not escape the pool**.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val pool = LaunchPool(4)
|
|
||||||
val d1 = pool.launch { computeSomething() }
|
|
||||||
val d2 = pool.launch { computeOther() }
|
|
||||||
pool.closeAndJoin()
|
|
||||||
println(d1.await())
|
|
||||||
println(d2.await())
|
|
||||||
```
|
|
||||||
|
|
||||||
### `closeAndJoin()`
|
|
||||||
|
|
||||||
Stops accepting new tasks and suspends until all queued and running tasks complete normally. After this call, any further `launch` throws `IllegalStateException`. Idempotent — safe to call multiple times.
|
|
||||||
|
|
||||||
### `cancel()`
|
|
||||||
|
|
||||||
Immediately closes the queue and cancels all worker coroutines. Queued but unstarted tasks are discarded. After this call, `launch` throws `IllegalStateException`. Idempotent.
|
|
||||||
|
|
||||||
### `cancelAndJoin()`
|
|
||||||
|
|
||||||
Like `cancel()`, but also suspends until all worker coroutines have stopped. Useful when you need to be sure no worker code is still running before proceeding. Idempotent.
|
|
||||||
|
|
||||||
## Exception handling
|
|
||||||
|
|
||||||
Exceptions from submitted lambdas are captured per-task in the returned `Deferred`. The pool itself continues running after a task failure:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val pool = LaunchPool(2)
|
|
||||||
val good = pool.launch { 42 }
|
|
||||||
val bad = pool.launch { throw IllegalArgumentException("boom") }
|
|
||||||
pool.closeAndJoin()
|
|
||||||
|
|
||||||
assertEquals(42, good.await())
|
|
||||||
assertThrows(IllegalArgumentException) { bad.await() }
|
|
||||||
```
|
|
||||||
|
|
||||||
## Bounded queue / back-pressure
|
|
||||||
|
|
||||||
When `maxQueueSize` is set, the producer suspends if the queue fills up, providing automatic back-pressure:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
// 1 worker, queue of 2 — producer can be at most 2 tasks ahead of what's running
|
|
||||||
val pool = LaunchPool(1, 2)
|
|
||||||
val d1 = pool.launch { delay(10); "a" }
|
|
||||||
val d2 = pool.launch { delay(10); "b" }
|
|
||||||
val d3 = pool.launch { delay(10); "c" } // suspends until d1 is picked up by the worker
|
|
||||||
pool.closeAndJoin()
|
|
||||||
```
|
|
||||||
|
|
||||||
## Collecting all results
|
|
||||||
|
|
||||||
`launch` returns a `Deferred`, so you can collect results with `joinAll()`:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val pool = LaunchPool(4)
|
|
||||||
val jobs = (1..10).map { n -> pool.launch { n * n } }
|
|
||||||
pool.closeAndJoin()
|
|
||||||
val results = jobs.joinAll()
|
|
||||||
// results == [1, 4, 9, 16, 25, 36, 49, 64, 81, 100]
|
|
||||||
```
|
|
||||||
|
|
||||||
## Concurrency limit in practice
|
|
||||||
|
|
||||||
With `maxWorkers = 2`, at most 2 tasks run simultaneously regardless of how many are queued:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val mu = Mutex()
|
|
||||||
var active = 0
|
|
||||||
var maxSeen = 0
|
|
||||||
|
|
||||||
val pool = LaunchPool(2)
|
|
||||||
(1..8).map {
|
|
||||||
pool.launch {
|
|
||||||
mu.withLock { active++; if (active > maxSeen) maxSeen = active }
|
|
||||||
delay(5)
|
|
||||||
mu.withLock { active-- }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pool.closeAndJoin()
|
|
||||||
assert(maxSeen <= 2)
|
|
||||||
```
|
|
||||||
|
|
||||||
## See also
|
|
||||||
|
|
||||||
- [parallelism.md](parallelism.md) — `launch`, `Deferred`, `Mutex`, `Channel`, and coroutine basics
|
|
||||||
- [Channel.md](Channel.md) — the underlying channel primitive used by `LaunchPool`
|
|
||||||
@ -1,66 +0,0 @@
|
|||||||
# Legacy Digest Functions (`lyng.legacy_digest`)
|
|
||||||
|
|
||||||
> ⚠️ **Security warning:** The functions in this module use cryptographically broken
|
|
||||||
> algorithms. Do **not** use them for passwords, digital signatures, integrity
|
|
||||||
> verification against adversarial tampering, or any other security-sensitive
|
|
||||||
> purpose. They exist solely for compatibility with legacy protocols and file
|
|
||||||
> formats that require specific hash values.
|
|
||||||
|
|
||||||
Import when you need to produce a SHA-1 digest for an existing protocol or format:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.legacy_digest
|
|
||||||
```
|
|
||||||
|
|
||||||
## `LegacyDigest` Object
|
|
||||||
|
|
||||||
### `sha1(data): String`
|
|
||||||
|
|
||||||
Computes the SHA-1 digest of `data` and returns it as a 40-character lowercase
|
|
||||||
hex string.
|
|
||||||
|
|
||||||
`data` can be:
|
|
||||||
|
|
||||||
| Type | Behaviour |
|
|
||||||
|----------|----------------------------------------|
|
|
||||||
| `String` | Encoded as UTF-8, then hashed |
|
|
||||||
| `Buffer` | Raw bytes hashed directly |
|
|
||||||
| anything | Falls back to `toString()` then UTF-8 |
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.legacy_digest
|
|
||||||
|
|
||||||
// String input
|
|
||||||
val h = LegacyDigest.sha1("abc")
|
|
||||||
assertEquals("a9993e364706816aba3e25717850c26c9cd0d89d", h)
|
|
||||||
|
|
||||||
// Empty string
|
|
||||||
assertEquals("da39a3ee5e6b4b0d3255bfef95601890afd80709", LegacyDigest.sha1(""))
|
|
||||||
```
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.legacy_digest
|
|
||||||
import lyng.buffer
|
|
||||||
|
|
||||||
// Buffer input (raw bytes)
|
|
||||||
val buf = Buffer.decodeHex("616263") // 0x61 0x62 0x63 = "abc"
|
|
||||||
assertEquals("a9993e364706816aba3e25717850c26c9cd0d89d", LegacyDigest.sha1(buf))
|
|
||||||
```
|
|
||||||
|
|
||||||
## Implementation Notes
|
|
||||||
|
|
||||||
- Pure Kotlin/KMP — no native libraries or extra dependencies.
|
|
||||||
- Follows FIPS 180-4.
|
|
||||||
- The output is always lowercase hex, never uppercase or binary.
|
|
||||||
|
|
||||||
## When to Use
|
|
||||||
|
|
||||||
Use `lyng.legacy_digest` only when an external system you cannot change requires
|
|
||||||
a SHA-1 value, for example:
|
|
||||||
|
|
||||||
- old git-style content addresses
|
|
||||||
- some OAuth 1.0 / HMAC-SHA1 signature schemes
|
|
||||||
- legacy file checksums defined in published specs
|
|
||||||
|
|
||||||
For any new design choose a current hash function (SHA-256 or better) once
|
|
||||||
Lyng adds a `lyng.digest` module.
|
|
||||||
75
docs/List.md
75
docs/List.md
@ -1,8 +1,6 @@
|
|||||||
# List built-in class
|
# List built-in class
|
||||||
|
|
||||||
Mutable list of any objects.
|
Mutable list of any objects.
|
||||||
For immutable list values, see [ImmutableList].
|
|
||||||
For observable mutable lists and change hooks, see [ObservableList].
|
|
||||||
|
|
||||||
It's class in Lyng is `List`:
|
It's class in Lyng is `List`:
|
||||||
|
|
||||||
@ -30,13 +28,6 @@ There is a shortcut for the last:
|
|||||||
|
|
||||||
__Important__ negative indexes works wherever indexes are used, e.g. in insertion and removal methods too.
|
__Important__ negative indexes works wherever indexes are used, e.g. in insertion and removal methods too.
|
||||||
|
|
||||||
The language also allows multi-selector indexing syntax such as `value[i, j]`, but `List` itself uses a single selector only:
|
|
||||||
|
|
||||||
- `list[index]` for one element
|
|
||||||
- `list[range]` for a slice copy
|
|
||||||
|
|
||||||
Multi-selector indexing is intended for custom indexers such as `Matrix`.
|
|
||||||
|
|
||||||
## Concatenation
|
## Concatenation
|
||||||
|
|
||||||
You can concatenate lists or iterable objects:
|
You can concatenate lists or iterable objects:
|
||||||
@ -45,16 +36,6 @@ You can concatenate lists or iterable objects:
|
|||||||
assert( [4,5] + (1..3) == [4, 5, 1, 2, 3])
|
assert( [4,5] + (1..3) == [4, 5, 1, 2, 3])
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
## Constructing lists
|
|
||||||
|
|
||||||
Besides literals, you can build a list by size using `List.fill`:
|
|
||||||
|
|
||||||
val squares = List.fill(5) { i -> i * i }
|
|
||||||
assertEquals([0, 1, 4, 9, 16], squares)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
`List.fill(size) { ... }` calls the block once for each index from `0` to `size - 1` and returns a new mutable list.
|
|
||||||
|
|
||||||
## Appending
|
## Appending
|
||||||
|
|
||||||
To append to lists, use `+=` with elements, lists and any [Iterable] instances, but beware it will
|
To append to lists, use `+=` with elements, lists and any [Iterable] instances, but beware it will
|
||||||
@ -174,9 +155,6 @@ List could be sorted in place, just like [Collection] provide sorted copies, in
|
|||||||
| `[index]` | get or set element at index | Int |
|
| `[index]` | get or set element at index | Int |
|
||||||
| `[Range]` | get slice of the array (copy) | Range |
|
| `[Range]` | get slice of the array (copy) | Range |
|
||||||
| `+=` | append element(s) (2) | List or Obj |
|
| `+=` | append element(s) (2) | List or Obj |
|
||||||
| `List.fill(size, block)` | build a new list from indices `0..<size` | Int, Callable |
|
|
||||||
| `List.fill(size,capacity,block)` | same, pre-allocating capacity slots | Int, Int, Callable |
|
|
||||||
| `ensureCapacity(count)` | pre-allocate storage for at least `count` elements without reallocation (5) | Int |
|
|
||||||
| `sort()` | in-place sort, natural order | void |
|
| `sort()` | in-place sort, natural order | void |
|
||||||
| `sortBy(predicate)` | in-place sort bu `predicate` call result (3) | void |
|
| `sortBy(predicate)` | in-place sort bu `predicate` call result (3) | void |
|
||||||
| `sortWith(comparator)` | in-place sort using `comarator` function (4) | void |
|
| `sortWith(comparator)` | in-place sort using `comarator` function (4) | void |
|
||||||
@ -199,57 +177,8 @@ order, e.g. is same as `list.sortWith { a,b -> predicate(a) <=> predicate(b) }`
|
|||||||
positive if first is greater, and zero if they are equal. For example, the equvalent comparator
|
positive if first is greater, and zero if they are equal. For example, the equvalent comparator
|
||||||
for `sort()` will be `sort { a, b -> a <=> b }
|
for `sort()` will be `sort { a, b -> a <=> b }
|
||||||
|
|
||||||
(5)
|
|
||||||
: if the current capacity is already ≥ `count`, this is a no-op. Otherwise the internal storage
|
|
||||||
is reallocated to hold at least `count` elements. Use this before a bulk `+=` loop to avoid
|
|
||||||
repeated reallocations. `List.fill(size, capacity, block)` calls this automatically.
|
|
||||||
|
|
||||||
It inherits from [Iterable] too and thus all iterable methods are applicable to any list.
|
It inherits from [Iterable] too and thus all iterable methods are applicable to any list.
|
||||||
|
|
||||||
## Observable list hooks
|
|
||||||
|
|
||||||
Observable hooks are provided by module `lyng.observable` and are opt-in:
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
|
|
||||||
val src = [1,2,3]
|
|
||||||
val xs = src.observable()
|
|
||||||
assert(xs is ObservableList<Int>)
|
|
||||||
|
|
||||||
var before = 0
|
|
||||||
var after = 0
|
|
||||||
xs.beforeChange { before++ }
|
|
||||||
xs.onChange { after++ }
|
|
||||||
|
|
||||||
xs += 4
|
|
||||||
xs[0] = 100
|
|
||||||
assertEquals([100,2,3,4], xs)
|
|
||||||
assertEquals(2, before)
|
|
||||||
assertEquals(2, after)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
`beforeChange` runs before mutation commit and may reject it by throwing exception (typically `ChangeRejectionException` from the same module):
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
val xs = [1,2].observable()
|
|
||||||
xs.beforeChange { throw ChangeRejectionException("read only") }
|
|
||||||
assertThrows(ChangeRejectionException) { xs += 3 }
|
|
||||||
assertEquals([1,2], xs)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
`changes()` returns `Flow<ListChange<T>>` of committed events:
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
val xs = [10,20].observable()
|
|
||||||
val it = xs.changes().iterator()
|
|
||||||
xs += 30
|
|
||||||
assert(it.hasNext())
|
|
||||||
val e = it.next()
|
|
||||||
assert(e is ListInsert<Int>)
|
|
||||||
assertEquals([30], (e as ListInsert<Int>).values)
|
|
||||||
it.cancelIteration()
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Member inherited from Array
|
## Member inherited from Array
|
||||||
|
|
||||||
| name | meaning | type |
|
| name | meaning | type |
|
||||||
@ -267,6 +196,4 @@ Observable hooks are provided by module `lyng.observable` and are opt-in:
|
|||||||
|
|
||||||
[Range]: Range.md
|
[Range]: Range.md
|
||||||
|
|
||||||
[Iterable]: Iterable.md
|
[Iterable]: Iterable.md
|
||||||
[ImmutableList]: ImmutableList.md
|
|
||||||
[ObservableList]: ObservableList.md
|
|
||||||
@ -3,7 +3,6 @@
|
|||||||
Map is a mutable collection of key-value pairs, where keys are unique. You can create maps in two ways:
|
Map is a mutable collection of key-value pairs, where keys are unique. You can create maps in two ways:
|
||||||
- with the constructor `Map(...)` or `.toMap()` helpers; and
|
- with the constructor `Map(...)` or `.toMap()` helpers; and
|
||||||
- with map literals using braces: `{ "key": value, id: expr, id: }`.
|
- with map literals using braces: `{ "key": value, id: expr, id: }`.
|
||||||
For immutable map values, see [ImmutableMap].
|
|
||||||
|
|
||||||
When constructing from a list, each list item must be a [Collection] with exactly 2 elements, for example, a [List].
|
When constructing from a list, each list item must be a [Collection] with exactly 2 elements, for example, a [List].
|
||||||
|
|
||||||
@ -95,8 +94,7 @@ Or iterate its key-value pairs that are instances of [MapEntry] class:
|
|||||||
|
|
||||||
val map = Map( ["foo", 1], ["bar", "buzz"], [42, "answer"] )
|
val map = Map( ["foo", 1], ["bar", "buzz"], [42, "answer"] )
|
||||||
for( entry in map ) {
|
for( entry in map ) {
|
||||||
val e: MapEntry = entry as MapEntry
|
println("map[%s] = %s"(entry.key, entry.value))
|
||||||
println("map[%s] = %s"(e.key, e.value))
|
|
||||||
}
|
}
|
||||||
void
|
void
|
||||||
>>> map[foo] = 1
|
>>> map[foo] = 1
|
||||||
@ -177,5 +175,4 @@ Notes:
|
|||||||
- Spreads inside map literals and `+`/`+=` merges allow any objects as keys.
|
- Spreads inside map literals and `+`/`+=` merges allow any objects as keys.
|
||||||
- When you need computed or non-string keys, use the constructor form `Map(...)`, map literals with computed keys (if supported), or build entries with `=>` and then merge.
|
- When you need computed or non-string keys, use the constructor form `Map(...)`, map literals with computed keys (if supported), or build entries with `=>` and then merge.
|
||||||
|
|
||||||
[Collection](Collection.md)
|
[Collection](Collection.md)
|
||||||
[ImmutableMap]: ImmutableMap.md
|
|
||||||
192
docs/Matrix.md
192
docs/Matrix.md
@ -1,192 +0,0 @@
|
|||||||
# Matrix (`lyng.matrix`)
|
|
||||||
|
|
||||||
`lyng.matrix` adds dense immutable `Matrix` and `Vector` types for linear algebra.
|
|
||||||
|
|
||||||
Import it when you need matrix or vector arithmetic:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
```
|
|
||||||
|
|
||||||
## Construction
|
|
||||||
|
|
||||||
Create vectors from a flat list and matrices from nested row lists:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val v: Vector = vector([1, 2, 3])
|
|
||||||
val m: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
|
|
||||||
assertEquals([1.0, 2.0, 3.0], v.toList())
|
|
||||||
assertEquals([[1.0, 2.0, 3.0], [4.0, 5.0, 6.0]], m.toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
Factory methods are also available:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val z: Vector = Vector.zeros(3)
|
|
||||||
val i: Matrix = Matrix.identity(3)
|
|
||||||
val m: Matrix = Matrix.zeros(2, 4)
|
|
||||||
```
|
|
||||||
|
|
||||||
All elements are standard double-precision numeric values internally.
|
|
||||||
|
|
||||||
## Shapes
|
|
||||||
|
|
||||||
Matrices may have any rectangular geometry:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
|
|
||||||
assertEquals(2, a.rows)
|
|
||||||
assertEquals(3, a.cols)
|
|
||||||
assertEquals([2, 3], a.shape)
|
|
||||||
assertEquals(false, a.isSquare)
|
|
||||||
```
|
|
||||||
|
|
||||||
Vectors expose:
|
|
||||||
|
|
||||||
- `size`
|
|
||||||
- `length` as an alias of `size`
|
|
||||||
|
|
||||||
## Matrix Operations
|
|
||||||
|
|
||||||
Supported matrix operations:
|
|
||||||
|
|
||||||
- `+` and `-` for element-wise matrix arithmetic
|
|
||||||
- `*` for matrix-matrix product
|
|
||||||
- `*` and `/` by a scalar
|
|
||||||
- `transpose()`
|
|
||||||
- `trace()`
|
|
||||||
- `rank()`
|
|
||||||
- `determinant()`
|
|
||||||
- `inverse()`
|
|
||||||
- `solve(rhs)` for `Vector` or `Matrix` right-hand sides
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
val b: Matrix = matrix([[7, 8], [9, 10], [11, 12]])
|
|
||||||
val product: Matrix = a * b
|
|
||||||
assertEquals([[58.0, 64.0], [139.0, 154.0]], product.toList())
|
|
||||||
assertEquals([[1.0, 4.0], [2.0, 5.0], [3.0, 6.0]], a.transpose().toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
Inverse and solve:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[4, 7], [2, 6]])
|
|
||||||
val rhs: Vector = vector([1, 0])
|
|
||||||
|
|
||||||
val inv: Matrix = a.inverse()
|
|
||||||
val x: Vector = a.solve(rhs)
|
|
||||||
|
|
||||||
assert(abs(a.determinant() - 10.0) < 1e-9)
|
|
||||||
assert(abs(inv.get(0, 0) - 0.6) < 1e-9)
|
|
||||||
assert(abs(x.get(0) - 0.6) < 1e-9)
|
|
||||||
```
|
|
||||||
|
|
||||||
## Vector Operations
|
|
||||||
|
|
||||||
Supported vector operations:
|
|
||||||
|
|
||||||
- `+` and `-`
|
|
||||||
- scalar `*` and `/`
|
|
||||||
- `dot(other)`
|
|
||||||
- `norm()`
|
|
||||||
- `normalize()`
|
|
||||||
- `cross(other)` for 3D vectors
|
|
||||||
- `outer(other)` producing a matrix
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Vector = vector([1, 2, 3])
|
|
||||||
val b: Vector = vector([2, 0, 0])
|
|
||||||
|
|
||||||
assertEquals(2.0, a.dot(b))
|
|
||||||
assertEquals([0.2672612419124244, 0.5345224838248488, 0.8017837257372732], a.normalize().toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
## Indexing and Slicing
|
|
||||||
|
|
||||||
`Matrix` supports both method-style indexing and bracket syntax.
|
|
||||||
|
|
||||||
Scalar access:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val m: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
|
|
||||||
assertEquals(6.0, m.get(1, 2))
|
|
||||||
assertEquals(6.0, m[1, 2])
|
|
||||||
```
|
|
||||||
|
|
||||||
Bracket indexing accepts two selectors: `[row, col]`.
|
|
||||||
Each selector may be either:
|
|
||||||
|
|
||||||
- an `Int`
|
|
||||||
- a `Range`
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val m: Matrix = matrix([[1, 2, 3, 4], [5, 6, 7, 8], [9, 10, 11, 12]])
|
|
||||||
|
|
||||||
assertEquals(7.0, m[1, 2])
|
|
||||||
val columnSlice: Matrix = m[0..2, 2]
|
|
||||||
val topLeft: Matrix = m[0..1, 0..1]
|
|
||||||
val tail: Matrix = m[1.., 1..]
|
|
||||||
assertEquals([[3.0], [7.0], [11.0]], columnSlice.toList())
|
|
||||||
assertEquals([[1.0, 2.0], [5.0, 6.0]], topLeft.toList())
|
|
||||||
assertEquals([[6.0, 7.0, 8.0], [10.0, 11.0, 12.0]], tail.toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
Shape rules:
|
|
||||||
|
|
||||||
- `m[Int, Int]` returns a `Real`
|
|
||||||
- `m[Range, Int]` returns an `Nx1` `Matrix`
|
|
||||||
- `m[Int, Range]` returns a `1xM` `Matrix`
|
|
||||||
- `m[Range, Range]` returns a submatrix
|
|
||||||
|
|
||||||
Open-ended ranges are supported:
|
|
||||||
|
|
||||||
- `m[..1, ..1]`
|
|
||||||
- `m[1.., 1..]`
|
|
||||||
- `m[.., 2]`
|
|
||||||
|
|
||||||
Stepped ranges are not supported in matrix slicing.
|
|
||||||
|
|
||||||
Slices currently return new matrices, not views.
|
|
||||||
|
|
||||||
## Rows and Columns
|
|
||||||
|
|
||||||
If you want plain lists instead of a sliced matrix:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
|
|
||||||
assertEquals([4.0, 5.0, 6.0], a.row(1))
|
|
||||||
assertEquals([2.0, 5.0], a.column(1))
|
|
||||||
```
|
|
||||||
|
|
||||||
## Backend Notes
|
|
||||||
|
|
||||||
The matrix module uses a platform-specific backend where available and falls back to pure Kotlin where needed.
|
|
||||||
|
|
||||||
The public Lyng API stays the same across platforms.
|
|
||||||
245
docs/OOP.md
245
docs/OOP.md
@ -9,7 +9,7 @@ Lyng supports first class OOP constructs, based on classes with multiple inherit
|
|||||||
The class clause looks like
|
The class clause looks like
|
||||||
|
|
||||||
class Point(x,y)
|
class Point(x,y)
|
||||||
assertEquals("Point", Point.className)
|
assert( Point is Class )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
It creates new `Class` with two fields. Here is the more practical sample:
|
It creates new `Class` with two fields. Here is the more practical sample:
|
||||||
@ -376,10 +376,11 @@ Functions defined inside a class body are methods, and unless declared
|
|||||||
`private` are available to be called from outside the class:
|
`private` are available to be called from outside the class:
|
||||||
|
|
||||||
class Point(x,y) {
|
class Point(x,y) {
|
||||||
// private method:
|
|
||||||
private fun d2() { x*x + y*y }
|
|
||||||
// public method declaration:
|
// public method declaration:
|
||||||
fun length() { sqrt(d2()) }
|
fun length() { sqrt(d2()) }
|
||||||
|
|
||||||
|
// private method:
|
||||||
|
private fun d2() {x*x + y*y}
|
||||||
}
|
}
|
||||||
val p = Point(3,4)
|
val p = Point(3,4)
|
||||||
// private called from inside public: OK
|
// private called from inside public: OK
|
||||||
@ -454,43 +455,6 @@ Key rules and features:
|
|||||||
- For arbitrary receivers, use casts: `(expr as Type).member(...)` or `(expr as? Type)?.member(...)`.
|
- For arbitrary receivers, use casts: `(expr as Type).member(...)` or `(expr as? Type)?.member(...)`.
|
||||||
- Qualified access does not relax visibility.
|
- Qualified access does not relax visibility.
|
||||||
|
|
||||||
### Receiver-stack lambdas
|
|
||||||
|
|
||||||
Qualified `this@Type` is also used outside inheritance when a lambda has multiple visible receivers.
|
|
||||||
This is common in DSL-style builders.
|
|
||||||
|
|
||||||
- `A & B` means one receiver value that implements both types.
|
|
||||||
- `context(A, B) C.()->R` means a receiver stack:
|
|
||||||
- primary `this` is `C`
|
|
||||||
- outer/context receivers are `A`, then `B`
|
|
||||||
- Unqualified lookup checks the primary receiver first.
|
|
||||||
- If the primary receiver does not define a member and several outer/context receivers do, Lyng reports a compile-time ambiguity. Use `this@Type` to select one explicitly.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
class Html { fun title() = "html" }
|
|
||||||
class Head { fun title() = "head" }
|
|
||||||
class Body
|
|
||||||
|
|
||||||
val block: context(Html, Head) Body.()->String = {
|
|
||||||
// title() // compile-time ambiguity: Html vs Head
|
|
||||||
this@Html.title()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Context receivers can also constrain extension functions. The extension is visible only when the required receiver is
|
|
||||||
already in the implicit receiver stack:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
class Tag { fun addText(text: String) { /* ... */ } }
|
|
||||||
|
|
||||||
context(Tag)
|
|
||||||
fun String.unaryPlus() {
|
|
||||||
this@Tag.addText(this)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
- Field inheritance (`val`/`var`) and collisions
|
- Field inheritance (`val`/`var`) and collisions
|
||||||
- Instance storage is kept per declaring class, internally disambiguated; unqualified read/write resolves to the first match in the resolution order (leftmost base).
|
- Instance storage is kept per declaring class, internally disambiguated; unqualified read/write resolves to the first match in the resolution order (leftmost base).
|
||||||
- Qualified read/write (via `this@Type` or casts) targets the chosen ancestor’s storage.
|
- Qualified read/write (via `this@Type` or casts) targets the chosen ancestor’s storage.
|
||||||
@ -662,14 +626,10 @@ Unary operators are overloaded by defining methods with no arguments:
|
|||||||
|
|
||||||
| Operator | Method Name |
|
| Operator | Method Name |
|
||||||
| :--- | :--- |
|
| :--- | :--- |
|
||||||
| `+a` | `unaryPlus()` |
|
|
||||||
| `-a` | `negate()` |
|
| `-a` | `negate()` |
|
||||||
| `!a` | `logicalNot()` |
|
| `!a` | `logicalNot()` |
|
||||||
| `~a` | `bitNot()` |
|
| `~a` | `bitNot()` |
|
||||||
|
|
||||||
`unaryPlus()` is useful in DSL-style builders where `+"text"` should append text to
|
|
||||||
the current receiver. See [samples/html_builder_dsl.lyng](samples/html_builder_dsl.lyng).
|
|
||||||
|
|
||||||
### Assignment Operators
|
### Assignment Operators
|
||||||
|
|
||||||
Assignment operators like `+=` first attempt to call a specific assignment method. If that method is not defined, they fall back to a combination of the binary operator and a regular assignment (e.g., `a = a + b`).
|
Assignment operators like `+=` first attempt to call a specific assignment method. If that method is not defined, they fall back to a combination of the binary operator and a regular assignment (e.g., `a = a + b`).
|
||||||
@ -1019,7 +979,7 @@ You can mark a field or a method as static. This is borrowed from Java as more p
|
|||||||
|
|
||||||
static fun exclamation() {
|
static fun exclamation() {
|
||||||
// here foo is a regular var:
|
// here foo is a regular var:
|
||||||
Value.foo.x + "!"
|
foo.x + "!"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assertEquals( Value.foo.x, "foo" )
|
assertEquals( Value.foo.x, "foo" )
|
||||||
@ -1030,164 +990,57 @@ You can mark a field or a method as static. This is borrowed from Java as more p
|
|||||||
assertEquals( "bar!", Value.exclamation() )
|
assertEquals( "bar!", Value.exclamation() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Static fields can be accessed from static methods via the class qualifier:
|
As usual, private statics are not accessible from the outside:
|
||||||
|
|
||||||
class Test {
|
class Test {
|
||||||
static var data = "foo"
|
// private, inacessible from outside protected data:
|
||||||
static fun getData() { Test.data }
|
private static var data = null
|
||||||
|
|
||||||
|
// the interface to access and change it:
|
||||||
|
static fun getData() { data }
|
||||||
|
static fun setData(value) { data = value }
|
||||||
}
|
}
|
||||||
|
|
||||||
assertEquals( "foo", Test.getData() )
|
// no direct access:
|
||||||
Test.data = "bar"
|
assertThrows { Test.data }
|
||||||
assertEquals("bar", Test.getData() )
|
|
||||||
|
// accessible with the interface:
|
||||||
|
assertEquals( null, Test.getData() )
|
||||||
|
Test.setData("fubar")
|
||||||
|
assertEquals("fubar", Test.getData() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
# Extension members
|
# Extending classes
|
||||||
|
|
||||||
Sometimes an existing type or named singleton object is missing some particular functionality that can be _added to it_ without rewriting its inner logic and without using its private state. In this case, _extension members_ can be used.
|
It sometimes happen that the class is missing some particular functionality that can be _added to it_ without rewriting its inner logic and using its private state. In this case _extension members_ could be used.
|
||||||
|
|
||||||
## Extension methods
|
## Extension methods
|
||||||
|
|
||||||
For example, we want to create an extension method that would test if a value can be interpreted as an integer:
|
For example, we want to create an extension method that would test if some object of unknown type contains something that can be interpreted as an integer. In this case we _extend_ class `Object`, as it is the parent class for any instance of any type:
|
||||||
|
|
||||||
fun Int.isInteger() { true }
|
fun Object.isInteger() {
|
||||||
fun Real.isInteger() { this.toInt() == this }
|
when(this) {
|
||||||
fun String.isInteger() { (this.toReal() as Real).isInteger() }
|
// already Int?
|
||||||
|
is Int -> true
|
||||||
|
|
||||||
// Let's test:
|
// real, but with no declimal part?
|
||||||
|
is Real -> toInt() == this
|
||||||
|
|
||||||
|
// string with int or real reuusig code above
|
||||||
|
is String -> toReal().isInteger()
|
||||||
|
|
||||||
|
// otherwise, no:
|
||||||
|
else -> false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Let's test:
|
||||||
assert( 12.isInteger() == true )
|
assert( 12.isInteger() == true )
|
||||||
assert( 12.1.isInteger() == false )
|
assert( 12.1.isInteger() == false )
|
||||||
assert( "5".isInteger() )
|
assert( "5".isInteger() )
|
||||||
assert( ! "5.2".isInteger() )
|
assert( ! "5.2".isInteger() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Extension methods normally act like instance members. If declared as `static`, they are called on the type object itself:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
static fun List<T>.fill(size: Int, block: (Int)->T): List<T> { ... }
|
|
||||||
|
|
||||||
val tens = List.fill(5) { it * 10 }
|
|
||||||
```
|
|
||||||
|
|
||||||
## Extending singleton `object` declarations
|
|
||||||
|
|
||||||
Named singleton objects can also be extension receivers. Use the object name as the receiver type:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
object Config {
|
|
||||||
fun base() = "cfg"
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Config.describe(value) {
|
|
||||||
this.base() + ":" + value.toString()
|
|
||||||
}
|
|
||||||
|
|
||||||
val Config.tag get() = this.base() + ":tag"
|
|
||||||
|
|
||||||
assertEquals("cfg:42", Config.describe(42))
|
|
||||||
assertEquals("cfg:tag", Config.tag)
|
|
||||||
```
|
|
||||||
|
|
||||||
This differs from extending a class in one important way:
|
|
||||||
|
|
||||||
- `fun Point.foo()` adds a member-like extension for all `Point` instances.
|
|
||||||
- `fun Config.foo()` adds a member-like extension only for the single named object `Config`.
|
|
||||||
|
|
||||||
The same rules still apply:
|
|
||||||
|
|
||||||
- Extensions on singleton objects are scope-isolated just like class extensions.
|
|
||||||
- They cannot access the object's `private` members.
|
|
||||||
- Inside the extension body, `this` is the singleton object itself.
|
|
||||||
|
|
||||||
## Extension indexers
|
|
||||||
|
|
||||||
Bracket syntax is just another form of member dispatch. When you write `value[i]` or `value[i] = x`, Lyng lowers it to `getAt(...)` and `putAt(...)`.
|
|
||||||
|
|
||||||
That means indexers can be extended in the same way as methods and properties.
|
|
||||||
|
|
||||||
### Extending indexers on classes
|
|
||||||
|
|
||||||
Use `override fun Type.getAt(...)` and `override fun Type.putAt(...)`:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
class BoxStore {
|
|
||||||
val data = {"name": "alice"}
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun BoxStore.getAt(key: String): Object? {
|
|
||||||
data[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun BoxStore.putAt(key: String, value: Object) {
|
|
||||||
data[key] = value
|
|
||||||
}
|
|
||||||
|
|
||||||
val store = BoxStore()
|
|
||||||
assertEquals("alice", store["name"])
|
|
||||||
store["name"] = "bob"
|
|
||||||
assertEquals("bob", store["name"])
|
|
||||||
```
|
|
||||||
|
|
||||||
As with other extension members, this does not modify the original class declaration. It adds indexer behavior only in the scope where the extension is visible.
|
|
||||||
|
|
||||||
### Extending indexers on singleton `object` declarations
|
|
||||||
|
|
||||||
Named singleton objects work the same way:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
object Storage
|
|
||||||
|
|
||||||
var storageData = {}
|
|
||||||
|
|
||||||
override fun Storage.getAt(key: String): Object? {
|
|
||||||
storageData[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun Storage.putAt(key: String, value: Object) {
|
|
||||||
storageData[key] = value
|
|
||||||
}
|
|
||||||
|
|
||||||
Storage["name"] = "alice"
|
|
||||||
val name: String? = Storage["name"]
|
|
||||||
assertEquals("alice", name)
|
|
||||||
```
|
|
||||||
|
|
||||||
This is the indexer equivalent of `fun Config.foo()`: the extension applies to that single named object, not to all instances of some class.
|
|
||||||
|
|
||||||
### Selector packing
|
|
||||||
|
|
||||||
Index syntax can contain more than one selector:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
value[i]
|
|
||||||
value[i, j]
|
|
||||||
```
|
|
||||||
|
|
||||||
The same packing rules still apply for extension indexers:
|
|
||||||
|
|
||||||
- `value[i]` calls `getAt(i)` or `putAt(i, value)`
|
|
||||||
- `value[i, j]` passes `[i, j]` as one list-like index object
|
|
||||||
- `value[i, j, k]` passes `[i, j, k]`
|
|
||||||
|
|
||||||
So if you want multi-selector indexing, define the receiver to accept that packed index object.
|
|
||||||
|
|
||||||
### About types and generics
|
|
||||||
|
|
||||||
In practice, extension indexers are usually best declared with `Object?` for reads and `Object` for writes:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
override fun Storage.getAt(key: String): Object? { ... }
|
|
||||||
override fun Storage.putAt(key: String, value: Object) { ... }
|
|
||||||
```
|
|
||||||
|
|
||||||
Then use the expected type at the call site:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val name: String? = Storage["name"]
|
|
||||||
```
|
|
||||||
|
|
||||||
Explicit generic arguments do not fit naturally onto `[]` syntax, so typed assignment on read is usually the clearest approach.
|
|
||||||
|
|
||||||
## Extension properties
|
## Extension properties
|
||||||
|
|
||||||
Just like methods, you can extend existing classes with properties. These can be defined using simple initialization (for `val` only) or with custom accessors.
|
Just like methods, you can extend existing classes with properties. These can be defined using simple initialization (for `val` only) or with custom accessors.
|
||||||
@ -1283,7 +1136,7 @@ The same we can provide writable dynamic fields (var-type), adding set method:
|
|||||||
// mutable field
|
// mutable field
|
||||||
"bar" -> storedValueForBar
|
"bar" -> storedValueForBar
|
||||||
|
|
||||||
else -> throw SymbolNotFound()
|
else -> throw SymbolNotFoundException()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
set { name, value ->
|
set { name, value ->
|
||||||
@ -1351,24 +1204,8 @@ collection's sugar won't work with it:
|
|||||||
assertEquals("buzz", x[0])
|
assertEquals("buzz", x[0])
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Multiple selectors are packed into one list index object:
|
If you want dynamic to function like an array, create a [feature
|
||||||
|
request](https://gitea.sergeych.net/SergeychWorks/lyng/issues).
|
||||||
val x = dynamic {
|
|
||||||
get {
|
|
||||||
if( it == [1, 2] ) "hit"
|
|
||||||
else null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assertEquals("hit", x[1, 2])
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
So:
|
|
||||||
|
|
||||||
- `x[i]` passes `i`
|
|
||||||
- `x[i, j]` passes `[i, j]`
|
|
||||||
- `x[i, j, k]` passes `[i, j, k]`
|
|
||||||
|
|
||||||
This is the same rule used by Kotlin-backed `getAt` / `putAt` indexers in embedding.
|
|
||||||
|
|
||||||
# Theory
|
# Theory
|
||||||
|
|
||||||
|
|||||||
@ -1,70 +0,0 @@
|
|||||||
# ObservableList module
|
|
||||||
|
|
||||||
`ObservableList` lives in explicit module `lyng.observable`.
|
|
||||||
|
|
||||||
Import it first:
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Create from a regular mutable list:
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
val xs = [1,2,3].observable()
|
|
||||||
assert(xs is ObservableList<Int>)
|
|
||||||
assertEquals([1,2,3], xs)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Hook flow
|
|
||||||
|
|
||||||
Event order is:
|
|
||||||
1. `beforeChange(change)` listeners
|
|
||||||
2. mutation commit
|
|
||||||
3. `onChange(change)` listeners
|
|
||||||
4. `changes()` flow emission
|
|
||||||
|
|
||||||
Rejection is done by throwing in `beforeChange`.
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
val xs = [1,2].observable()
|
|
||||||
xs.beforeChange {
|
|
||||||
throw ChangeRejectionException("no mutation")
|
|
||||||
}
|
|
||||||
assertThrows(ChangeRejectionException) { xs += 3 }
|
|
||||||
assertEquals([1,2], xs)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Subscriptions
|
|
||||||
|
|
||||||
`beforeChange` and `onChange` return `Subscription`.
|
|
||||||
Call `cancel()` to unsubscribe.
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
val xs = [1].observable()
|
|
||||||
var hits = 0
|
|
||||||
val sub = xs.onChange { hits++ }
|
|
||||||
xs += 2
|
|
||||||
sub.cancel()
|
|
||||||
xs += 3
|
|
||||||
assertEquals(1, hits)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Change events
|
|
||||||
|
|
||||||
`changes()` returns `Flow<ListChange<T>>` with concrete event classes:
|
|
||||||
- `ListInsert`
|
|
||||||
- `ListSet`
|
|
||||||
- `ListRemove`
|
|
||||||
- `ListClear`
|
|
||||||
- `ListReorder`
|
|
||||||
|
|
||||||
import lyng.observable
|
|
||||||
val xs = [10,20].observable()
|
|
||||||
val it = xs.changes().iterator()
|
|
||||||
xs[1] = 200
|
|
||||||
val ev = it.next()
|
|
||||||
assert(ev is ListSet<Int>)
|
|
||||||
assertEquals(20, (ev as ListSet<Int>).oldValue)
|
|
||||||
assertEquals(200, ev.newValue)
|
|
||||||
it.cancelIteration()
|
|
||||||
>>> void
|
|
||||||
@ -1,309 +0,0 @@
|
|||||||
# Operator Interop Registry
|
|
||||||
|
|
||||||
`lyng.operators` provides a runtime registry for mixed-class binary operators.
|
|
||||||
|
|
||||||
Import it when you want expressions such as:
|
|
||||||
|
|
||||||
- `1 + MyType(...)`
|
|
||||||
- `2 < MyType(...)`
|
|
||||||
- `3 == MyType(...)`
|
|
||||||
|
|
||||||
to work without modifying the built-in `Int` or `Real` classes.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.operators
|
|
||||||
```
|
|
||||||
|
|
||||||
## Why This Exists
|
|
||||||
|
|
||||||
If your class defines:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
class Amount(val value: Int) {
|
|
||||||
fun plus(other: Amount) = Amount(value + other.value)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
then:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
Amount(1) + Amount(2)
|
|
||||||
```
|
|
||||||
|
|
||||||
works, because the left operand already knows how to add another `Amount`.
|
|
||||||
|
|
||||||
But:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
1 + Amount(2)
|
|
||||||
```
|
|
||||||
|
|
||||||
does not naturally work, because `Int` has not been rewritten to know about `Amount`.
|
|
||||||
|
|
||||||
The operator interop registry solves exactly that problem.
|
|
||||||
|
|
||||||
## Mental Model
|
|
||||||
|
|
||||||
Registration describes a mixed pair:
|
|
||||||
|
|
||||||
- left class `L`
|
|
||||||
- right class `R`
|
|
||||||
- common class `C`
|
|
||||||
|
|
||||||
When Lyng sees `L op R`, it:
|
|
||||||
|
|
||||||
1. converts `L -> C`
|
|
||||||
2. converts `R -> C`
|
|
||||||
3. evaluates the operator as `C op C`
|
|
||||||
|
|
||||||
So the registry is a bridge, not a separate arithmetic engine.
|
|
||||||
|
|
||||||
## API
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
OperatorInterop.register(
|
|
||||||
leftClass,
|
|
||||||
rightClass,
|
|
||||||
commonClass,
|
|
||||||
operators,
|
|
||||||
leftToCommon,
|
|
||||||
rightToCommon
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
Parameters:
|
|
||||||
|
|
||||||
- `leftClass`: original left operand class
|
|
||||||
- `rightClass`: original right operand class
|
|
||||||
- `commonClass`: class that will actually execute the operator methods
|
|
||||||
- `operators`: list of operators enabled for this pair
|
|
||||||
- `leftToCommon`: conversion from left operand to common class
|
|
||||||
- `rightToCommon`: conversion from right operand to common class
|
|
||||||
|
|
||||||
## Supported Operators
|
|
||||||
|
|
||||||
`BinaryOperator` values:
|
|
||||||
|
|
||||||
- `Plus`
|
|
||||||
- `Minus`
|
|
||||||
- `Mul`
|
|
||||||
- `Div`
|
|
||||||
- `Mod`
|
|
||||||
- `Compare`
|
|
||||||
- `Equals`
|
|
||||||
|
|
||||||
Meaning:
|
|
||||||
|
|
||||||
- `Compare` enables `<`, `<=`, `>`, `>=`, and `<=>`
|
|
||||||
- `Equals` enables `==` and `!=`
|
|
||||||
|
|
||||||
## Minimal Working Example
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
package test.decimalbox
|
|
||||||
import lyng.operators
|
|
||||||
|
|
||||||
class DecimalBox(val value: Int) {
|
|
||||||
fun plus(other: DecimalBox) = DecimalBox(value + other.value)
|
|
||||||
fun minus(other: DecimalBox) = DecimalBox(value - other.value)
|
|
||||||
fun mul(other: DecimalBox) = DecimalBox(value * other.value)
|
|
||||||
fun div(other: DecimalBox) = DecimalBox(value / other.value)
|
|
||||||
fun mod(other: DecimalBox) = DecimalBox(value % other.value)
|
|
||||||
fun compareTo(other: DecimalBox) = value <=> other.value
|
|
||||||
}
|
|
||||||
|
|
||||||
OperatorInterop.register(
|
|
||||||
Int,
|
|
||||||
DecimalBox,
|
|
||||||
DecimalBox,
|
|
||||||
[
|
|
||||||
BinaryOperator.Plus,
|
|
||||||
BinaryOperator.Minus,
|
|
||||||
BinaryOperator.Mul,
|
|
||||||
BinaryOperator.Div,
|
|
||||||
BinaryOperator.Mod,
|
|
||||||
BinaryOperator.Compare,
|
|
||||||
BinaryOperator.Equals
|
|
||||||
],
|
|
||||||
{ x: Int -> DecimalBox(x) },
|
|
||||||
{ x: DecimalBox -> x }
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
Then:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import test.decimalbox
|
|
||||||
|
|
||||||
assertEquals(DecimalBox(3), 1 + DecimalBox(2))
|
|
||||||
assertEquals(DecimalBox(1), 3 - DecimalBox(2))
|
|
||||||
assertEquals(DecimalBox(8), 4 * DecimalBox(2))
|
|
||||||
assertEquals(DecimalBox(4), 8 / DecimalBox(2))
|
|
||||||
assertEquals(DecimalBox(1), 7 % DecimalBox(2))
|
|
||||||
assert(1 < DecimalBox(2))
|
|
||||||
assert(2 <= DecimalBox(2))
|
|
||||||
assert(3 > DecimalBox(2))
|
|
||||||
assert(2 == DecimalBox(2))
|
|
||||||
assert(2 != DecimalBox(3))
|
|
||||||
```
|
|
||||||
|
|
||||||
## How Decimal Uses It
|
|
||||||
|
|
||||||
`lyng.decimal` uses this same mechanism so that:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
1 + 2.d
|
|
||||||
0.5 + 1.d
|
|
||||||
2 == 2.d
|
|
||||||
3 > 2.d
|
|
||||||
```
|
|
||||||
|
|
||||||
work naturally even though `Int` and `Real` themselves were not edited to know `Decimal`.
|
|
||||||
|
|
||||||
The shape is:
|
|
||||||
|
|
||||||
- `leftClass = Int` or `Real`
|
|
||||||
- `rightClass = Decimal`
|
|
||||||
- `commonClass = Decimal`
|
|
||||||
- convert built-ins into `Decimal`
|
|
||||||
- leave `Decimal` values unchanged
|
|
||||||
|
|
||||||
## Step-By-Step Pattern For Your Own Type
|
|
||||||
|
|
||||||
### 1. Pick the common class
|
|
||||||
|
|
||||||
Choose one class that will be the actual arithmetic domain.
|
|
||||||
|
|
||||||
For numeric-like types, that is usually your own class:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
class Rational(...)
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2. Implement operators on that class
|
|
||||||
|
|
||||||
The common class should define the operations you plan to register.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
class Rational(val num: Int, val den: Int) {
|
|
||||||
fun plus(other: Rational) = Rational(num * other.den + other.num * den, den * other.den)
|
|
||||||
fun minus(other: Rational) = Rational(num * other.den - other.num * den, den * other.den)
|
|
||||||
fun mul(other: Rational) = Rational(num * other.num, den * other.den)
|
|
||||||
fun div(other: Rational) = Rational(num * other.den, den * other.num)
|
|
||||||
fun compareTo(other: Rational) = (num * other.den) <=> (other.num * den)
|
|
||||||
|
|
||||||
static fun fromInt(value: Int) = Rational(value, 1)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### 3. Register the mixed pair
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.operators
|
|
||||||
|
|
||||||
OperatorInterop.register(
|
|
||||||
Int,
|
|
||||||
Rational,
|
|
||||||
Rational,
|
|
||||||
[
|
|
||||||
BinaryOperator.Plus,
|
|
||||||
BinaryOperator.Minus,
|
|
||||||
BinaryOperator.Mul,
|
|
||||||
BinaryOperator.Div,
|
|
||||||
BinaryOperator.Compare,
|
|
||||||
BinaryOperator.Equals
|
|
||||||
],
|
|
||||||
{ x: Int -> Rational.fromInt(x) },
|
|
||||||
{ x: Rational -> x }
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
### 4. Use it
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
assertEquals(Rational(3, 2), 1 + Rational(1, 2))
|
|
||||||
assert(Rational(3, 2) == Rational(3, 2))
|
|
||||||
assert(2 > Rational(3, 2))
|
|
||||||
```
|
|
||||||
|
|
||||||
## Registering More Than One Built-in Type
|
|
||||||
|
|
||||||
If you want both `Int + MyType` and `Real + MyType`, register both pairs explicitly:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
OperatorInterop.register(
|
|
||||||
Int,
|
|
||||||
MyType,
|
|
||||||
MyType,
|
|
||||||
[BinaryOperator.Plus, BinaryOperator.Compare, BinaryOperator.Equals],
|
|
||||||
{ x: Int -> MyType.fromInt(x) },
|
|
||||||
{ x: MyType -> x }
|
|
||||||
)
|
|
||||||
|
|
||||||
OperatorInterop.register(
|
|
||||||
Real,
|
|
||||||
MyType,
|
|
||||||
MyType,
|
|
||||||
[BinaryOperator.Plus, BinaryOperator.Compare, BinaryOperator.Equals],
|
|
||||||
{ x: Real -> MyType.fromReal(x) },
|
|
||||||
{ x: MyType -> x }
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
Each mixed pair is independent.
|
|
||||||
|
|
||||||
## Pure Lyng Registration
|
|
||||||
|
|
||||||
This mechanism is intentionally useful from pure Lyng code, not only from Kotlin-backed modules.
|
|
||||||
|
|
||||||
That means you can:
|
|
||||||
|
|
||||||
- declare a class in Lyng
|
|
||||||
- define its operators in Lyng
|
|
||||||
- register mixed operand bridges in Lyng
|
|
||||||
|
|
||||||
without touching compiler internals.
|
|
||||||
|
|
||||||
## Where To Register
|
|
||||||
|
|
||||||
Register once during module initialization.
|
|
||||||
|
|
||||||
Top-level module code is a good place:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
package my.rational
|
|
||||||
import lyng.operators
|
|
||||||
|
|
||||||
class Rational(...)
|
|
||||||
|
|
||||||
OperatorInterop.register(...)
|
|
||||||
```
|
|
||||||
|
|
||||||
That keeps registration close to the type declaration and makes importing the module enough to activate the interop.
|
|
||||||
|
|
||||||
## What Registration Does Not Do
|
|
||||||
|
|
||||||
The registry does not:
|
|
||||||
|
|
||||||
- invent operators your common class does not implement
|
|
||||||
- change the original `Int`, `Real`, or other built-ins
|
|
||||||
- automatically cover every class pair
|
|
||||||
- replace normal method overload resolution when the left-hand class already knows what to do
|
|
||||||
|
|
||||||
It only teaches Lyng how to bridge a specific mixed pair into a common class for the listed operators.
|
|
||||||
|
|
||||||
## Recommended Design Rules
|
|
||||||
|
|
||||||
If you want interop to feel natural:
|
|
||||||
|
|
||||||
- choose one obvious common class
|
|
||||||
- make conversions explicit and unsurprising
|
|
||||||
- implement `compareTo` if you want ordering operators
|
|
||||||
- register `Equals` whenever mixed equality should work
|
|
||||||
- keep the registered operator list minimal and accurate
|
|
||||||
|
|
||||||
For decimal-like semantics, also read [Decimal.md](Decimal.md).
|
|
||||||
@ -25,23 +25,6 @@ Exclusive end ranges are adopted from kotlin either:
|
|||||||
assert(4 in r)
|
assert(4 in r)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Descending finite ranges are explicit too:
|
|
||||||
|
|
||||||
val r = 5 downTo 1
|
|
||||||
assert(r.isDescending)
|
|
||||||
assert(r.toList() == [5,4,3,2,1])
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Use `downUntil` when the lower bound should be excluded:
|
|
||||||
|
|
||||||
val r = 5 downUntil 1
|
|
||||||
assert(r.toList() == [5,4,3,2])
|
|
||||||
assert(1 !in r)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
This is explicit by design: `5..1` is not treated as a reverse range. It is an
|
|
||||||
ordinary ascending range with no values in it when iterated.
|
|
||||||
|
|
||||||
In any case, we can test an object to belong to using `in` and `!in` and
|
In any case, we can test an object to belong to using `in` and `!in` and
|
||||||
access limits:
|
access limits:
|
||||||
|
|
||||||
@ -90,23 +73,6 @@ but
|
|||||||
>>> 2
|
>>> 2
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Descending ranges work in `for` loops exactly the same way:
|
|
||||||
|
|
||||||
for( i in 3 downTo 1 )
|
|
||||||
println(i)
|
|
||||||
>>> 3
|
|
||||||
>>> 2
|
|
||||||
>>> 1
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
And with an exclusive lower bound:
|
|
||||||
|
|
||||||
for( i in 3 downUntil 1 )
|
|
||||||
println(i)
|
|
||||||
>>> 3
|
|
||||||
>>> 2
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
### Stepped ranges
|
### Stepped ranges
|
||||||
|
|
||||||
Use `step` to change the iteration increment. The range bounds still define membership,
|
Use `step` to change the iteration increment. The range bounds still define membership,
|
||||||
@ -114,18 +80,9 @@ so iteration ends when the next value is no longer in the range.
|
|||||||
|
|
||||||
assert( [1,3,5] == (1..5 step 2).toList() )
|
assert( [1,3,5] == (1..5 step 2).toList() )
|
||||||
assert( [1,3] == (1..<5 step 2).toList() )
|
assert( [1,3] == (1..<5 step 2).toList() )
|
||||||
assert( [5,3,1] == (5 downTo 1 step 2).toList() )
|
|
||||||
assert( ['a','c','e'] == ('a'..'e' step 2).toList() )
|
assert( ['a','c','e'] == ('a'..'e' step 2).toList() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Descending ranges still use a positive `step`; the direction comes from
|
|
||||||
`downTo` / `downUntil`:
|
|
||||||
|
|
||||||
assert( ['e','c','a'] == ('e' downTo 'a' step 2).toList() )
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
A negative step with `downTo` / `downUntil` is invalid.
|
|
||||||
|
|
||||||
Real ranges require an explicit step:
|
Real ranges require an explicit step:
|
||||||
|
|
||||||
assert( [0,0.25,0.5,0.75,1.0] == (0.0..1.0 step 0.25).toList() )
|
assert( [0,0.25,0.5,0.75,1.0] == (0.0..1.0 step 0.25).toList() )
|
||||||
@ -140,7 +97,7 @@ Open-ended ranges require an explicit step to iterate:
|
|||||||
|
|
||||||
You can use Char as both ends of the closed range:
|
You can use Char as both ends of the closed range:
|
||||||
|
|
||||||
val r = 'a'..'c'
|
val r = 'a' .. 'c'
|
||||||
assert( 'b' in r)
|
assert( 'b' in r)
|
||||||
assert( 'e' !in r)
|
assert( 'e' !in r)
|
||||||
for( ch in r )
|
for( ch in r )
|
||||||
@ -162,7 +119,6 @@ Exclusive end char ranges are supported too:
|
|||||||
|-----------------|------------------------------|---------------|
|
|-----------------|------------------------------|---------------|
|
||||||
| contains(other) | used in `in` | Range, or Any |
|
| contains(other) | used in `in` | Range, or Any |
|
||||||
| isEndInclusive | true for '..' | Bool |
|
| isEndInclusive | true for '..' | Bool |
|
||||||
| isDescending | true for `downTo`/`downUntil`| Bool |
|
|
||||||
| isOpen | at any end | Bool |
|
| isOpen | at any end | Bool |
|
||||||
| isIntRange | both start and end are Int | Bool |
|
| isIntRange | both start and end are Int | Bool |
|
||||||
| step | explicit iteration step | Any? |
|
| step | explicit iteration step | Any? |
|
||||||
|
|||||||
@ -19,8 +19,6 @@ you can use it's class to ensure type:
|
|||||||
|-----------------|-------------------------------------------------------------|------|
|
|-----------------|-------------------------------------------------------------|------|
|
||||||
| `.roundToInt()` | round to nearest int like round(x) | Int |
|
| `.roundToInt()` | round to nearest int like round(x) | Int |
|
||||||
| `.toInt()` | convert integer part of real to `Int` dropping decimal part | Int |
|
| `.toInt()` | convert integer part of real to `Int` dropping decimal part | Int |
|
||||||
| `.isInfinite()` | true when the value is `Infinity` or `-Infinity` | Bool |
|
|
||||||
| `.isNaN()` | true when the value is `NaN` | Bool |
|
|
||||||
| `.clamp(range)` | clamp value within range boundaries | Real |
|
| `.clamp(range)` | clamp value within range boundaries | Real |
|
||||||
| | | |
|
| | | |
|
||||||
| | | |
|
| | | |
|
||||||
|
|||||||
@ -24,14 +24,13 @@ counterpart, _not match_ operator `!~`:
|
|||||||
|
|
||||||
When you need to find groups, and more detailed match information, use `Regex.find`:
|
When you need to find groups, and more detailed match information, use `Regex.find`:
|
||||||
|
|
||||||
val result: RegexMatch? = Regex("abc(\d)(\d)(\d)").find( "bad456 good abc123")
|
val result = Regex("abc(\d)(\d)(\d)").find( "bad456 good abc123")
|
||||||
assert( result != null )
|
assert( result != null )
|
||||||
val match: RegexMatch = result as RegexMatch
|
assertEquals( 12 .. 17, result.range )
|
||||||
assertEquals( 12 ..< 17, match.range )
|
assertEquals( "abc123", result[0] )
|
||||||
assertEquals( "abc123", match[0] )
|
assertEquals( "1", result[1] )
|
||||||
assertEquals( "1", match[1] )
|
assertEquals( "2", result[2] )
|
||||||
assertEquals( "2", match[2] )
|
assertEquals( "3", result[3] )
|
||||||
assertEquals( "3", match[3] )
|
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Note that the object `RegexMatch`, returned by [Regex.find], behaves much like in many other languages: it provides the
|
Note that the object `RegexMatch`, returned by [Regex.find], behaves much like in many other languages: it provides the
|
||||||
@ -40,12 +39,11 @@ index range and groups matches as indexes.
|
|||||||
Match operator actually also provides `RegexMatch` in `$~` reserved variable (borrowed from Ruby too):
|
Match operator actually also provides `RegexMatch` in `$~` reserved variable (borrowed from Ruby too):
|
||||||
|
|
||||||
assert( "bad456 good abc123" =~ "abc(\d)(\d)(\d)".re )
|
assert( "bad456 good abc123" =~ "abc(\d)(\d)(\d)".re )
|
||||||
val match2: RegexMatch = $~ as RegexMatch
|
assertEquals( 12 .. 17, $~.range )
|
||||||
assertEquals( 12 ..< 17, match2.range )
|
assertEquals( "abc123", $~[0] )
|
||||||
assertEquals( "abc123", match2[0] )
|
assertEquals( "1", $~[1] )
|
||||||
assertEquals( "1", match2[1] )
|
assertEquals( "2", $~[2] )
|
||||||
assertEquals( "2", match2[2] )
|
assertEquals( "3", $~[3] )
|
||||||
assertEquals( "3", match2[3] )
|
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
This is often more readable than calling `find`.
|
This is often more readable than calling `find`.
|
||||||
@ -61,19 +59,7 @@ string can be either left or right operator, but not both:
|
|||||||
|
|
||||||
Also, string indexing is Regex-aware, and works like `Regex.find` (_not findall!_):
|
Also, string indexing is Regex-aware, and works like `Regex.find` (_not findall!_):
|
||||||
|
|
||||||
assert( "cd" == ("abcdef"[ "c.".re ] as RegexMatch).value )
|
assert( "cd" == "abcdef"[ "c.".re ].value )
|
||||||
>>> void
|
|
||||||
|
|
||||||
Regex replacement is exposed on `String.replace` and `String.replaceFirst`:
|
|
||||||
|
|
||||||
assertEquals( "v#.#.#", "v1.2.3".replace( "\d+".re, "#" ) )
|
|
||||||
assertEquals( "v[1].[2].[3]", "v1.2.3".replace( "(\d+)".re ) { m -> "[" + m[1] + "]" } )
|
|
||||||
assertEquals( "year-04-08", "2026-04-08".replaceFirst( "\d+".re, "year" ) )
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
When `replace` takes a plain `String`, it is treated literally, not as a regex pattern:
|
|
||||||
|
|
||||||
assertEquals( "a-b-c", "a.b.c".replace( ".", "-" ) )
|
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
|
|
||||||
@ -102,3 +88,4 @@ When `replace` takes a plain `String`, it is treated literally, not as a regex p
|
|||||||
[List]: List.md
|
[List]: List.md
|
||||||
|
|
||||||
[Range]: Range.md
|
[Range]: Range.md
|
||||||
|
|
||||||
|
|||||||
10
docs/Set.md
10
docs/Set.md
@ -1,8 +1,7 @@
|
|||||||
# Set built-in class
|
# List built-in class
|
||||||
|
|
||||||
Mutable set of any objects: a group of different objects, no repetitions.
|
Mutable set of any objects: a group of different objects, no repetitions.
|
||||||
Sets are not ordered, order of appearance does not matter.
|
Sets are not ordered, order of appearance does not matter.
|
||||||
For immutable set values, see [ImmutableSet].
|
|
||||||
|
|
||||||
val set = Set(1,2,3, "foo")
|
val set = Set(1,2,3, "foo")
|
||||||
assert( 1 in set )
|
assert( 1 in set )
|
||||||
@ -27,8 +26,8 @@ no indexing. Use [set.toList] as needed.
|
|||||||
|
|
||||||
// intersection
|
// intersection
|
||||||
assertEquals( Set(1,4), Set(3, 1, 4).intersect(Set(2, 4, 1)) )
|
assertEquals( Set(1,4), Set(3, 1, 4).intersect(Set(2, 4, 1)) )
|
||||||
// or simple (intersection)
|
// or simple
|
||||||
assertEquals( Set(1,4), Set(3, 1, 4).intersect(Set(2, 4, 1)) )
|
assertEquals( Set(1,4), Set(3, 1, 4) * Set(2, 4, 1) )
|
||||||
|
|
||||||
// To find collection elements not present in another collection, use the
|
// To find collection elements not present in another collection, use the
|
||||||
// subtract() or `-`:
|
// subtract() or `-`:
|
||||||
@ -92,5 +91,4 @@ Sets are only equal when contains exactly same elements, order, as was said, is
|
|||||||
Also, it inherits methods from [Iterable].
|
Also, it inherits methods from [Iterable].
|
||||||
|
|
||||||
|
|
||||||
[Range]: Range.md
|
[Range]: Range.md
|
||||||
[ImmutableSet]: ImmutableSet.md
|
|
||||||
@ -154,10 +154,9 @@ Function annotation can have more args specified at call time. There arguments m
|
|||||||
@Registered("bar")
|
@Registered("bar")
|
||||||
fun foo2() { "called foo2" }
|
fun foo2() { "called foo2" }
|
||||||
|
|
||||||
val fooFn: Callable = registered["foo"] as Callable
|
assertEquals(registered["foo"](), "called foo")
|
||||||
val barFn: Callable = registered["bar"] as Callable
|
assertEquals(registered["bar"](), "called foo2")
|
||||||
assertEquals(fooFn(), "called foo")
|
>>> void
|
||||||
assertEquals(barFn(), "called foo2")
|
|
||||||
|
|
||||||
[parallelism]: parallelism.md
|
[parallelism]: parallelism.md
|
||||||
|
|
||||||
|
|||||||
@ -1,252 +0,0 @@
|
|||||||
# Lyng Language Reference for AI Agents (Current Compiler State)
|
|
||||||
|
|
||||||
[//]: # (excludeFromIndex)
|
|
||||||
|
|
||||||
Purpose: dense, implementation-first reference for generating valid Lyng code.
|
|
||||||
|
|
||||||
Primary sources used: `lynglib/src/commonMain/kotlin/net/sergeych/lyng/{Parser,Token,Compiler,Script,TypeDecl}.kt`, `lynglib/stdlib/lyng/root.lyng`, tests in `lynglib/src/commonTest` and `lynglib/src/jvmTest`.
|
|
||||||
|
|
||||||
## 1. Ground Rules
|
|
||||||
- Resolution is compile-time-first. Avoid runtime name/member lookup assumptions.
|
|
||||||
- `lyng.stdlib` is auto-seeded for normal scripts (default import manager).
|
|
||||||
- Use explicit casts when receiver type is unknown (`Object`/`Obj`).
|
|
||||||
- Prefer modern null-safe operators (`?.`, `?:`/`??`, `?=`, `as?`, `!!`).
|
|
||||||
- Do not rely on fallback opcodes or dynamic member fallback semantics.
|
|
||||||
|
|
||||||
## 2. Lexical Syntax
|
|
||||||
- Comments: `// line`, `/* block */`.
|
|
||||||
- Strings: `"..."` or `` `...` `` (supports escapes). Multiline string content is normalized by indentation logic.
|
|
||||||
- AI generation preference: use `"..."` by default, including multiline strings; `"` strings are also multiline-capable and should be preferred for ordinary code/doc/SQL text. Use backtick strings mainly when the content contains many double quotes and backticks would make the source clearer.
|
|
||||||
- Shared escapes: `\n`, `\r`, `\t`, `\\`, `\uXXXX` (4 hex digits).
|
|
||||||
- Delimiter escapes: `\"` inside `"..."`, ``\` `` inside `` `...` ``.
|
|
||||||
- Unicode escapes use exactly 4 hex digits (for example: `"\u0416"` -> `Ж`).
|
|
||||||
- Unknown `\x` escapes in strings are preserved literally as two characters (`\` and `x`).
|
|
||||||
- String interpolation is supported:
|
|
||||||
- identifier form: `"$name"` or `` `$name` ``
|
|
||||||
- expression form: `"${expr}"` or `` `${expr}` ``
|
|
||||||
- escaped dollar: `"\$"`, `"$$"`, `` `\$` ``, and `` `$$` `` all produce literal `$`.
|
|
||||||
- `\\$x` means backslash + interpolated `x` in either delimiter form.
|
|
||||||
- Per-file opt-out is supported via leading comment directive:
|
|
||||||
- `// feature: interpolation: off`
|
|
||||||
- with this directive, `$...` stays literal text.
|
|
||||||
- Numbers: `Int` (`123`, `1_000`), `Real` (`1.2`, `1e3`), hex (`0xFF`).
|
|
||||||
- Char: `'a'`, escaped chars supported.
|
|
||||||
- Supported escapes: `\n`, `\r`, `\t`, `\'`, `\\`, `\uXXXX` (4 hex digits).
|
|
||||||
- Backslash character in a char literal must be written as `'\\'` (forms like `'\'` are invalid).
|
|
||||||
- Labels:
|
|
||||||
- statement label: `loop@ for (...) { ... }`
|
|
||||||
- label reference: `break@loop`, `continue@loop`, `return@fnLabel`
|
|
||||||
- Keywords/tokens include (contextual in many places):
|
|
||||||
- declarations: `fun`/`fn`, `val`, `var`, `class`, `object`, `interface`, `enum`, `type`, `init`
|
|
||||||
- modifiers: `private`, `protected`, `static`, `abstract`, `closed`, `override`, `extern`, `open`
|
|
||||||
- flow: `if`, `else`, `when`, `for`, `while`, `do`, `try`, `catch`, `finally`, `throw`, `return`, `break`, `continue`
|
|
||||||
|
|
||||||
## 3. Literals and Core Expressions
|
|
||||||
- Scalars: `null`, `true`, `false`, `void`.
|
|
||||||
- List literal: `[a, b, c]`, spreads with `...`.
|
|
||||||
- Spread positions: beginning, middle, end are all valid: `[...a]`, `[0, ...a, 4]`, `[head, ...mid, tail]`.
|
|
||||||
- Spread source must be a `List` at runtime (non-list spread raises an error).
|
|
||||||
- Map literal: `{ key: value, x:, ...otherMap }`.
|
|
||||||
- `x:` means shorthand `x: x`.
|
|
||||||
- Map spread source must be a `Map`.
|
|
||||||
- Range literals:
|
|
||||||
- inclusive: `a..b`
|
|
||||||
- exclusive end: `a..<b`
|
|
||||||
- descending inclusive: `a downTo b`
|
|
||||||
- descending exclusive end: `a downUntil b`
|
|
||||||
- open-ended forms are supported (`a..`, `..b`, `..`).
|
|
||||||
- optional step: `a..b step 2`, `a downTo b step 2`
|
|
||||||
- Lambda literal:
|
|
||||||
- with params: `{ x, y -> x + y }`
|
|
||||||
- implicit `it`: `{ it + 1 }`
|
|
||||||
- Ternary conditional is supported: `cond ? thenExpr : elseExpr`.
|
|
||||||
|
|
||||||
## 3.1 Splats in Calls and Lambdas
|
|
||||||
- Declaration-side variadic parameters use ellipsis suffix:
|
|
||||||
- functions: `fun f(head, tail...) { ... }`
|
|
||||||
- lambdas: `{ x, rest... -> ... }`
|
|
||||||
- Call-side splats use `...expr` and are expanded by argument kind:
|
|
||||||
- positional splat: `f(...[1,2,3])`
|
|
||||||
- named splat: `f(...{ a: 1, b: 2 })` (map-style)
|
|
||||||
- Runtime acceptance for splats:
|
|
||||||
- positional splat accepts `List` and general `Iterable` (iterable is converted to list first).
|
|
||||||
- named splat accepts `Map` with string keys only.
|
|
||||||
- Ordering/validation rules (enforced):
|
|
||||||
- positional argument cannot follow named arguments (except trailing-block parsing case).
|
|
||||||
- positional splat cannot follow named arguments.
|
|
||||||
- duplicate named arguments are errors (including duplicates introduced via named splat).
|
|
||||||
- unknown named parameters are errors.
|
|
||||||
- variadic parameter itself cannot be passed as a named argument (`fun g(args..., tail)` then `g(args: ...)` is invalid).
|
|
||||||
- Trailing block + named arguments:
|
|
||||||
- if the last callable parameter is already provided by name in parentheses, adding a trailing block is invalid.
|
|
||||||
|
|
||||||
## 4. Operators (implemented)
|
|
||||||
- Assignment: `=`, `+=`, `-=`, `*=`, `/=`, `%=`, `?=`.
|
|
||||||
- Logical: `||`, `&&`, unary `!`.
|
|
||||||
- Unary arithmetic/bitwise: unary `+`, unary `-`, `~`.
|
|
||||||
- Bitwise: `|`, `^`, `&`, `~`, shifts `<<`, `>>`.
|
|
||||||
- Equality/comparison: `==`, `!=`, `===`, `!==`, `<`, `<=`, `>`, `>=`, `<=>`, `=~`, `!~`.
|
|
||||||
- Type/containment: `is`, `!is`, `in`, `!in`, `as`, `as?`.
|
|
||||||
- Null-safe family:
|
|
||||||
- member access: `?.`
|
|
||||||
- safe index: `?[i]`, `?[i, j]`
|
|
||||||
- safe invoke: `?(...)`
|
|
||||||
- safe block invoke: `?{ ... }`
|
|
||||||
- elvis: `?:` and `??`.
|
|
||||||
- Increment/decrement: prefix and postfix `++`, `--`.
|
|
||||||
- Indexing syntax:
|
|
||||||
- single selector: `a[i]`
|
|
||||||
- multiple selectors: `a[i, j, k]`
|
|
||||||
- language-level indexing with multiple selectors is passed to `getAt`/`putAt` as one list-like index object, not as multiple method arguments.
|
|
||||||
- indexers can also be supplied by extension members, including named singleton `object` receivers via `override fun Storage.getAt(...)` / `putAt(...)`.
|
|
||||||
- example: `m[0..2, 2]`.
|
|
||||||
|
|
||||||
## 5. Declarations
|
|
||||||
- Variables:
|
|
||||||
- `val` immutable, `var` mutable.
|
|
||||||
- top-level/local `val` must be initialized.
|
|
||||||
- class `val` may be late-initialized, but must be assigned in class body/init before class parse ends.
|
|
||||||
- destructuring declaration: `val [a, b, rest...] = expr`.
|
|
||||||
- destructuring declaration details:
|
|
||||||
- allowed in `val` and `var` declarations.
|
|
||||||
- supports nested patterns: `val [a, [b, c...], d] = rhs`.
|
|
||||||
- supports at most one splat (`...`) per pattern level.
|
|
||||||
- RHS must be a `List`.
|
|
||||||
- without splat: RHS must have at least as many elements as pattern arity.
|
|
||||||
- with splat: head/tail elements are bound directly, splat receives a `List`.
|
|
||||||
- Functions:
|
|
||||||
- `fun` and `fn` are equivalent.
|
|
||||||
- full body: `fun f(x) { ... }`
|
|
||||||
- shorthand: `fun f(x) = expr`.
|
|
||||||
- generics: `fun f<T>(x: T): T`.
|
|
||||||
- extension functions: `fun Type.name(...) { ... }`.
|
|
||||||
- context-aware extension functions: `context(Tag) fun String.unaryPlus() { this@Tag.addText(this) }`.
|
|
||||||
- named singleton `object` declarations can be extension receivers too: `fun Config.describe(...) { ... }`, `val Config.tag get() = ...`.
|
|
||||||
- static extension functions are callable on the type object: `static fun List<T>.fill(...)` -> `List.fill(...)`.
|
|
||||||
- delegated callable: `fun f(...) by delegate`.
|
|
||||||
- Type aliases:
|
|
||||||
- `type Name = TypeExpr`
|
|
||||||
- generic: `type Box<T> = List<T>`
|
|
||||||
- aliases are expanded structurally.
|
|
||||||
- Classes/objects/enums/interfaces:
|
|
||||||
- `interface` is parsed as abstract class synonym.
|
|
||||||
- `object` supports named singleton and anonymous object expression forms.
|
|
||||||
- enums support lifted entries: `enum E* { A, B }`.
|
|
||||||
- multiple inheritance is supported; override is enforced when overriding base members.
|
|
||||||
- Properties/accessors in class body:
|
|
||||||
- accessor form supports `get`/`set`, including `private set`/`protected set`.
|
|
||||||
|
|
||||||
## 6. Control Flow
|
|
||||||
- `if` is expression-like.
|
|
||||||
- `compile if (cond) { ... } else { ... }` is a compile-time-only conditional.
|
|
||||||
- current condition grammar is restricted to `defined(NameOr.Package)`, `!`, `&&`, `||`, and parentheses.
|
|
||||||
- the untaken branch is skipped by the compiler and is not name-resolved or type-checked.
|
|
||||||
- `when(value) { ... }` supported.
|
|
||||||
- branch conditions support equality, `in`, `!in`, `is`, `!is`, and `nullable` predicate.
|
|
||||||
- `when { ... }` (subject-less) is currently not implemented.
|
|
||||||
- Loops: `for`, `while`, `do ... while`.
|
|
||||||
- loop `else` blocks are supported.
|
|
||||||
- `break value` can return a loop result.
|
|
||||||
- Exceptions: `try/catch/finally`, `throw`.
|
|
||||||
|
|
||||||
## 6.1 Destructuring Assignment (implemented)
|
|
||||||
- Reassignment form is supported (not only declaration):
|
|
||||||
- `[x, y] = [y, x]`
|
|
||||||
- Semantics match destructuring declaration:
|
|
||||||
- nested patterns allowed.
|
|
||||||
- at most one splat per pattern level.
|
|
||||||
- RHS must be a `List`.
|
|
||||||
- too few RHS elements raises runtime error.
|
|
||||||
- Targets in pattern are variables parsed from identifier patterns.
|
|
||||||
|
|
||||||
## 7. Type System (current behavior)
|
|
||||||
- Non-null by default (`T`), nullable with `T?`.
|
|
||||||
- `as` (checked cast), `as?` (safe cast returning `null`), `!!` non-null assertion.
|
|
||||||
- Type expressions support:
|
|
||||||
- unions `A | B`
|
|
||||||
- intersections `A & B`
|
|
||||||
- function types `(A, B)->R` and receiver form `Receiver.(A)->R`
|
|
||||||
- receiver-stack function types via `context(A, B) Receiver.(P)->R`
|
|
||||||
- variadics in function type via ellipsis (`T...`)
|
|
||||||
- `A & B` means one value implementing both types.
|
|
||||||
- `context(A, B) Receiver.(P)->R` is different: it declares an ordered implicit-receiver stack where `Receiver` is primary `this`, then `A`, then `B`.
|
|
||||||
- Nested receiver lambdas keep outer receivers in scope; unqualified lookup prefers the innermost receiver, and `this@Type` can select an outer/context receiver explicitly.
|
|
||||||
- If the primary receiver does not provide a member and multiple outer/context receivers do, the lookup is a compile-time ambiguity and must be disambiguated with `this@Type`.
|
|
||||||
- Generics:
|
|
||||||
- type params on classes/functions/type aliases
|
|
||||||
- bounds via `:` with union/intersection expressions
|
|
||||||
- declaration-site variance via `in` / `out`
|
|
||||||
- Generic function/class/type syntax examples:
|
|
||||||
- function: `fun choose<T>(a: T, b: T): T = a`
|
|
||||||
- class: `class Box<T>(val value: T)`
|
|
||||||
- alias: `type PairList<T> = List<List<T>>`
|
|
||||||
- Untyped params default to `Object` (`x`) or `Object?` (`x?` shorthand).
|
|
||||||
- Untyped `var x` starts as `Unset`; first assignment fixes type tracking in compiler.
|
|
||||||
|
|
||||||
## 7.1 Generics Runtime Model and Bounds (AI-critical)
|
|
||||||
- Lyng generic type information is operational in script execution contexts; do not assume JVM-style full erasure.
|
|
||||||
- Generic call type arguments can be:
|
|
||||||
- explicit at call site (`f<Int>(1)` style),
|
|
||||||
- inferred from runtime values/declared arg types,
|
|
||||||
- defaulted from type parameter defaults (or `Any` fallback).
|
|
||||||
- At function execution, generic type parameters are runtime-bound as constants in scope:
|
|
||||||
- simple non-null class-like types are bound as `ObjClass`,
|
|
||||||
- complex/nullable/union/intersection forms are bound as `ObjTypeExpr`.
|
|
||||||
- Practical implication for generated code:
|
|
||||||
- inside generic code, treat type params as usable type objects in `is`/`in`/type-expression logic (not as purely compile-time placeholders).
|
|
||||||
- example pattern: `if (value is T) { ... }`.
|
|
||||||
- Bound syntax (implemented):
|
|
||||||
- intersection bound: `fun f<T: A & B>(x: T) { ... }`
|
|
||||||
- union bound: `fun g<T: A | B>(x: T) { ... }`
|
|
||||||
- Bound checks happen at two points:
|
|
||||||
- compile-time call checking for resolvable generic calls,
|
|
||||||
- runtime re-check while binding type params for actual invocation.
|
|
||||||
- Bound satisfaction is currently class-hierarchy based for class-resolvable parts (including union/intersection combination rules).
|
|
||||||
- Keep expectations realistic:
|
|
||||||
- extern-generic runtime ABI for full instance-level generic metadata is still proposal-level (`proposals/extern_generic_runtime_abi.md`), so avoid assuming fully materialized generic-instance metadata everywhere.
|
|
||||||
|
|
||||||
## 7.2 Differences vs Java / Kotlin / Scala
|
|
||||||
- Java:
|
|
||||||
- Java generics are erased at runtime (except reflection metadata and raw `Class` tokens).
|
|
||||||
- Lyng generic params in script execution are runtime-bound type objects, so generated code can reason about `T` directly.
|
|
||||||
- Kotlin:
|
|
||||||
- Kotlin on JVM is mostly erased; full runtime type access usually needs `inline reified`.
|
|
||||||
- Lyng generic function execution binds `T` without requiring an inline/reified escape hatch.
|
|
||||||
- Scala:
|
|
||||||
- Scala has richer static typing but still runs on JVM erasure model unless carrying explicit runtime evidence (`TypeTag`, etc.).
|
|
||||||
- Lyng exposes runtime-bound type expressions/classes directly in generic execution scope.
|
|
||||||
- AI generation rule:
|
|
||||||
- do not port JVM-language assumptions like “`T` unavailable at runtime unless reified/tagged”.
|
|
||||||
- in Lyng, prefer direct type-expression-driven branching when useful, but avoid assuming extern object generic args are always introspectable today.
|
|
||||||
|
|
||||||
## 8. OOP, Members, and Dispatch
|
|
||||||
- Multiple inheritance with C3-style linearization behavior is implemented in class machinery.
|
|
||||||
- Disambiguation helpers are supported:
|
|
||||||
- qualified this: `this@Base.member()`
|
|
||||||
- cast view: `(obj as Base).member()`
|
|
||||||
- In nested receiver lambdas, `this@Type` can target any receiver visible through the receiver stack, not just inheritance ancestors.
|
|
||||||
- On unknown receiver types, compiler allows only Object-safe members:
|
|
||||||
- `toString`, `toInspectString`, `let`, `also`, `apply`, `run`
|
|
||||||
- Other members require known receiver type or explicit cast.
|
|
||||||
|
|
||||||
## 9. Delegation (`by`)
|
|
||||||
- Works for `val`, `var`, and `fun`.
|
|
||||||
- Expected delegate hooks in practice:
|
|
||||||
- `getValue(thisRef, name)`
|
|
||||||
- `setValue(thisRef, name, newValue)`
|
|
||||||
- `invoke(thisRef, name, args...)` for delegated callables
|
|
||||||
- optional `bind(name, access, thisRef)`
|
|
||||||
- `@Transient` is recognized for declarations/params and affects serialization/equality behavior.
|
|
||||||
|
|
||||||
## 10. Modules and Imports
|
|
||||||
- `package` and `import module.name` are supported.
|
|
||||||
- Import form is module-only (no aliasing/selective import syntax in parser).
|
|
||||||
- Default module ecosystem includes:
|
|
||||||
- auto-seeded: `lyng.stdlib`
|
|
||||||
- available by import: `lyng.observable`, `lyng.buffer`, `lyng.serialization`, `lyng.time`
|
|
||||||
- extra module (when installed): `lyng.io.fs`, `lyng.io.process`
|
|
||||||
|
|
||||||
## 11. Current Limitations / Avoid
|
|
||||||
- No subject-less `when { ... }` yet.
|
|
||||||
- No regex literal tokenization (`/.../`); use `Regex("...")` or `"...".re`.
|
|
||||||
- Do not generate runtime name fallback patterns from legacy docs.
|
|
||||||
@ -1,14 +0,0 @@
|
|||||||
# AI notes: publish JVM CLI updates with `bin/local_jrelease`
|
|
||||||
|
|
||||||
[//]: # (excludeFromIndex)
|
|
||||||
|
|
||||||
When a change affects the JVM CLI launcher used as `jlyng`, refresh the installed local distribution with:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
bin/local_jrelease
|
|
||||||
```
|
|
||||||
|
|
||||||
Why:
|
|
||||||
- `jlyng` in this repo is installed from `~/bin/jlyng-jvm/lyng-jvm`, not directly from `lyng/build/install`.
|
|
||||||
- Manual copying from Gradle build output can leave the actual launcher on `PATH` stale.
|
|
||||||
- `bin/local_jrelease` rebuilds `lyng/build/distributions/lyng-jvm.zip`, reinstalls it under `~/bin/jlyng-jvm`, and recreates the `~/bin/jlyng` symlink.
|
|
||||||
@ -1,10 +0,0 @@
|
|||||||
# AI notes: heading levels must be consecutive
|
|
||||||
|
|
||||||
[//]: # (excludeFromIndex)
|
|
||||||
|
|
||||||
When editing repository documentation:
|
|
||||||
|
|
||||||
- Use heading levels in order: `#`, then `##`, then `###`, and so on.
|
|
||||||
- Do not skip levels, for example `#` directly to `###`.
|
|
||||||
- Keep the heading tree balanced inside each document; sibling sections should use the same level.
|
|
||||||
- If you add a subsection and the parent is `##`, the child must be `###`.
|
|
||||||
@ -1,7 +1,5 @@
|
|||||||
# AI notes: avoid Kotlin/Wasm invalid IR with suspend lambdas
|
# AI notes: avoid Kotlin/Wasm invalid IR with suspend lambdas
|
||||||
|
|
||||||
[//]: # (excludeFromIndex)
|
|
||||||
|
|
||||||
## Do
|
## Do
|
||||||
- Prefer explicit `object : Statement()` with `override suspend fun execute(...)` when building compiler statements.
|
- Prefer explicit `object : Statement()` with `override suspend fun execute(...)` when building compiler statements.
|
||||||
- Keep `Statement` objects non-lambda, especially in compiler hot paths like parsing/var declarations.
|
- Keep `Statement` objects non-lambda, especially in compiler hot paths like parsing/var declarations.
|
||||||
|
|||||||
@ -1,105 +0,0 @@
|
|||||||
# Lyng Stdlib Reference for AI Agents (Compact)
|
|
||||||
|
|
||||||
[//]: # (excludeFromIndex)
|
|
||||||
|
|
||||||
Purpose: fast overview of what is available by default and what must be imported.
|
|
||||||
|
|
||||||
Sources: `lynglib/src/commonMain/kotlin/net/sergeych/lyng/Script.kt`, `lynglib/stdlib/lyng/root.lyng`, `lynglib/src/commonMain/kotlin/net/sergeych/lyng/stdlib_included/observable_lyng.kt`.
|
|
||||||
|
|
||||||
## 1. Default Availability
|
|
||||||
- Normal scripts are auto-seeded with `lyng.stdlib` (default import manager path).
|
|
||||||
- Root runtime scope also exposes global constants/functions directly.
|
|
||||||
|
|
||||||
## 2. Core Global Functions (Root Scope)
|
|
||||||
- IO/debug: `print`, `println`, `traceScope`.
|
|
||||||
- Invocation/util: `call`, `run`, `dynamic`, `cached`, `lazy`.
|
|
||||||
- Assertions/tests: `assert`, `assertEquals`/`assertEqual`, `assertNotEquals`, `assertThrows`.
|
|
||||||
- Preconditions: `require`, `check`.
|
|
||||||
- Async/concurrency: `launch`, `yield`, `flow`, `delay`.
|
|
||||||
- `Deferred.cancel()` cancels an active task.
|
|
||||||
- `Deferred.await()` throws `CancellationException` if that task was cancelled.
|
|
||||||
- `Iterable<Deferred>.joinAll()` awaits every deferred in iteration order and returns a `List` of results.
|
|
||||||
- Math: `floor`, `ceil`, `round`, `sin`, `cos`, `tan`, `asin`, `acos`, `atan`, `sinh`, `cosh`, `tanh`, `asinh`, `acosh`, `atanh`, `exp`, `ln`, `log10`, `log2`, `pow`, `sqrt`, `abs`, `clamp`.
|
|
||||||
- These helpers also accept `lyng.decimal.Decimal`.
|
|
||||||
- Exact Decimal path today: `abs`, `floor`, `ceil`, `round`, and `pow` with integral exponent.
|
|
||||||
- Temporary Decimal path for the rest: convert `Decimal -> Real`, compute, then convert back to `Decimal`.
|
|
||||||
- Treat that bridge as temporary; prefer native Decimal implementations when they become available.
|
|
||||||
|
|
||||||
## 3. Core Global Constants/Types
|
|
||||||
- Values: `Unset`, `π`.
|
|
||||||
- Primitive/class symbols: `Object`, `Int`, `Real`, `Bool`, `Char`, `String`, `Class`, `Callable`.
|
|
||||||
- Collections/types: `Iterable`, `Iterator`, `Collection`, `Array`, `List`, `ImmutableList`, `Set`, `ImmutableSet`, `Map`, `ImmutableMap`, `MapEntry`, `Range`, `RingBuffer`.
|
|
||||||
- Random: singleton `Random` and class `SeededRandom`.
|
|
||||||
- Async types: `Deferred`, `CompletableDeferred`, `Mutex`, `Flow`, `FlowBuilder`.
|
|
||||||
- Async exception: `CancellationException`.
|
|
||||||
- Delegation types: `Delegate`, `DelegateContext`.
|
|
||||||
- Regex types: `Regex`, `RegexMatch`.
|
|
||||||
- Also present: `Math.PI` namespace constant.
|
|
||||||
|
|
||||||
## 4. `lyng.stdlib` Module Surface (from `root.lyng`)
|
|
||||||
### 4.1 Extern class declarations
|
|
||||||
- Exceptions/delegation base: `Exception`, `CancellationException`, `IllegalArgumentException`, `NotImplementedException`, `Delegate`.
|
|
||||||
- Collections and iterables: `Iterable<T>`, `Iterator<T>`, `Collection<T>`, `Array<T>`, `List<T>`, `ImmutableList<T>`, `Set<T>`, `ImmutableSet<T>`, `Map<K,V>`, `ImmutableMap<K,V>`, `MapEntry<K,V>`, `RingBuffer<T>`.
|
|
||||||
- Host iterator bridge: `KotlinIterator<T>`.
|
|
||||||
- Random APIs: `extern object Random`, `extern class SeededRandom`.
|
|
||||||
|
|
||||||
### 4.2 High-use extension APIs
|
|
||||||
- Iteration/filtering: `forEach`, `filter`, `filterFlow`, `filterNotNull`, `filterFlowNotNull`, `drop`, `dropLast`, `takeLast`.
|
|
||||||
- Search/predicates: `findFirst`, `findFirstOrNull`, `any`, `all`, `count`, `first`, `last`.
|
|
||||||
- Mapping/aggregation: `map`, `flatMap`, `flatten`, `sum`, `sumOf`, `minOf`, `maxOf`.
|
|
||||||
- Ordering and list building: `sorted`, `sortedBy`, `shuffled`, `List.sort`, `List.sortBy`, `List.fill`.
|
|
||||||
- `List.fill(size) { index -> ... }` constructs a new `List<T>` by evaluating the block once per index from `0` to `size - 1`.
|
|
||||||
- String helper: `joinToString`, `String.re`.
|
|
||||||
|
|
||||||
### 4.3 Delegation helpers
|
|
||||||
- `enum DelegateAccess { Val, Var, Callable }`
|
|
||||||
- `interface Delegate<T,ThisRefType=void>` with `getValue`, `setValue`, `invoke`, `bind`.
|
|
||||||
- `class lazy<T,...>` delegate implementation.
|
|
||||||
- `fun with(self, block)` helper.
|
|
||||||
|
|
||||||
### 4.4 Other module-level symbols
|
|
||||||
- `$~` (last regex match object).
|
|
||||||
- `TODO(message?)` utility.
|
|
||||||
- `StackTraceEntry` class.
|
|
||||||
- `Random.nextInt()`, `Random.nextFloat()`, `Random.next(range)`, `Random.seeded(seed)`.
|
|
||||||
- `SeededRandom.nextInt()`, `SeededRandom.nextFloat()`, `SeededRandom.next(range)`.
|
|
||||||
|
|
||||||
## 5. Additional Built-in Modules (import explicitly)
|
|
||||||
- `import lyng.observable`
|
|
||||||
- `Observable`, `Subscription`, `ObservableList`, `ListChange` and change subtypes, `ChangeRejectionException`.
|
|
||||||
- `import lyng.decimal`
|
|
||||||
- `Decimal`, `DecimalContext`, `DecimalRounding`, `withDecimalContext(...)`.
|
|
||||||
- Kotlin host helper: `ScopeFacade.newDecimal(BigDecimal)` wraps an ionspin host decimal as a Lyng `Decimal`.
|
|
||||||
- `import lyng.complex`
|
|
||||||
- `Complex`, `complex(re, im)`, `cis(angle)`, and numeric embedding extensions such as `2.i` / `3.re`.
|
|
||||||
- `import lyng.matrix`
|
|
||||||
- `Matrix`, `Vector`, `matrix(rows)`, `vector(values)`, dense linear algebra, inversion, solving, and matrix slicing with `m[row, col]`.
|
|
||||||
- `import lyng.buffer`
|
|
||||||
- `Buffer`, `MutableBuffer`.
|
|
||||||
- `import lyng.legacy_digest`
|
|
||||||
- `LegacyDigest.sha1(data): String` — SHA-1 hex digest; `data` may be `String` (UTF-8) or `Buffer` (raw bytes).
|
|
||||||
- ⚠️ Cryptographically broken. Use only for legacy protocol / file-format compatibility.
|
|
||||||
- `import lyng.serialization`
|
|
||||||
- `Lynon` serialization utilities.
|
|
||||||
- `import lyng.time`
|
|
||||||
- `Instant`, `Date`, `DateTime`, `Duration`, and module `delay`.
|
|
||||||
|
|
||||||
## 6. Optional (lyngio) Modules
|
|
||||||
Requires installing `lyngio` into the import manager from host code.
|
|
||||||
- `import lyng.io.fs` (filesystem `Path` API)
|
|
||||||
- `import lyng.io.process` (process execution API)
|
|
||||||
- `import lyng.io.console` (console capabilities, geometry, ANSI/output, events)
|
|
||||||
- `import lyng.io.http` (HTTP/HTTPS client API)
|
|
||||||
- `import lyng.io.http.server` (minimal HTTP/1.1 and WebSocket server API)
|
|
||||||
- `import lyng.io.ws` (WebSocket client API; currently supported on JVM, capability-gated elsewhere)
|
|
||||||
- `import lyng.io.net` (TCP/UDP transport API; currently supported on JVM, capability-gated elsewhere)
|
|
||||||
- `import lyng.io.html` (pure Lyng HTML builder DSL: `html { body { h3 { +"text" } } }`)
|
|
||||||
- Shared network value-type packages are also available when installed by host code:
|
|
||||||
- `import lyng.io.http.types` (`HttpHeaders`)
|
|
||||||
- `import lyng.io.ws.types` (`WsMessage`)
|
|
||||||
- `import lyng.io.net.types` (`IpVersion`, `SocketAddress`, `Datagram`)
|
|
||||||
|
|
||||||
## 7. AI Generation Tips
|
|
||||||
- Assume `lyng.stdlib` APIs exist in regular script contexts.
|
|
||||||
- For platform-sensitive code (`fs`, `process`, `console`, `http`, `ws`, `net`), gate assumptions and mention required module install.
|
|
||||||
- Prefer extension-method style (`items.filter { ... }`) and standard scope helpers (`let`/`also`/`apply`/`run`).
|
|
||||||
@ -34,18 +34,13 @@ Valid examples:
|
|||||||
|
|
||||||
Ellipsis are used to declare variadic arguments. It basically means "all the arguments available here". It means, ellipsis argument could be in any part of the list, being, end or middle, but there could be only one ellipsis argument and it must not have default value, its default value is always `[]`, en empty list.
|
Ellipsis are used to declare variadic arguments. It basically means "all the arguments available here". It means, ellipsis argument could be in any part of the list, being, end or middle, but there could be only one ellipsis argument and it must not have default value, its default value is always `[]`, en empty list.
|
||||||
|
|
||||||
Ellipsis can also appear in **function types** to denote a variadic position:
|
|
||||||
|
|
||||||
var f: (Int, Object..., String)->Real
|
|
||||||
var anyArgs: (...)->Int // shorthand for (Object...)->Int
|
|
||||||
|
|
||||||
Ellipsis argument receives what is left from arguments after processing regular one that could be before or after.
|
Ellipsis argument receives what is left from arguments after processing regular one that could be before or after.
|
||||||
|
|
||||||
Ellipsis could be a first argument:
|
Ellipsis could be a first argument:
|
||||||
|
|
||||||
fun testCountArgs(data...,size) {
|
fun testCountArgs(data...,size) {
|
||||||
assert(size is Int)
|
assert(size is Int)
|
||||||
assertEquals(size, (data as List).size)
|
assertEquals(size, data.size)
|
||||||
}
|
}
|
||||||
testCountArgs( 1, 2, "three", 3)
|
testCountArgs( 1, 2, "three", 3)
|
||||||
>>> void
|
>>> void
|
||||||
@ -54,7 +49,7 @@ Ellipsis could also be a last one:
|
|||||||
|
|
||||||
fun testCountArgs(size, data...) {
|
fun testCountArgs(size, data...) {
|
||||||
assert(size is Int)
|
assert(size is Int)
|
||||||
assertEquals(size, (data as List).size)
|
assertEquals(size, data.size)
|
||||||
}
|
}
|
||||||
testCountArgs( 3, 10, 2, "three")
|
testCountArgs( 3, 10, 2, "three")
|
||||||
>>> void
|
>>> void
|
||||||
@ -63,7 +58,7 @@ Or in the middle:
|
|||||||
|
|
||||||
fun testCountArgs(size, data..., textToReturn) {
|
fun testCountArgs(size, data..., textToReturn) {
|
||||||
assert(size is Int)
|
assert(size is Int)
|
||||||
assertEquals(size, (data as List).size)
|
assertEquals(size, data.size)
|
||||||
textToReturn
|
textToReturn
|
||||||
}
|
}
|
||||||
testCountArgs( 3, 10, 2, "three", "All OK")
|
testCountArgs( 3, 10, 2, "three", "All OK")
|
||||||
|
|||||||
@ -1,12 +0,0 @@
|
|||||||
# Some resources to download
|
|
||||||
|
|
||||||
## Lync CLI tool
|
|
||||||
|
|
||||||
- [lyng-linuxX64.zip](/distributables/lyng-linuxX64.zip) CLI tool for linuxX64: nodependencies, small monolith executable binary.
|
|
||||||
- [lyng-jvm.zip](/distributables/lyng-jvm.zip) JVM CLI distribution: download, unpack, and run `lyng-jvm/bin/lyng`.
|
|
||||||
|
|
||||||
## IDE plugins
|
|
||||||
|
|
||||||
- [lyng-textmate.zip](../../lyng/distributables/lyng-textmate.zip) Texmate-compatible bundle with syntax coloring (could be outdated)
|
|
||||||
|
|
||||||
- [lyng-idea-0.0.5-SNAPSHOT.zip](/distributables/lyng-idea-0.0.5-SNAPSHOT.zip) - plugin for IntelliJ-compatible IDE
|
|
||||||
@ -1,12 +1,10 @@
|
|||||||
# Embedding Lyng in your Kotlin project
|
# Embedding Lyng in your Kotlin project
|
||||||
|
|
||||||
[//]: # (topMenu)
|
|
||||||
|
|
||||||
Lyng is a tiny, embeddable, Kotlin‑first scripting language. This page shows, step by step, how to:
|
Lyng is a tiny, embeddable, Kotlin‑first scripting language. This page shows, step by step, how to:
|
||||||
|
|
||||||
- add Lyng to your build
|
- add Lyng to your build
|
||||||
- create a runtime and execute scripts
|
- create a runtime and execute scripts
|
||||||
- declare extern globals in Lyng and bind them from Kotlin
|
- define functions and variables from Kotlin
|
||||||
- read variable values back in Kotlin
|
- read variable values back in Kotlin
|
||||||
- call Lyng functions from Kotlin
|
- call Lyng functions from Kotlin
|
||||||
- create your own packages and import them in Lyng
|
- create your own packages and import them in Lyng
|
||||||
@ -38,60 +36,21 @@ dependencies {
|
|||||||
|
|
||||||
If you use Kotlin Multiplatform, add the dependency in the `commonMain` source set (and platform‑specific sets if you need platform APIs).
|
If you use Kotlin Multiplatform, add the dependency in the `commonMain` source set (and platform‑specific sets if you need platform APIs).
|
||||||
|
|
||||||
### 2) Preferred runtime: `EvalSession`
|
### 2) Create a runtime (Scope) and execute scripts
|
||||||
|
|
||||||
For host applications, prefer `EvalSession` as the main way to run scripts.
|
The easiest way to get a ready‑to‑use scope with standard packages is via `Script.newScope()`.
|
||||||
It owns one reusable Lyng scope, serializes `eval(...)` calls, and governs coroutines started from Lyng `launch { ... }`.
|
|
||||||
|
|
||||||
Main entrypoints:
|
|
||||||
|
|
||||||
- `session.eval(code)` / `session.eval(source)`
|
|
||||||
- `session.getScope()` when you need low-level binding APIs
|
|
||||||
- `session.cancel()` to cancel active session-owned coroutines
|
|
||||||
- `session.join()` to wait for active session-owned coroutines
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
fun main() = kotlinx.coroutines.runBlocking {
|
|
||||||
val session = EvalSession()
|
|
||||||
|
|
||||||
// Evaluate a one‑liner
|
|
||||||
val result = session.eval("1 + 2 * 3")
|
|
||||||
println("Lyng result: $result") // ObjReal/ObjInt etc.
|
|
||||||
|
|
||||||
// Optional lifecycle management
|
|
||||||
session.join()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
The session creates its underlying scope lazily. If you need raw low-level APIs, get the scope explicitly:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
```
|
|
||||||
|
|
||||||
Use `cancel()` / `join()` to govern async work started by scripts:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
val session = EvalSession()
|
|
||||||
session.eval("""launch { delay(1000); println("done") }""")
|
|
||||||
session.cancel()
|
|
||||||
session.join()
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2.1) Low-level runtime: `Scope`
|
|
||||||
|
|
||||||
Use `Scope` directly when you intentionally want lower-level control.
|
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
fun main() = kotlinx.coroutines.runBlocking {
|
fun main() = kotlinx.coroutines.runBlocking {
|
||||||
val scope = Script.newScope() // suspends on first init
|
val scope = Script.newScope() // suspends on first init
|
||||||
|
|
||||||
|
// Evaluate a one‑liner
|
||||||
val result = scope.eval("1 + 2 * 3")
|
val result = scope.eval("1 + 2 * 3")
|
||||||
println("Lyng result: $result")
|
println("Lyng result: $result") // ObjReal/ObjInt etc.
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
You can also pre‑compile a script and execute it multiple times on the same scope:
|
You can also pre‑compile a script and execute it multiple times:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val script = Compiler.compile("""
|
val script = Compiler.compile("""
|
||||||
@ -104,98 +63,34 @@ val run1 = script.execute(scope)
|
|||||||
val run2 = script.execute(scope)
|
val run2 = script.execute(scope)
|
||||||
```
|
```
|
||||||
|
|
||||||
`Scope.eval("...")` is the low-level shortcut that compiles and executes on the given scope.
|
`Scope.eval("...")` is a shortcut that compiles and executes on the given scope.
|
||||||
For most embedding use cases, prefer `session.eval("...")`.
|
|
||||||
|
|
||||||
### 3) Preferred: bind extern globals from Kotlin
|
### 3) Define variables from Kotlin
|
||||||
|
|
||||||
For module-level APIs, the default workflow is:
|
To expose data to Lyng, add constants (read‑only) or mutable variables to the scope. All values in Lyng are `Obj` instances; the core types live in `net.sergeych.lyng.obj`.
|
||||||
|
|
||||||
1. declare globals in Lyng using `extern fun` / `extern val` / `extern var`;
|
|
||||||
2. bind Kotlin implementation via `ModuleScope.globalBinder()`.
|
|
||||||
|
|
||||||
This is also the recommended way to expose a Kotlin-backed value that should behave like a true
|
|
||||||
Lyng global variable/property. If you need `x` to read/write through Kotlin on every access, use
|
|
||||||
`extern var` / `extern val` plus `bindGlobalVar(...)`.
|
|
||||||
|
|
||||||
Do not use `addConst(...)` for this case: `addConst(...)` installs a value, not a Kotlin-backed
|
|
||||||
property accessor. It is appropriate for fixed values and objects, but not for a global that should
|
|
||||||
delegate reads/writes back into Kotlin state.
|
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
import net.sergeych.lyng.bridge.*
|
// Read‑only constant
|
||||||
import net.sergeych.lyng.obj.ObjInt
|
scope.addConst("pi", ObjReal(3.14159))
|
||||||
import net.sergeych.lyng.obj.ObjString
|
|
||||||
|
|
||||||
val session = EvalSession()
|
// Mutable variable: create or update
|
||||||
val scope = session.getScope()
|
scope.addOrUpdateItem("counter", ObjInt(0))
|
||||||
val im = Script.defaultImportManager.copy()
|
|
||||||
im.addPackage("my.api") { module ->
|
|
||||||
module.eval("""
|
|
||||||
extern fun globalFun(v: Int): Int
|
|
||||||
extern var globalProp: String
|
|
||||||
extern val globalVersion: String
|
|
||||||
""".trimIndent())
|
|
||||||
|
|
||||||
val binder = module.globalBinder()
|
// Use it from Lyng
|
||||||
|
scope.eval("counter = counter + 1")
|
||||||
binder.bindGlobalFun1<Int>("globalFun") { v ->
|
|
||||||
ObjInt.of((v + 1).toLong())
|
|
||||||
}
|
|
||||||
|
|
||||||
var prop = "initial"
|
|
||||||
binder.bindGlobalVar(
|
|
||||||
name = "globalProp",
|
|
||||||
get = { prop },
|
|
||||||
set = { prop = it }
|
|
||||||
)
|
|
||||||
|
|
||||||
binder.bindGlobalVar(
|
|
||||||
name = "globalVersion",
|
|
||||||
get = { "1.0.0" } // readonly: setter omitted
|
|
||||||
)
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Usage from Lyng:
|
Tip: Lyng values can be converted back to Kotlin with `toKotlin(scope)`:
|
||||||
|
|
||||||
```lyng
|
|
||||||
import my.api
|
|
||||||
|
|
||||||
assertEquals(42, globalFun(41))
|
|
||||||
assertEquals("initial", globalProp)
|
|
||||||
globalProp = "changed"
|
|
||||||
assertEquals("changed", globalProp)
|
|
||||||
assertEquals("1.0.0", globalVersion)
|
|
||||||
```
|
|
||||||
|
|
||||||
Minimal rule of thumb:
|
|
||||||
|
|
||||||
- use `bindGlobalFun(...)` for global functions
|
|
||||||
- use `bindGlobalVar(...)` for Kotlin-backed global variables/properties
|
|
||||||
- use `addConst(...)` only for fixed values/objects that do not need getter/setter behavior
|
|
||||||
|
|
||||||
For custom argument handling and full runtime access:
|
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
binder.bindGlobalFun("sum3") {
|
val current = (scope.eval("counter")).toKotlin(scope) // Any? (e.g., Int/Double/String/List)
|
||||||
requireExactCount(3)
|
|
||||||
ObjInt.of((int(0) + int(1) + int(2)).toLong())
|
|
||||||
}
|
|
||||||
|
|
||||||
binder.bindGlobalFunRaw("echoRaw") { _, args ->
|
|
||||||
args.firstAndOnly()
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### 4) Low-level: direct functions/variables from Kotlin
|
### 4) Add Kotlin‑backed functions
|
||||||
|
|
||||||
Use this when you intentionally want raw `Scope` APIs. For most module APIs, prefer section 3.
|
Use `Scope.addFn`/`addVoidFn` to register functions implemented in Kotlin. Inside the lambda, use `this.args` to access arguments and return an `Obj`.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
|
|
||||||
// A function returning value
|
// A function returning value
|
||||||
scope.addFn<ObjInt>("inc") {
|
scope.addFn<ObjInt>("inc") {
|
||||||
val x = args.firstAndOnly() as ObjInt
|
val x = args.firstAndOnly() as ObjInt
|
||||||
@ -214,95 +109,16 @@ scope.addVoidFn("log") {
|
|||||||
// }
|
// }
|
||||||
|
|
||||||
// Call them from Lyng
|
// Call them from Lyng
|
||||||
session.eval("val y = inc(41); log('Answer:', y)")
|
scope.eval("val y = inc(41); log('Answer:', y)")
|
||||||
```
|
```
|
||||||
|
|
||||||
You can register multiple names (aliases) at once: `addFn<ObjInt>("inc", "increment") { ... }`.
|
You can register multiple names (aliases) at once: `addFn<ObjInt>("inc", "increment") { ... }`.
|
||||||
|
|
||||||
Scope-backed Kotlin lambdas receive a `ScopeFacade` (not a full `Scope`). For migration and convenience, these utilities are available on the facade:
|
|
||||||
|
|
||||||
- Access: `args`, `pos`, `thisObj`, `get(name)`
|
|
||||||
- Invocation: `call(...)`, `resolve(...)`, `assign(...)`, `toStringOf(...)`, `inspect(...)`, `trace(...)`
|
|
||||||
- Args helpers: `requiredArg<T>()`, `requireOnlyArg<T>()`, `requireExactCount(...)`, `requireNoArgs()`, `thisAs<T>()`
|
|
||||||
- Errors: `raiseError(...)`, `raiseClassCastError(...)`, `raiseIllegalArgument(...)`, `raiseIllegalState(...)`, `raiseNoSuchElement(...)`,
|
|
||||||
`raiseSymbolNotFound(...)`, `raiseNotImplemented(...)`, `raiseNPE()`, `raiseIndexOutOfBounds(...)`, `raiseIllegalAssignment(...)`,
|
|
||||||
`raiseUnset(...)`, `raiseNotFound(...)`, `raiseAssertionFailed(...)`, `raiseIllegalOperation(...)`, `raiseIterationFinished()`
|
|
||||||
|
|
||||||
If you truly need the full `Scope` (e.g., for low-level interop), use `requireScope()` explicitly.
|
|
||||||
|
|
||||||
### 4.5) Indexers from Kotlin: `getAt` and `putAt`
|
|
||||||
|
|
||||||
Lyng bracket syntax is dispatched through `getAt` and `putAt`.
|
|
||||||
|
|
||||||
That means:
|
|
||||||
|
|
||||||
- `x[i]` calls `getAt(index)`
|
|
||||||
- `x[i] = value` calls `putAt(index, value)` or `setAt(index, value)`
|
|
||||||
- field-like `x["name"]` also uses the same index path unless you expose a real field/property
|
|
||||||
|
|
||||||
For Kotlin-backed classes, bind indexers as ordinary methods named `getAt` and `putAt`:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
moduleScope.eval("""
|
|
||||||
extern class Grid {
|
|
||||||
override fun getAt(index: List<Int>): Int
|
|
||||||
override fun putAt(index: List<Int>, value: Int): void
|
|
||||||
}
|
|
||||||
""".trimIndent())
|
|
||||||
|
|
||||||
moduleScope.bind("Grid") {
|
|
||||||
init { _ -> data = IntArray(4) }
|
|
||||||
|
|
||||||
addFun("getAt") {
|
|
||||||
val index = args.requiredArg<ObjList>(0)
|
|
||||||
val row = (index.list[0] as ObjInt).value.toInt()
|
|
||||||
val col = (index.list[1] as ObjInt).value.toInt()
|
|
||||||
val data = (thisObj as ObjInstance).data as IntArray
|
|
||||||
ObjInt.of(data[row * 2 + col].toLong())
|
|
||||||
}
|
|
||||||
|
|
||||||
addFun("putAt") {
|
|
||||||
val index = args.requiredArg<ObjList>(0)
|
|
||||||
val value = args.requiredArg<ObjInt>(1).value.toInt()
|
|
||||||
val row = (index.list[0] as ObjInt).value.toInt()
|
|
||||||
val col = (index.list[1] as ObjInt).value.toInt()
|
|
||||||
val data = (thisObj as ObjInstance).data as IntArray
|
|
||||||
data[row * 2 + col] = value
|
|
||||||
ObjVoid
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Usage from Lyng:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val g = Grid()
|
|
||||||
g[0, 1] = 42
|
|
||||||
assertEquals(42, g[0, 1])
|
|
||||||
```
|
|
||||||
|
|
||||||
Important rule: multiple selectors inside brackets are packed into one index object.
|
|
||||||
So:
|
|
||||||
|
|
||||||
- `x[i]` passes `i`
|
|
||||||
- `x[i, j]` passes a `List` containing `[i, j]`
|
|
||||||
- `x[i, j, k]` passes `[i, j, k]`
|
|
||||||
|
|
||||||
This applies equally to:
|
|
||||||
|
|
||||||
- Kotlin-backed classes
|
|
||||||
- Lyng classes overriding `getAt`
|
|
||||||
- `dynamic { get { ... } set { ... } }`
|
|
||||||
|
|
||||||
If you want multi-axis slicing semantics, decode that list yourself in `getAt`.
|
|
||||||
|
|
||||||
### 5) Add Kotlin‑backed fields
|
### 5) Add Kotlin‑backed fields
|
||||||
|
|
||||||
If you need a simple field (with a value) instead of a computed property, use `createField`. This adds a field to the class that will be present in all its instances.
|
If you need a simple field (with a value) instead of a computed property, use `createField`. This adds a field to the class that will be present in all its instances.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
val myClass = ObjClass("MyClass")
|
val myClass = ObjClass("MyClass")
|
||||||
|
|
||||||
// Add a read-only field (constant)
|
// Add a read-only field (constant)
|
||||||
@ -330,8 +146,6 @@ println(instance.count) // -> 5
|
|||||||
Properties in Lyng are pure accessors (getters and setters) and do not have automatic backing fields. You can add them to a class using `addProperty`.
|
Properties in Lyng are pure accessors (getters and setters) and do not have automatic backing fields. You can add them to a class using `addProperty`.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
val myClass = ObjClass("MyClass")
|
val myClass = ObjClass("MyClass")
|
||||||
var internalValue: Long = 10
|
var internalValue: Long = 10
|
||||||
|
|
||||||
@ -374,12 +188,6 @@ For extensions and libraries, the **preferred** workflow is Lyng‑first: declar
|
|||||||
|
|
||||||
This keeps Lyng semantics (visibility, overrides, type checks) in Lyng, while Kotlin supplies the behavior.
|
This keeps Lyng semantics (visibility, overrides, type checks) in Lyng, while Kotlin supplies the behavior.
|
||||||
|
|
||||||
Pure extern declarations use the simplified rule set:
|
|
||||||
- `extern class` / `extern object` are declaration-only ABI surfaces.
|
|
||||||
- Every member in their body is implicitly extern (you may still write `extern`, but it is redundant).
|
|
||||||
- Plain Lyng member implementations inside `extern class` / `extern object` are not allowed.
|
|
||||||
- Put Lyng behavior into regular classes or extension methods.
|
|
||||||
|
|
||||||
```lyng
|
```lyng
|
||||||
// Lyng side (in a module)
|
// Lyng side (in a module)
|
||||||
class Counter {
|
class Counter {
|
||||||
@ -388,22 +196,7 @@ class Counter {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Note: members of `extern class` / `extern object` are treated as extern by default, so the compiler emits ABI slots that Kotlin bindings attach to. This applies to functions and properties bound via `addFun` / `addVal` / `addVar`.
|
Note: members must be marked `extern` so the compiler emits the ABI slots that Kotlin bindings attach to. This applies to functions and properties bound via `addFun` / `addVal` / `addVar`.
|
||||||
|
|
||||||
Example of pure extern class declaration:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
extern class HostCounter {
|
|
||||||
var value: Int
|
|
||||||
fun inc(by: Int): Int
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
If you need Lyng-side convenience behavior, add it as an extension:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
fun HostCounter.bump() = inc(1)
|
|
||||||
```
|
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
// Kotlin side (binding)
|
// Kotlin side (binding)
|
||||||
@ -413,14 +206,14 @@ moduleScope.eval("class Counter { extern var value: Int; extern fun inc(by: Int)
|
|||||||
moduleScope.bind("Counter") {
|
moduleScope.bind("Counter") {
|
||||||
addVar(
|
addVar(
|
||||||
name = "value",
|
name = "value",
|
||||||
get = { thisObj.readField(this, "value").value },
|
get = { _, self -> self.readField(this, "value").value },
|
||||||
set = { v -> thisObj.writeField(this, "value", v) }
|
set = { _, self, v -> self.writeField(this, "value", v) }
|
||||||
)
|
)
|
||||||
addFun("inc") {
|
addFun("inc") { _, self, args ->
|
||||||
val by = args.requiredArg<ObjInt>(0).value
|
val by = args.requiredArg<ObjInt>(0).value
|
||||||
val current = thisObj.readField(this, "value").value as ObjInt
|
val current = self.readField(this, "value").value as ObjInt
|
||||||
val next = ObjInt(current.value + by)
|
val next = ObjInt(current.value + by)
|
||||||
thisObj.writeField(this, "value", next)
|
self.writeField(this, "value", next)
|
||||||
next
|
next
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -432,75 +225,14 @@ Notes:
|
|||||||
- Use [LyngClassBridge] to bind by name/module, or by an already resolved `ObjClass`.
|
- Use [LyngClassBridge] to bind by name/module, or by an already resolved `ObjClass`.
|
||||||
- Use `ObjInstance.data` / `ObjClass.classData` to attach Kotlin‑side state when needed.
|
- Use `ObjInstance.data` / `ObjClass.classData` to attach Kotlin‑side state when needed.
|
||||||
|
|
||||||
### 6.5a) Bind Kotlin implementations to declared Lyng objects
|
|
||||||
|
|
||||||
For `extern object` declarations, bind implementations to the singleton instance using `ModuleScope.bindObject`.
|
|
||||||
This mirrors class binding but targets an already created object instance.
|
|
||||||
As with class binding, you must first add/evaluate the Lyng declaration into that module scope, then bind Kotlin handlers.
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
// Kotlin side (binding)
|
|
||||||
val moduleScope = importManager.createModuleScope(Pos.builtIn, "bridge.obj")
|
|
||||||
|
|
||||||
// 1) Seed the module with the Lyng declaration first
|
|
||||||
moduleScope.eval("""
|
|
||||||
extern object HostObject {
|
|
||||||
extern fun add(a: Int, b: Int): Int
|
|
||||||
extern val status: String
|
|
||||||
extern var count: Int
|
|
||||||
}
|
|
||||||
""".trimIndent())
|
|
||||||
|
|
||||||
// 2) Then bind Kotlin implementations to that declared object
|
|
||||||
moduleScope.bindObject("HostObject") {
|
|
||||||
classData = "OK"
|
|
||||||
init { _ -> data = 0L }
|
|
||||||
addFun("add") {
|
|
||||||
val a = args.requiredArg<ObjInt>(0).value
|
|
||||||
val b = args.requiredArg<ObjInt>(1).value
|
|
||||||
ObjInt.of(a + b)
|
|
||||||
}
|
|
||||||
addVal("status") { ObjString(classData as String) }
|
|
||||||
addVar(
|
|
||||||
"count",
|
|
||||||
get = { ObjInt.of((thisObj as ObjInstance).data as Long) },
|
|
||||||
set = { value -> (thisObj as ObjInstance).data = (value as ObjInt).value }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Notes:
|
|
||||||
|
|
||||||
- Required order: declare/eval Lyng object in the module first, then call `bindObject(...)`.
|
|
||||||
This is the pattern covered by `BridgeBindingTest.testExternObjectBinding`.
|
|
||||||
- Members must be extern (explicitly, or implicitly via `extern object`) so the compiler emits ABI slots for Kotlin bindings.
|
|
||||||
- You can also bind by name/module via `LyngObjectBridge.bind(...)`.
|
|
||||||
|
|
||||||
Minimal `extern fun` example:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
val moduleScope = importManager.createModuleScope(Pos.builtIn, "bridge.ping")
|
|
||||||
|
|
||||||
moduleScope.eval("""
|
|
||||||
extern object HostObject {
|
|
||||||
extern fun ping(): Int
|
|
||||||
}
|
|
||||||
""".trimIndent())
|
|
||||||
|
|
||||||
moduleScope.bindObject("HostObject") {
|
|
||||||
addFun("ping") { ObjInt.of(7) }
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### 6.6) Preferred: Kotlin reflection bridge for call‑by‑name
|
### 6.6) Preferred: Kotlin reflection bridge for call‑by‑name
|
||||||
|
|
||||||
For Kotlin code that needs dynamic access to Lyng variables, functions, or members, use the bridge resolver.
|
For Kotlin code that needs dynamic access to Lyng variables, functions, or members, use the bridge resolver.
|
||||||
It provides explicit, cached handles and predictable lookup rules.
|
It provides explicit, cached handles and predictable lookup rules.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val session = EvalSession()
|
val scope = Script.newScope()
|
||||||
val scope = session.getScope()
|
scope.eval("""
|
||||||
session.eval("""
|
|
||||||
val x = 40
|
val x = 40
|
||||||
fun add(a, b) = a + b
|
fun add(a, b) = a + b
|
||||||
class Box { var value = 1 }
|
class Box { var value = 1 }
|
||||||
@ -515,7 +247,7 @@ val x = resolver.resolveVal("x").get(scope)
|
|||||||
val sum = (resolver as BridgeCallByName).callByName(scope, "add", Arguments(ObjInt(1), ObjInt(2)))
|
val sum = (resolver as BridgeCallByName).callByName(scope, "add", Arguments(ObjInt(1), ObjInt(2)))
|
||||||
|
|
||||||
// Member access
|
// Member access
|
||||||
val box = session.eval("Box()")
|
val box = scope.eval("Box()")
|
||||||
val valueHandle = resolver.resolveMemberVar(box, "value")
|
val valueHandle = resolver.resolveMemberVar(box, "value")
|
||||||
valueHandle.set(scope, ObjInt(10))
|
valueHandle.set(scope, ObjInt(10))
|
||||||
val value = valueHandle.get(scope)
|
val value = valueHandle.get(scope)
|
||||||
@ -526,14 +258,12 @@ val value = valueHandle.get(scope)
|
|||||||
The simplest approach: evaluate an expression that yields the value and convert it.
|
The simplest approach: evaluate an expression that yields the value and convert it.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val session = EvalSession()
|
val kotlinAnswer = scope.eval("(1 + 2) * 3").toKotlin(scope) // -> 9 (Int)
|
||||||
val scope = session.getScope()
|
|
||||||
val kotlinAnswer = session.eval("(1 + 2) * 3").toKotlin(scope) // -> 9 (Int)
|
|
||||||
|
|
||||||
// After scripts manipulate your vars:
|
// After scripts manipulate your vars:
|
||||||
scope.addOrUpdateItem("name", ObjString("Lyng"))
|
scope.addOrUpdateItem("name", ObjString("Lyng"))
|
||||||
session.eval("name = name + ' rocks!'")
|
scope.eval("name = name + ' rocks!'")
|
||||||
val kotlinName = session.eval("name").toKotlin(scope) // -> "Lyng rocks!"
|
val kotlinName = scope.eval("name").toKotlin(scope) // -> "Lyng rocks!"
|
||||||
```
|
```
|
||||||
|
|
||||||
Advanced: you can also grab a variable record directly via `scope.get(name)` and work with its `Obj` value, but evaluating `"name"` is often clearer and enforces Lyng semantics consistently.
|
Advanced: you can also grab a variable record directly via `scope.get(name)` and work with its `Obj` value, but evaluating `"name"` is often clearer and enforces Lyng semantics consistently.
|
||||||
@ -546,20 +276,16 @@ There are two convenient patterns.
|
|||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
// Suppose Lyng defines: fun add(a, b) = a + b
|
// Suppose Lyng defines: fun add(a, b) = a + b
|
||||||
val session = EvalSession()
|
scope.eval("fun add(a, b) = a + b")
|
||||||
val scope = session.getScope()
|
|
||||||
session.eval("fun add(a, b) = a + b")
|
|
||||||
|
|
||||||
val sum = session.eval("add(20, 22)").toKotlin(scope) // -> 42
|
val sum = scope.eval("add(20, 22)").toKotlin(scope) // -> 42
|
||||||
```
|
```
|
||||||
|
|
||||||
2) Call a Lyng function by name via a prepared call scope:
|
2) Call a Lyng function by name via a prepared call scope:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
// Ensure the function exists in the scope
|
// Ensure the function exists in the scope
|
||||||
val session = EvalSession()
|
scope.eval("fun add(a, b) = a + b")
|
||||||
val scope = session.getScope()
|
|
||||||
session.eval("fun add(a, b) = a + b")
|
|
||||||
|
|
||||||
// Look up the function object
|
// Look up the function object
|
||||||
val addFn = scope.get("add")!!.value as Statement
|
val addFn = scope.get("add")!!.value as Statement
|
||||||
@ -587,47 +313,27 @@ Key concepts:
|
|||||||
Register a Kotlin‑built package:
|
Register a Kotlin‑built package:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
import net.sergeych.lyng.bridge.*
|
val scope = Script.newScope()
|
||||||
import net.sergeych.lyng.obj.ObjInt
|
|
||||||
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
|
|
||||||
// Access the import manager behind this scope
|
// Access the import manager behind this scope
|
||||||
val im: ImportManager = scope.importManager
|
val im: ImportManager = scope.importManager
|
||||||
|
|
||||||
// Register a package "my.tools"
|
// Register a package "my.tools"
|
||||||
im.addPackage("my.tools") { module: ModuleScope ->
|
im.addPackage("my.tools") { module: ModuleScope ->
|
||||||
module.eval(
|
// Expose symbols inside the module scope
|
||||||
"""
|
module.addConst("version", ObjString("1.0"))
|
||||||
extern val version: String
|
module.addFn<ObjInt>("triple") {
|
||||||
extern var status: String
|
val x = args.firstAndOnly() as ObjInt
|
||||||
extern fun triple(x: Int): Int
|
ObjInt(x.value * 3)
|
||||||
""".trimIndent()
|
|
||||||
)
|
|
||||||
val binder = module.globalBinder()
|
|
||||||
var status = "ready"
|
|
||||||
binder.bindGlobalVar(
|
|
||||||
name = "version",
|
|
||||||
get = { "1.0" }
|
|
||||||
)
|
|
||||||
binder.bindGlobalVar(
|
|
||||||
name = "status",
|
|
||||||
get = { status },
|
|
||||||
set = { status = it }
|
|
||||||
)
|
|
||||||
binder.bindGlobalFun1<Int>("triple") { x ->
|
|
||||||
ObjInt.of((x * 3).toLong())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use it from Lyng
|
// Use it from Lyng
|
||||||
session.eval("""
|
scope.eval("""
|
||||||
import my.tools.*
|
import my.tools.*
|
||||||
val v = triple(14)
|
val v = triple(14)
|
||||||
status = "busy"
|
|
||||||
""")
|
""")
|
||||||
val v = session.eval("v").toKotlin(scope) // -> 42
|
val v = scope.eval("v").toKotlin(scope) // -> 42
|
||||||
```
|
```
|
||||||
|
|
||||||
Register a package from Lyng source text:
|
Register a package from Lyng source text:
|
||||||
@ -641,27 +347,24 @@ val pkgText = """
|
|||||||
|
|
||||||
scope.importManager.addTextPackages(pkgText)
|
scope.importManager.addTextPackages(pkgText)
|
||||||
|
|
||||||
session.eval("""
|
scope.eval("""
|
||||||
import math.extra.*
|
import math.extra.*
|
||||||
val s = sqr(12)
|
val s = sqr(12)
|
||||||
""")
|
""")
|
||||||
val s = session.eval("s").toKotlin(scope) // -> 144
|
val s = scope.eval("s").toKotlin(scope) // -> 144
|
||||||
```
|
```
|
||||||
|
|
||||||
You can also register from parsed `Source` instances via `addSourcePackages(source)`.
|
You can also register from parsed `Source` instances via `addSourcePackages(source)`.
|
||||||
|
|
||||||
### 10) Executing from files, security, and isolation
|
### 10) Executing from files, security, and isolation
|
||||||
|
|
||||||
- To run code from a file, read it and pass to `session.eval(text)` or compile with `Compiler.compile(Source(fileName, text))`.
|
- To run code from a file, read it and pass to `scope.eval(text)` or compile with `Compiler.compile(Source(fileName, text))`.
|
||||||
- `ImportManager` takes an optional `SecurityManager` if you need to restrict what packages or operations are available. By default, `Script.defaultImportManager` allows everything suitable for embedded use; clamp it down in sandboxed environments.
|
- `ImportManager` takes an optional `SecurityManager` if you need to restrict what packages or operations are available. By default, `Script.defaultImportManager` allows everything suitable for embedded use; clamp it down in sandboxed environments.
|
||||||
- For isolation, prefer a fresh `EvalSession()` per request. Use `Scope.new()` / `Script.newScope()` when you specifically need low-level raw scopes or modules.
|
- For isolation, create fresh modules/scopes via `Scope.new()` or `Script.newScope()` when you need a clean environment per request.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
// Preferred per-request runtime:
|
// Fresh module based on the default manager, without the standard prelude
|
||||||
val isolatedSession = EvalSession()
|
val isolated = net.sergeych.lyng.Scope.new()
|
||||||
|
|
||||||
// Low-level fresh module based on the default manager, without the standard prelude:
|
|
||||||
val isolatedScope = net.sergeych.lyng.Scope.new()
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### 11) Tips and troubleshooting
|
### 11) Tips and troubleshooting
|
||||||
@ -696,11 +399,8 @@ To simplify handling these objects from Kotlin, several extension methods are pr
|
|||||||
You can serialize Lyng exception objects using `Lynon` to transmit them across boundaries and then rethrow them.
|
You can serialize Lyng exception objects using `Lynon` to transmit them across boundaries and then rethrow them.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
session.eval("throw MyUserException(404, \"Not Found\")")
|
scope.eval("throw MyUserException(404, \"Not Found\")")
|
||||||
} catch (e: ExecutionError) {
|
} catch (e: ExecutionError) {
|
||||||
// 1. Serialize the Lyng exception object
|
// 1. Serialize the Lyng exception object
|
||||||
val encoded: UByteArray = lynonEncodeAny(scope, e.errorObject)
|
val encoded: UByteArray = lynonEncodeAny(scope, e.errorObject)
|
||||||
|
|||||||
@ -109,24 +109,6 @@ Examples (T = A | B):
|
|||||||
B in T // true
|
B in T // true
|
||||||
T is A | B // true
|
T is A | B // true
|
||||||
|
|
||||||
# Nullability checks for types
|
|
||||||
|
|
||||||
Use `is nullable` to check whether a type expression accepts `null`:
|
|
||||||
|
|
||||||
T is nullable
|
|
||||||
T !is nullable
|
|
||||||
|
|
||||||
This works with concrete and generic types:
|
|
||||||
|
|
||||||
fun describe<T>(x: T): String = when (T) {
|
|
||||||
nullable -> "nullable"
|
|
||||||
else -> "non-null"
|
|
||||||
}
|
|
||||||
|
|
||||||
Equivalent legacy form:
|
|
||||||
|
|
||||||
null is T
|
|
||||||
|
|
||||||
# Practical examples
|
# Practical examples
|
||||||
|
|
||||||
fun acceptInts<T: Int>(xs: List<T>) { }
|
fun acceptInts<T: Int>(xs: List<T>) { }
|
||||||
|
|||||||
@ -9,12 +9,9 @@ should be compatible with other IDEA flavors, notably [OpenIDE](https://openide.
|
|||||||
- reformat code (indents, spaces)
|
- reformat code (indents, spaces)
|
||||||
- reformat on paste
|
- reformat on paste
|
||||||
- smart enter key
|
- smart enter key
|
||||||
- `.lyng.d` definition files (merged into analysis for completion, navigation, Quick Docs, and error checking)
|
|
||||||
|
|
||||||
Features are configurable via the plugin settings page, in system settings.
|
Features are configurable via the plugin settings page, in system settings.
|
||||||
|
|
||||||
See `docs/lyng_d_files.md` for `.lyng.d` syntax and examples.
|
|
||||||
|
|
||||||
> Recommended for IntelliJ-based IDEs: While IntelliJ can import TextMate bundles
|
> Recommended for IntelliJ-based IDEs: While IntelliJ can import TextMate bundles
|
||||||
> (Settings/Preferences → Editor → TextMate Bundles), the native Lyng plugin provides
|
> (Settings/Preferences → Editor → TextMate Bundles), the native Lyng plugin provides
|
||||||
> better support (formatting, smart enter, background analysis, etc.). Prefer installing
|
> better support (formatting, smart enter, background analysis, etc.). Prefer installing
|
||||||
@ -27,6 +24,6 @@ See `docs/lyng_d_files.md` for `.lyng.d` syntax and examples.
|
|||||||
- Alternatively, if/when the plugin is published to a marketplace, you will be able to install it
|
- Alternatively, if/when the plugin is published to a marketplace, you will be able to install it
|
||||||
directly from the “Marketplace” tab (not yet available).
|
directly from the “Marketplace” tab (not yet available).
|
||||||
|
|
||||||
### [Download plugin v0.0.5-SNAPSHOT](https://lynglang.com/distributables/lyng-idea-0.0.5-SNAPSHOT.zip)
|
### [Download plugin v0.0.2-SNAPSHOT](https://lynglang.com/distributables/lyng-idea-0.0.2-SNAPSHOT.zip)
|
||||||
|
|
||||||
Your ideas and bugreports are welcome on the [project gitea page](https://gitea.sergeych.net/SergeychWorks/lyng/issues)
|
Your ideas and bugreports are welcome on the [project gitea page](https://gitea.sergeych.net/SergeychWorks/lyng/issues)
|
||||||
@ -1,32 +1,9 @@
|
|||||||
# Json support
|
# Json support
|
||||||
|
|
||||||
Lyng now has two distinct JSON-facing layers:
|
Since 1.0.5 we start adding JSON support. Versions 1,0,6* support serialization of the basic types, including lists and
|
||||||
|
maps, and simple classes. Multiple inheritance may produce incorrect results, it is work in progress.
|
||||||
|
|
||||||
- plain JSON projection:
|
## Serialization in Lyng
|
||||||
- `Obj.toJson()`
|
|
||||||
- `Obj.toJsonString()`
|
|
||||||
- canonical JSON round-trip format:
|
|
||||||
- `Json.encode(value)`
|
|
||||||
- `Json.decode(text)`
|
|
||||||
- typed canonical JSON round-trip format:
|
|
||||||
- `Json.encodeAs(Type, value)`
|
|
||||||
- `Json.decodeAs(Type, text)`
|
|
||||||
|
|
||||||
Use the first when you need ordinary JSON for interop.
|
|
||||||
|
|
||||||
Use the second when you need Lyng value round-trip semantics through JSON text with no schema.
|
|
||||||
|
|
||||||
Use the third when both sides already know the Lyng type and you want the same round-trip semantics with fewer type
|
|
||||||
tags in the JSON.
|
|
||||||
|
|
||||||
This distinction is intentional:
|
|
||||||
|
|
||||||
- plain JSON projection is optimized for compatibility with ordinary JSON tooling
|
|
||||||
- canonical `Json.encode()` is optimized for semantic fidelity to Lyng and Lynon and stays self-describing
|
|
||||||
- typed canonical `Json.encodeAs()` is optimized for the same fidelity when the schema is provided externally
|
|
||||||
- these goals conflict for values such as sets, exceptions, singleton objects, buffers, and maps with non-string keys
|
|
||||||
|
|
||||||
## Plain JSON projection in Lyng
|
|
||||||
|
|
||||||
// in lyng
|
// in lyng
|
||||||
assertEquals("{\"a\":1}", {a: 1}.toJsonString())
|
assertEquals("{\"a\":1}", {a: 1}.toJsonString())
|
||||||
@ -43,8 +20,7 @@ Simple classes serialization is supported:
|
|||||||
assertEquals( "{\"foo\":1,\"bar\":2}", Point(1,2).toJsonString() )
|
assertEquals( "{\"foo\":1,\"bar\":2}", Point(1,2).toJsonString() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Note that mutable members are serialized by default. You can exclude any member (including constructor parameters) from
|
Note that mutable members are serialized by default. You can exclude any member (including constructor parameters) from JSON serialization using the `@Transient` attribute:
|
||||||
JSON serialization using the `@Transient` attribute:
|
|
||||||
|
|
||||||
import lyng.serialization
|
import lyng.serialization
|
||||||
|
|
||||||
@ -55,7 +31,7 @@ JSON serialization using the `@Transient` attribute:
|
|||||||
assertEquals( "{\"bar\":2,\"visible\":100}", Point2(1,2).toJsonString() )
|
assertEquals( "{\"bar\":2,\"visible\":100}", Point2(1,2).toJsonString() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Note that if you override plain JSON serialization:
|
Note that if you override json serialization:
|
||||||
|
|
||||||
import lyng.serialization
|
import lyng.serialization
|
||||||
|
|
||||||
@ -70,8 +46,8 @@ Note that if you override plain JSON serialization:
|
|||||||
assertEquals( "{\"custom\":true}", Point2(1,2).toJsonString() )
|
assertEquals( "{\"custom\":true}", Point2(1,2).toJsonString() )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Custom serialization of user classes is possible by overriding `toJsonObject`. It must return an object which is
|
Custom serialization of user classes is possible by overriding `toJsonObject` method. It must return an object which is
|
||||||
serializable to JSON. Most often it is a map, but any object is accepted:
|
serializable to Json. Most often it is a map, but any object is accepted, that makes it very flexible:
|
||||||
|
|
||||||
import lyng.serialization
|
import lyng.serialization
|
||||||
|
|
||||||
@ -94,87 +70,12 @@ serializable to JSON. Most often it is a map, but any object is accepted:
|
|||||||
Please note that `toJsonString` should be used to get serialized string representation of the object. Don't call
|
Please note that `toJsonString` should be used to get serialized string representation of the object. Don't call
|
||||||
`toJsonObject` directly, it is not intended to be used outside the serialization library.
|
`toJsonObject` directly, it is not intended to be used outside the serialization library.
|
||||||
|
|
||||||
## Canonical Json round-trip format
|
|
||||||
|
|
||||||
`Json.encode()` and `Json.decode()` are now the JSON equivalents of `Lynon.encode()` and `Lynon.decode()`.
|
|
||||||
|
|
||||||
They still use JSON text, but they add Lyng-specific type tags where plain JSON would otherwise lose information.
|
|
||||||
|
|
||||||
When a map already fits ordinary JSON object rules, canonical JSON keeps that traditional object shape. In particular,
|
|
||||||
maps with string keys are still serialized as JSON objects, not as tagged entry lists.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.serialization
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
enum Color { Red, Green }
|
|
||||||
class Point(x,y) { var z = 42 }
|
|
||||||
|
|
||||||
val p = Point(1,2)
|
|
||||||
p.z = 99
|
|
||||||
|
|
||||||
val value = List(
|
|
||||||
p,
|
|
||||||
Map([1, "one"], ["two", 2]),
|
|
||||||
Set(1,2,3),
|
|
||||||
"hello".encodeUtf8(),
|
|
||||||
Date(2026,4,15),
|
|
||||||
Color.Green
|
|
||||||
)
|
|
||||||
|
|
||||||
assertEquals(value, Json.decode(Json.encode(value)))
|
|
||||||
```
|
|
||||||
|
|
||||||
The canonical `Json` format is intended for Lyng-to-Lyng transfer through JSON text.
|
|
||||||
|
|
||||||
The plain `toJson()` projection is intended for ordinary JSON interop.
|
|
||||||
|
|
||||||
Canonical `Json.encode()` should be read as the JSON analogue of `Lynon.encode()`: when Lynon already preserves a
|
|
||||||
Lyng distinction, canonical JSON tries to preserve it too, using tags only where ordinary JSON is insufficient.
|
|
||||||
|
|
||||||
## Typed canonical Json round-trip format
|
|
||||||
|
|
||||||
`Json.encodeAs(Type, value)` and `Json.decodeAs(Type, text)` use the same canonical rules, but with a declared target
|
|
||||||
type available during the whole traversal.
|
|
||||||
|
|
||||||
This changes one thing only: type tags may be omitted when the declared type is already exact enough to restore the
|
|
||||||
value unambiguously.
|
|
||||||
|
|
||||||
The same map rule still applies here: `Map<String, T>` stays a normal JSON object, while non-string-key maps fall back
|
|
||||||
to canonical entry encoding.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.serialization
|
|
||||||
|
|
||||||
closed class Point(x: Int, y: Int)
|
|
||||||
closed class Segment(a: Point, b: Point)
|
|
||||||
|
|
||||||
val value = Segment(Point(0, 1), Point(2, 3))
|
|
||||||
val encoded = Json.encodeAs(Segment, value)
|
|
||||||
|
|
||||||
assertEquals("{\"a\":{\"x\":0,\"y\":1},\"b\":{\"x\":2,\"y\":3}}", encoded)
|
|
||||||
assertEquals(value, Json.decodeAs(Segment, encoded))
|
|
||||||
```
|
|
||||||
|
|
||||||
Subtype information is still preserved when the declared type is wider than the runtime one. For example, if a field is
|
|
||||||
declared as `Base` but contains `Derived`, canonical subtype tags remain in that field.
|
|
||||||
|
|
||||||
This is why the APIs are split:
|
|
||||||
|
|
||||||
- `toJson()` stays plain and interop-friendly
|
|
||||||
- `Json.encode()` stays fully self-describing and safe to decode without a schema
|
|
||||||
- `Json.encodeAs()` uses the supplied schema to reduce noise, but only where that schema is sufficient
|
|
||||||
|
|
||||||
## Kotlin side interfaces
|
## Kotlin side interfaces
|
||||||
|
|
||||||
The "Batteries included" principle is also applied to serialization.
|
The "Batteries included" principle is also applied to serialization.
|
||||||
|
|
||||||
- `Obj.toJson()` provides Kotlin `JsonElement` for the plain JSON projection
|
- `Obj.toJson()` provides Kotlin `JsonElement`
|
||||||
- `Obj.toJsonString()` provides plain JSON string representation
|
- `Obj.toJsonString()` provides Json string representation
|
||||||
- `Obj.decodeSerializableWith()` and `Obj.decodeSerializable()` allows to decode Lyng classes as Kotlin objects using
|
- `Obj.decodeSerializableWith()` and `Obj.decodeSerializable()` allows to decode Lyng classes as Kotlin objects using
|
||||||
`kotlinx.serialization`:
|
`kotlinx.serialization`:
|
||||||
|
|
||||||
@ -203,9 +104,10 @@ suspend inline fun <reified T> Obj.decodeSerializable(scope: Scope = Scope()) =
|
|||||||
decodeSerializableWith<T>(serializer<T>(), scope)
|
decodeSerializableWith<T>(serializer<T>(), scope)
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that Lyng-to-Kotlin deserialization with `kotlinx.serialization` is based on the plain JSON projection,
|
Note that lyng-2-kotlin deserialization with `kotlinx.serialization` uses JsonElement as information carrier without
|
||||||
not the canonical `Json.encode()` format. It uses `JsonElement` as the information carrier without formatting and
|
formatting and parsing actual Json strings. This is why we use `Json.decodeFromJsonElement` instead of
|
||||||
parsing actual JSON strings. This is why we use `Json.decodeFromJsonElement` instead of `Json.decodeFromString`.
|
`Json.decodeFromString`. Such an approach gives satisfactory performance without writing and supporting custom
|
||||||
|
`kotlinx.serialization` codecs.
|
||||||
|
|
||||||
### Pitfall: JSON objects and Map<String, Any?>
|
### Pitfall: JSON objects and Map<String, Any?>
|
||||||
|
|
||||||
@ -220,8 +122,7 @@ data class TestJson2(
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun deserializeMapWithJsonTest() = runTest {
|
fun deserializeMapWithJsonTest() = runTest {
|
||||||
val session = EvalSession()
|
val x = eval("""
|
||||||
val x = session.eval("""
|
|
||||||
import lyng.serialization
|
import lyng.serialization
|
||||||
{ value: 1, inner: { "foo": 1, "bar": 2 }}
|
{ value: 1, inner: { "foo": 1, "bar": 2 }}
|
||||||
""".trimIndent()).decodeSerializable<TestJson2>()
|
""".trimIndent()).decodeSerializable<TestJson2>()
|
||||||
@ -232,8 +133,7 @@ fun deserializeMapWithJsonTest() = runTest {
|
|||||||
|
|
||||||
But what if your map has objects of different types? The approach of using polymorphism is partially applicable, but what to do with `{ one: 1, two: "two" }`?
|
But what if your map has objects of different types? The approach of using polymorphism is partially applicable, but what to do with `{ one: 1, two: "two" }`?
|
||||||
|
|
||||||
The answer is simple: use `JsonObject` in your deserializable object. This class is capable of holding any JSON types
|
The answer is pretty simple: use `JsonObject` in your deserializable object. This class is capable of holding any JSON types and structures and is sort of a silver bullet for such cases:
|
||||||
and structures:
|
|
||||||
|
|
||||||
~~~kotlin
|
~~~kotlin
|
||||||
@Serializable
|
@Serializable
|
||||||
@ -243,8 +143,7 @@ data class TestJson3(
|
|||||||
)
|
)
|
||||||
@Test
|
@Test
|
||||||
fun deserializeAnyMapWithJsonTest() = runTest {
|
fun deserializeAnyMapWithJsonTest() = runTest {
|
||||||
val session = EvalSession()
|
val x = eval("""
|
||||||
val x = session.eval("""
|
|
||||||
import lyng.serialization
|
import lyng.serialization
|
||||||
{ value: 12, inner: { "foo": 1, "bar": "two" }}
|
{ value: 12, inner: { "foo": 1, "bar": "two" }}
|
||||||
""".trimIndent()).decodeSerializable<TestJson3>()
|
""".trimIndent()).decodeSerializable<TestJson3>()
|
||||||
@ -253,71 +152,27 @@ fun deserializeAnyMapWithJsonTest() = runTest {
|
|||||||
~~~
|
~~~
|
||||||
|
|
||||||
|
|
||||||
## Supported shapes
|
# List of supported types
|
||||||
|
|
||||||
### Plain JSON projection
|
|
||||||
|
|
||||||
| Lyng type | JSON type | notes |
|
| Lyng type | JSON type | notes |
|
||||||
|-----------|-----------|-------------|
|
|-----------|-----------|-------------|
|
||||||
| `Int` | number | |
|
| `Int` | number | |
|
||||||
| `Real` | number | finite values only as plain numbers |
|
| `Real` | number | |
|
||||||
| `String` | string | |
|
| `String` | string | |
|
||||||
| `Bool` | boolean | |
|
| `Bool` | boolean | |
|
||||||
| `null` | null | |
|
| `null` | null | |
|
||||||
| `Instant` | string | ISO8601 (1) |
|
| `Instant` | string | ISO8601 (1) |
|
||||||
| `List` | array | (2) |
|
| `List` | array | (2) |
|
||||||
| `Map` | object | string keys only |
|
| `Map` | object | (2) |
|
||||||
| simple class instance | object | constructor fields + mutable vars |
|
|
||||||
| enum | string | entry name |
|
|
||||||
|
|
||||||
### Canonical `Json.encode`
|
|
||||||
|
|
||||||
This format can also round-trip:
|
|
||||||
|
|
||||||
- maps with non-string keys
|
|
||||||
- sets
|
|
||||||
- immutable collections
|
|
||||||
- buffers and bit buffers
|
|
||||||
- class instances
|
|
||||||
- singleton objects
|
|
||||||
- enums
|
|
||||||
- exceptions
|
|
||||||
- `Date`, `Instant`, `DateTime`
|
|
||||||
- non-finite reals
|
|
||||||
- `void`
|
|
||||||
|
|
||||||
### Typed canonical `Json.encodeAs`
|
|
||||||
|
|
||||||
This format round-trips the same value space as canonical `Json.encode`, but it can emit simpler JSON for:
|
|
||||||
|
|
||||||
- closed classes and other exactly-known class fields
|
|
||||||
- enums when the enum type is known
|
|
||||||
- typed collections whose element types are known
|
|
||||||
- nested object graphs where declared field types are precise
|
|
||||||
|
|
||||||
It still falls back to canonical tagged encoding when exact runtime type information would otherwise be lost.
|
|
||||||
|
|
||||||
It does so by adding Lyng-specific type tags only when necessary.
|
|
||||||
|
|
||||||
## Kotlin-side extension point for more formats
|
|
||||||
|
|
||||||
Additional formats can be exported from Kotlin modules by subclassing `ObjSerializationFormatClass` and registering the
|
|
||||||
format in module scope with `bindSerializationFormat(...)`.
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
module.bindSerializationFormat(
|
|
||||||
object : ObjSerializationFormatClass("MyFormat") {
|
|
||||||
override suspend fun encodeValue(scope: Scope, value: Obj): Obj = ...
|
|
||||||
override suspend fun decodeValue(scope: Scope, encoded: Obj): Obj = ...
|
|
||||||
}
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
This makes `MyFormat.encode(...)` and `MyFormat.decode(...)` available from Lyng after importing the module.
|
|
||||||
|
|
||||||
(1)
|
(1)
|
||||||
: ISO8601 flavor `1970-05-06T06:00:00.000Z` is used; number of fractional digits depends on truncation on
|
: ISO8601 flavor 1970-05-06T06:00:00.000Z in used; number of fractional digits depends on the truncation
|
||||||
`Instant`, see `Instant.truncateTo...` functions.
|
on [Instant](time.md), see `Instant.truncateTo...` functions.
|
||||||
|
|
||||||
(2)
|
(2)
|
||||||
: Lists may contain any values serializable by the selected JSON layer.
|
: List may contain any objects serializable to Json.
|
||||||
|
|
||||||
|
(3)
|
||||||
|
: Map keys must be strings, map values may be any objects serializable to Json.
|
||||||
|
|
||||||
|
|||||||
@ -1,113 +0,0 @@
|
|||||||
### lyng.io.console
|
|
||||||
|
|
||||||
`lyng.io.console` provides optional rich console support for terminal applications.
|
|
||||||
|
|
||||||
> **Note:** this module is part of `lyngio`. It must be explicitly installed into the import manager by host code.
|
|
||||||
>
|
|
||||||
> **CLI note:** the `lyng` CLI now installs `lyng.io.console` in its base scope by default, so scripts can simply `import lyng.io.console`.
|
|
||||||
|
|
||||||
#### Install in host
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.io.console.createConsoleModule
|
|
||||||
import net.sergeych.lyngio.console.security.PermitAllConsoleAccessPolicy
|
|
||||||
|
|
||||||
suspend fun initScope() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope = session.getScope()
|
|
||||||
createConsoleModule(PermitAllConsoleAccessPolicy, scope)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Use in Lyng script
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.console
|
|
||||||
|
|
||||||
println("supported = " + Console.isSupported())
|
|
||||||
println("tty = " + Console.isTty())
|
|
||||||
println("ansi = " + Console.ansiLevel())
|
|
||||||
println("geometry = " + Console.geometry())
|
|
||||||
|
|
||||||
Console.write("hello\n")
|
|
||||||
Console.home()
|
|
||||||
Console.clear()
|
|
||||||
Console.moveTo(1, 1)
|
|
||||||
Console.clearLine()
|
|
||||||
Console.enterAltScreen()
|
|
||||||
Console.leaveAltScreen()
|
|
||||||
Console.setCursorVisible(true)
|
|
||||||
Console.flush()
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Tetris sample
|
|
||||||
|
|
||||||
The repository includes a full interactive Tetris sample that demonstrates:
|
|
||||||
|
|
||||||
- alternate screen rendering
|
|
||||||
- raw keyboard input
|
|
||||||
- resize handling
|
|
||||||
- typed console events
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
Run it from the project root in a real TTY:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
lyng examples/tetris_console.lyng
|
|
||||||
```
|
|
||||||
|
|
||||||
#### API
|
|
||||||
|
|
||||||
- `Console.isSupported(): Bool` — whether console control is available on this platform/runtime.
|
|
||||||
- `Console.isTty(): Bool` — whether output is attached to a TTY.
|
|
||||||
- `Console.ansiLevel(): ConsoleAnsiLevel` — `NONE`, `BASIC16`, `ANSI256`, `TRUECOLOR`.
|
|
||||||
- `Console.geometry(): ConsoleGeometry?` — `{columns, rows}` as typed object or `null`.
|
|
||||||
- `Console.details(): ConsoleDetails` — consolidated capability object.
|
|
||||||
- `Console.write(text: String)` — writes to console output.
|
|
||||||
- `Console.flush()` — flushes buffered output.
|
|
||||||
- `Console.home()` — moves cursor to top-left.
|
|
||||||
- `Console.clear()` — clears visible screen.
|
|
||||||
- `Console.moveTo(row: Int, column: Int)` — moves cursor to 1-based row/column.
|
|
||||||
- `Console.clearLine()` — clears current line.
|
|
||||||
- `Console.enterAltScreen()` — switch to alternate screen buffer.
|
|
||||||
- `Console.leaveAltScreen()` — return to normal screen buffer.
|
|
||||||
- `Console.setCursorVisible(visible: Bool)` — shows/hides cursor.
|
|
||||||
- `Console.events(): ConsoleEventStream` — endless iterable source of typed events: `ConsoleResizeEvent`, `ConsoleKeyEvent`.
|
|
||||||
- `Console.setRawMode(enabled: Bool): Bool` — requests raw input mode, returns `true` if changed.
|
|
||||||
|
|
||||||
#### Event Iteration
|
|
||||||
|
|
||||||
Use events from a loop, typically in a separate coroutine:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
launch {
|
|
||||||
for (ev in Console.events()) {
|
|
||||||
if (ev is ConsoleKeyEvent) {
|
|
||||||
// handle key
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Event format
|
|
||||||
|
|
||||||
`Console.events()` emits `ConsoleEvent` with:
|
|
||||||
|
|
||||||
- `type: ConsoleEventType` — `UNKNOWN`, `RESIZE`, `KEY_DOWN`, `KEY_UP`
|
|
||||||
|
|
||||||
Additional fields:
|
|
||||||
|
|
||||||
- `ConsoleResizeEvent`: `columns`, `rows`
|
|
||||||
- `ConsoleKeyEvent`: `key`, `code`, `ctrl`, `alt`, `shift`, `meta`
|
|
||||||
|
|
||||||
#### Security policy
|
|
||||||
|
|
||||||
The module uses `ConsoleAccessPolicy` with operations:
|
|
||||||
|
|
||||||
- `WriteText(length)`
|
|
||||||
- `ReadEvents`
|
|
||||||
- `SetRawMode(enabled)`
|
|
||||||
|
|
||||||
For permissive mode, use `PermitAllConsoleAccessPolicy`.
|
|
||||||
@ -1,565 +0,0 @@
|
|||||||
# lyng.io.db — SQL database access for Lyng scripts
|
|
||||||
|
|
||||||
This module provides the portable SQL database contract for Lyng. The current shipped providers are SQLite via `lyng.io.db.sqlite` and a JVM-only JDBC bridge via `lyng.io.db.jdbc`.
|
|
||||||
|
|
||||||
> **Note:** `lyngio` is a separate library module. It must be explicitly added as a dependency to your host application and initialized in your Lyng scopes.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Install the module into a Lyng session
|
|
||||||
|
|
||||||
For SQLite-backed database access, install both the generic DB module and the SQLite provider:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
|
||||||
import net.sergeych.lyng.io.db.createDbModule
|
|
||||||
import net.sergeych.lyng.io.db.sqlite.createSqliteModule
|
|
||||||
|
|
||||||
suspend fun bootstrapDb() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope: Scope = session.getScope()
|
|
||||||
createDbModule(scope)
|
|
||||||
createSqliteModule(scope)
|
|
||||||
session.eval("""
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
""".trimIndent())
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
`createSqliteModule(...)` also registers the `sqlite:` scheme for generic `openDatabase(...)`.
|
|
||||||
|
|
||||||
For JVM JDBC-backed access, install the JDBC provider as well:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
|
||||||
import net.sergeych.lyng.io.db.createDbModule
|
|
||||||
import net.sergeych.lyng.io.db.jdbc.createJdbcModule
|
|
||||||
|
|
||||||
suspend fun bootstrapJdbc() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope: Scope = session.getScope()
|
|
||||||
createDbModule(scope)
|
|
||||||
createJdbcModule(scope)
|
|
||||||
session.eval("""
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.jdbc
|
|
||||||
""".trimIndent())
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
`createJdbcModule(...)` registers `jdbc:`, `h2:`, `postgres:`, and `postgresql:` for `openDatabase(...)`.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Using from Lyng scripts
|
|
||||||
|
|
||||||
Typed SQLite open helper:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
|
|
||||||
val db = openSqlite(":memory:")
|
|
||||||
|
|
||||||
val userCount = db.transaction { tx ->
|
|
||||||
tx.execute("create table user(id integer primary key autoincrement, name text not null)")
|
|
||||||
tx.execute("insert into user(name) values(?)", "Ada")
|
|
||||||
tx.execute("insert into user(name) values(?)", "Linus")
|
|
||||||
tx.select("select count(*) as count from user").toList()[0]["count"]
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(2, userCount)
|
|
||||||
```
|
|
||||||
|
|
||||||
Generic provider-based open:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
|
|
||||||
val db = openDatabase(
|
|
||||||
"sqlite:./app.db",
|
|
||||||
Map(
|
|
||||||
"foreignKeys" => true,
|
|
||||||
"busyTimeoutMillis" => 5000
|
|
||||||
)
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
JVM JDBC open with H2:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db.jdbc
|
|
||||||
|
|
||||||
val db = openH2("mem:demo;DB_CLOSE_DELAY=-1")
|
|
||||||
|
|
||||||
val names = db.transaction { tx ->
|
|
||||||
tx.execute("create table person(id bigint auto_increment primary key, name varchar(120) not null)")
|
|
||||||
tx.execute("insert into person(name) values(?)", "Ada")
|
|
||||||
tx.execute("insert into person(name) values(?)", "Linus")
|
|
||||||
tx.select("select name from person order by id").toList()
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals("Ada", names[0]["name"])
|
|
||||||
assertEquals("Linus", names[1]["name"])
|
|
||||||
```
|
|
||||||
|
|
||||||
Generic JDBC open through `openDatabase(...)`:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.jdbc
|
|
||||||
|
|
||||||
val db = openDatabase(
|
|
||||||
"jdbc:h2:mem:demo2;DB_CLOSE_DELAY=-1",
|
|
||||||
Map()
|
|
||||||
)
|
|
||||||
|
|
||||||
val answer = db.transaction { tx ->
|
|
||||||
tx.select("select 42 as answer").toList()[0]["answer"]
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(42, answer)
|
|
||||||
```
|
|
||||||
|
|
||||||
PostgreSQL typed open:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db.jdbc
|
|
||||||
|
|
||||||
val db = openPostgres(
|
|
||||||
"jdbc:postgresql://127.0.0.1/appdb",
|
|
||||||
"appuser",
|
|
||||||
"secret"
|
|
||||||
)
|
|
||||||
|
|
||||||
val titles = db.transaction { tx ->
|
|
||||||
tx.execute("create table if not exists task(id bigserial primary key, title text not null)")
|
|
||||||
tx.execute("insert into task(title) values(?)", "Ship JDBC provider")
|
|
||||||
tx.execute("insert into task(title) values(?)", "Test PostgreSQL path")
|
|
||||||
tx.select("select title from task order by id").toList()
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals("Ship JDBC provider", titles[0]["title"])
|
|
||||||
```
|
|
||||||
|
|
||||||
Nested transactions use real savepoint semantics:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
|
|
||||||
val db = openSqlite(":memory:")
|
|
||||||
|
|
||||||
db.transaction { tx ->
|
|
||||||
tx.execute("create table item(id integer primary key autoincrement, name text not null)")
|
|
||||||
tx.execute("insert into item(name) values(?)", "outer")
|
|
||||||
|
|
||||||
try {
|
|
||||||
tx.transaction { inner ->
|
|
||||||
inner.execute("insert into item(name) values(?)", "inner")
|
|
||||||
throw IllegalStateException("rollback nested")
|
|
||||||
}
|
|
||||||
} catch (_: IllegalStateException) {
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(1, tx.select("select count(*) as count from item").toList()[0]["count"])
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Intentional rollback without treating it as a backend failure:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
|
|
||||||
val db = openSqlite(":memory:")
|
|
||||||
|
|
||||||
assertThrows(RollbackException) {
|
|
||||||
db.transaction { tx ->
|
|
||||||
tx.execute("create table item(id integer primary key autoincrement, name text not null)")
|
|
||||||
tx.execute("insert into item(name) values(?)", "temporary")
|
|
||||||
throw RollbackException("stop here")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Runnable serialization sample
|
|
||||||
|
|
||||||
A complete runnable example is in [examples/sqlite_serialization.lyng](../examples/sqlite_serialization.lyng).
|
|
||||||
|
|
||||||
It uses:
|
|
||||||
|
|
||||||
- `@DbJson`
|
|
||||||
- `@DbLynon`
|
|
||||||
- `@DbExcept`
|
|
||||||
- `@cols(...)`, `@vals(...)`, `@set(...)`
|
|
||||||
- `decodeAs<T>()`
|
|
||||||
|
|
||||||
The current direct read form that works under `jlyng` is:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
tx.select("select * from item where id = ?", 1).decodeAs<Item>().first
|
|
||||||
```
|
|
||||||
|
|
||||||
If we want a shorter form such as:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
tx.selectAllAs<Item>("item where id = ?", 1).first
|
|
||||||
```
|
|
||||||
|
|
||||||
it should be added as a built-in `SqlTransaction` API. A pure Lyng generic wrapper around `decodeAs<T>()` does not currently preserve `T` reliably enough under `jlyng`.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Portable API
|
|
||||||
|
|
||||||
### `Database`
|
|
||||||
|
|
||||||
- `transaction(block)` — opens a transaction, commits on normal exit, rolls back on uncaught failure.
|
|
||||||
|
|
||||||
### `SqlTransaction`
|
|
||||||
|
|
||||||
- `select(clause, params...)` — execute a statement whose primary result is a row set.
|
|
||||||
- `execute(clause, params...)` — execute a side-effect statement and return `ExecutionResult`.
|
|
||||||
- `transaction(block)` — nested transaction with real savepoint semantics.
|
|
||||||
|
|
||||||
`select(...)` and `execute(...)` also support SQL object-expansion macros for declaration-driven writes:
|
|
||||||
|
|
||||||
- `@cols(?1)` — expand object argument `?1` to a comma-separated column list
|
|
||||||
- `@vals(?1)` — expand object argument `?1` to matching placeholders and bind values
|
|
||||||
- `@set(?1)` — expand object argument `?1` to `column = ?` pairs and bind values
|
|
||||||
|
|
||||||
Each macro also supports an optional clause-local exclusion list:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
tx.execute("update item set @set(?1 except: \"id\", \"createdAt\") where id = ?2", item, item.id)
|
|
||||||
```
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
tx.execute("insert into item(@cols(?1)) values(@vals(?1))", item)
|
|
||||||
tx.execute("update item set @set(?1) where id = ?2", item, item.id)
|
|
||||||
```
|
|
||||||
|
|
||||||
When a clause uses any of these macros, non-expanded scalar parameters in the same SQL string must use explicit indexed placeholders such as `?2`, `?3`, and so on.
|
|
||||||
|
|
||||||
### `ResultSet`
|
|
||||||
|
|
||||||
- `columns` — positional `SqlColumn` metadata, available before iteration.
|
|
||||||
- `size()` — result row count.
|
|
||||||
- `isEmpty()` — fast emptiness check where possible.
|
|
||||||
- `iterator()` — normal row iteration while the transaction is active.
|
|
||||||
- `toList()` — materialize detached `SqlRow` snapshots that may be used after the transaction ends.
|
|
||||||
- `decodeAs<T>()` — transaction-scoped iterable view that decodes each row into `T`.
|
|
||||||
|
|
||||||
### `SqlRow`
|
|
||||||
|
|
||||||
- `row[index]` — zero-based positional access.
|
|
||||||
- `row["columnName"]` — case-insensitive lookup by output column label.
|
|
||||||
- `row.decodeAs<T>()` — decode one row into a typed Lyng value.
|
|
||||||
|
|
||||||
Name-based access fails with `SqlUsageException` if the name is missing or ambiguous.
|
|
||||||
|
|
||||||
### `DbFieldAdapter`
|
|
||||||
|
|
||||||
Custom DB field projection hook used by `@DbDecodeWith(...)` and `@DbSerializeWith(...)`.
|
|
||||||
|
|
||||||
- `decode(rawValue, column, row, targetType)` — adapt one raw DB field value to a Lyng value for the requested target type.
|
|
||||||
- `encode(value, targetType)` — adapt one Lyng value to a direct DB-bindable value for SQL object expansion.
|
|
||||||
|
|
||||||
Use `@DbDecodeWith(adapter)` on class constructor parameters and class-body fields/properties that participate in `decodeAs<T>()`.
|
|
||||||
|
|
||||||
Use `@DbSerializeWith(adapter)` on constructor parameters and class-body fields/properties that participate in `@cols(...)`, `@vals(...)`, and `@set(...)` object expansion.
|
|
||||||
|
|
||||||
Annotation arguments are evaluated once when the declaration is created, and the resulting adapter instance is retained in declaration metadata.
|
|
||||||
|
|
||||||
### `ExecutionResult`
|
|
||||||
|
|
||||||
- `affectedRowsCount`
|
|
||||||
- `getGeneratedKeys()`
|
|
||||||
|
|
||||||
Statements that return rows directly, such as `... returning ...`, should use `select(...)`, not `execute(...)`.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Value mapping
|
|
||||||
|
|
||||||
Portable bind values:
|
|
||||||
|
|
||||||
- `null`
|
|
||||||
- `Bool`
|
|
||||||
- `Int`, `Double`, `Decimal`
|
|
||||||
- `String`
|
|
||||||
- `Buffer`
|
|
||||||
- `Date`, `DateTime`, `Instant`
|
|
||||||
|
|
||||||
Unsupported parameter values fail with `SqlUsageException`.
|
|
||||||
|
|
||||||
SQL object-expansion write rules:
|
|
||||||
|
|
||||||
- constructor parameters participate in projection by declaration order
|
|
||||||
- matching serializable class-body fields/properties also participate
|
|
||||||
- `@Transient` fields are excluded automatically
|
|
||||||
- `@DbExcept` fields are excluded automatically
|
|
||||||
- `except:` excludes additional fields for one specific macro use
|
|
||||||
- direct DB-bindable values are written as-is
|
|
||||||
- `@DbJson` fields are encoded as canonical JSON text
|
|
||||||
- `@DbLynon` fields are encoded as Lynon binary
|
|
||||||
- `@DbSerializeWith(adapter)` fields are encoded through the adapter
|
|
||||||
- unannotated non-bindable object fields fail with `SqlUsageException`
|
|
||||||
|
|
||||||
Write-side encoding is intentionally explicit. The runtime does not try to infer target DB column types from SQL text or backend metadata during statement preparation.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
|
|
||||||
class Payload(name: String, count: Int)
|
|
||||||
|
|
||||||
object TrimAdapter: DbFieldAdapter {
|
|
||||||
override fun encode(value, targetType) =
|
|
||||||
when(value) {
|
|
||||||
null -> null
|
|
||||||
else -> value.toString().trim()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class Item(
|
|
||||||
id: Int,
|
|
||||||
@DbSerializeWith(TrimAdapter) title: String,
|
|
||||||
@DbJson meta: Payload,
|
|
||||||
@DbLynon state: Payload
|
|
||||||
) {
|
|
||||||
var note: String = ""
|
|
||||||
@DbExcept var cache: String = ""
|
|
||||||
}
|
|
||||||
|
|
||||||
val db = openSqlite(":memory:")
|
|
||||||
val restored = db.transaction { tx ->
|
|
||||||
tx.execute(
|
|
||||||
"create table item(id integer not null, title text not null, meta text not null, state blob not null, note text not null)"
|
|
||||||
)
|
|
||||||
|
|
||||||
val item = Item(1, " first ", Payload("json", 10), Payload("bin", 20))
|
|
||||||
item.note = "created"
|
|
||||||
item.cache = "not stored"
|
|
||||||
|
|
||||||
tx.execute("insert into item(@cols(?1)) values(@vals(?1))", item)
|
|
||||||
|
|
||||||
item.title = " second "
|
|
||||||
item.meta = Payload("json2", 11)
|
|
||||||
item.state = Payload("bin2", 21)
|
|
||||||
item.note = "updated"
|
|
||||||
|
|
||||||
tx.execute(
|
|
||||||
"update item set @set(?1 except: \"id\") where id = ?2",
|
|
||||||
item,
|
|
||||||
item.id
|
|
||||||
)
|
|
||||||
|
|
||||||
tx.select("select id, title, meta, state, note from item").decodeAs<Item>().first
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals("second", restored.title)
|
|
||||||
assertEquals("json2", restored.meta.name)
|
|
||||||
assertEquals(21, restored.state.count)
|
|
||||||
assertEquals("updated", restored.note)
|
|
||||||
```
|
|
||||||
|
|
||||||
This example shows:
|
|
||||||
|
|
||||||
- `@DbSerializeWith(...)` trimming a string before write
|
|
||||||
- `@DbJson` storing structured data in a text column
|
|
||||||
- `@DbLynon` storing structured data in a binary column
|
|
||||||
- `@DbExcept` excluding a field from automatic projection
|
|
||||||
- `@set(... except: "id")` skipping one field for an update clause
|
|
||||||
- `decodeAs<Item>()` reconstructing the object on read
|
|
||||||
|
|
||||||
Portable result metadata categories:
|
|
||||||
|
|
||||||
- `Binary`
|
|
||||||
- `String`
|
|
||||||
- `Int`
|
|
||||||
- `Double`
|
|
||||||
- `Decimal`
|
|
||||||
- `Bool`
|
|
||||||
- `Date`
|
|
||||||
- `DateTime`
|
|
||||||
- `Instant`
|
|
||||||
|
|
||||||
Typed row decode rules:
|
|
||||||
|
|
||||||
- object/class targets map constructor parameters by column label, case-insensitively
|
|
||||||
- remaining matching serializable mutable fields are assigned after constructor call
|
|
||||||
- `@DbDecodeWith(adapter)` on a constructor parameter or class-body field/property takes precedence over built-in JSON/Lynon decoding
|
|
||||||
- `@DbDecodeWith(adapter)` must receive exactly one adapter instance implementing `DbFieldAdapter`
|
|
||||||
- adapter output must match the target member type or decoding fails with `SqlUsageException`
|
|
||||||
- missing required non-null constructor fields fail
|
|
||||||
- defaulted or nullable constructor fields may be omitted from the result
|
|
||||||
- extra result columns currently fail in strict mode
|
|
||||||
- if a row has exactly one column, that value may be decoded directly as the requested target type
|
|
||||||
- JSON-like native column types (`json`, `jsonb`) are decoded through typed canonical `Json` when the target type is not `String`
|
|
||||||
- binary columns are decoded through `Lynon` when the target type is not `Buffer`
|
|
||||||
- `Buffer` targets keep the raw binary payload without Lynon decoding
|
|
||||||
- plain text columns are not implicitly treated as JSON
|
|
||||||
|
|
||||||
For temporal types, see [time functions](time.md).
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## SQLite provider
|
|
||||||
|
|
||||||
`lyng.io.db.sqlite` currently provides the first concrete backend.
|
|
||||||
|
|
||||||
Typed helper:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
openSqlite(
|
|
||||||
path: String,
|
|
||||||
readOnly: Bool = false,
|
|
||||||
createIfMissing: Bool = true,
|
|
||||||
foreignKeys: Bool = true,
|
|
||||||
busyTimeoutMillis: Int = 5000
|
|
||||||
): Database
|
|
||||||
```
|
|
||||||
|
|
||||||
Accepted generic URL forms:
|
|
||||||
|
|
||||||
- `sqlite::memory:`
|
|
||||||
- `sqlite:relative/path.db`
|
|
||||||
- `sqlite:/absolute/path.db`
|
|
||||||
|
|
||||||
Supported `openDatabase(..., extraParams)` keys for SQLite:
|
|
||||||
|
|
||||||
- `readOnly: Bool`
|
|
||||||
- `createIfMissing: Bool`
|
|
||||||
- `foreignKeys: Bool`
|
|
||||||
- `busyTimeoutMillis: Int`
|
|
||||||
|
|
||||||
SQLite write/read policy in v1:
|
|
||||||
|
|
||||||
- `Bool` writes as `0` / `1`
|
|
||||||
- `Decimal` writes as canonical text
|
|
||||||
- `Date` writes as `YYYY-MM-DD`
|
|
||||||
- `DateTime` writes as ISO local timestamp text without timezone
|
|
||||||
- `Instant` writes as ISO UTC timestamp text with explicit timezone marker
|
|
||||||
- `TIME*` values stay `String`
|
|
||||||
- `TIMESTAMP` / `DATETIME` reject timezone-bearing stored text
|
|
||||||
|
|
||||||
Open-time validation failures:
|
|
||||||
|
|
||||||
- malformed URL or bad option shape -> `IllegalArgumentException`
|
|
||||||
- runtime open failure -> `DatabaseException`
|
|
||||||
|
|
||||||
## JDBC provider
|
|
||||||
|
|
||||||
`lyng.io.db.jdbc` is currently implemented on the JVM target only. The `lyngio-jvm` artifact bundles and explicitly loads these JDBC drivers:
|
|
||||||
|
|
||||||
- SQLite
|
|
||||||
- H2
|
|
||||||
- PostgreSQL
|
|
||||||
|
|
||||||
Typed helpers:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
openJdbc(
|
|
||||||
connectionUrl: String,
|
|
||||||
user: String? = null,
|
|
||||||
password: String? = null,
|
|
||||||
driverClass: String? = null,
|
|
||||||
properties: Map<String, Object?>? = null
|
|
||||||
): Database
|
|
||||||
|
|
||||||
openH2(
|
|
||||||
connectionUrl: String,
|
|
||||||
user: String? = null,
|
|
||||||
password: String? = null,
|
|
||||||
properties: Map<String, Object?>? = null
|
|
||||||
): Database
|
|
||||||
|
|
||||||
openPostgres(
|
|
||||||
connectionUrl: String,
|
|
||||||
user: String? = null,
|
|
||||||
password: String? = null,
|
|
||||||
properties: Map<String, Object?>? = null
|
|
||||||
): Database
|
|
||||||
```
|
|
||||||
|
|
||||||
Accepted generic URL forms:
|
|
||||||
|
|
||||||
- `jdbc:h2:mem:test;DB_CLOSE_DELAY=-1`
|
|
||||||
- `h2:mem:test;DB_CLOSE_DELAY=-1`
|
|
||||||
- `jdbc:postgresql://localhost/app`
|
|
||||||
- `postgres://localhost/app`
|
|
||||||
- `postgresql://localhost/app`
|
|
||||||
|
|
||||||
Supported `openDatabase(..., extraParams)` keys for JDBC:
|
|
||||||
|
|
||||||
- `driverClass: String`
|
|
||||||
- `user: String`
|
|
||||||
- `password: String`
|
|
||||||
- `properties: Map<String, Object?>`
|
|
||||||
|
|
||||||
Behavior notes for the JDBC bridge:
|
|
||||||
|
|
||||||
- the portable `Database` / `SqlTransaction` API stays the same as for SQLite
|
|
||||||
- nested transactions use JDBC savepoints
|
|
||||||
- JDBC connection properties are built from `user`, `password`, and `properties`
|
|
||||||
- `properties` values are stringified before being passed to JDBC
|
|
||||||
- statements with row-returning clauses still must use `select(...)`, not `execute(...)`
|
|
||||||
|
|
||||||
Platform support for this provider:
|
|
||||||
|
|
||||||
- `lyng.io.db.jdbc` — JVM only
|
|
||||||
- `openH2(...)` — works out of the box with `lyngio-jvm`
|
|
||||||
- `openPostgres(...)` — driver included, but an actual PostgreSQL server is still required
|
|
||||||
|
|
||||||
PostgreSQL-specific notes:
|
|
||||||
|
|
||||||
- `openPostgres(...)` accepts either a full JDBC URL or shorthand forms such as `//localhost/app`
|
|
||||||
- local peer/trust setups may use an empty password string
|
|
||||||
- generated keys work with PostgreSQL `bigserial` / identity columns through `ExecutionResult.getGeneratedKeys()`
|
|
||||||
- for reproducible automated tests, prefer a disposable PostgreSQL instance such as Docker/Testcontainers instead of a long-lived shared server
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Lifetime rules
|
|
||||||
|
|
||||||
`ResultSet` is valid only while its owning transaction is active.
|
|
||||||
|
|
||||||
`SqlRow` values are detached snapshots once materialized, so this pattern is valid:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val rows = db.transaction { tx ->
|
|
||||||
tx.select("select name from person order by id").toList()
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals("Ada", rows[0]["name"])
|
|
||||||
```
|
|
||||||
|
|
||||||
This means:
|
|
||||||
|
|
||||||
- do not keep `ResultSet` objects after the transaction block returns
|
|
||||||
- materialize rows with `toList()` inside the transaction when they must outlive it
|
|
||||||
- the iterable returned by `decodeAs<T>()` is also transaction-scoped
|
|
||||||
- decoded objects produced while iterating `decodeAs<T>()` are detached ordinary Lyng values
|
|
||||||
|
|
||||||
The same rule applies to generated keys from `ExecutionResult.getGeneratedKeys()`: the `ResultSet` is transaction-scoped, but rows returned by `toList()` are detached.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Platform support
|
|
||||||
|
|
||||||
- `lyng.io.db` — generic contract, available when host code installs it
|
|
||||||
- `lyng.io.db.sqlite` — implemented on JVM and Linux Native in the current release tree
|
|
||||||
- `lyng.io.db.jdbc` — implemented on JVM in the current release tree
|
|
||||||
|
|
||||||
For the broader I/O overview, see [lyngio overview](lyngio.md).
|
|
||||||
@ -39,27 +39,23 @@ This brings in:
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
#### Install the module into a Lyng session
|
#### Install the module into a Lyng Scope
|
||||||
|
|
||||||
The filesystem module is not installed automatically. The preferred host runtime is `EvalSession`: create the session, get its underlying scope, install the module there, and execute scripts through the session. You can customize access control via `FsAccessPolicy`.
|
The filesystem module is not installed automatically. You must explicitly register it in the scope’s `ImportManager` using the installer. You can customize access control via `FsAccessPolicy`.
|
||||||
|
|
||||||
Kotlin (host) bootstrap example:
|
Kotlin (host) bootstrap example:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
import net.sergeych.lyng.Scope
|
||||||
import net.sergeych.lyng.io.fs.createFs
|
import net.sergeych.lyng.io.fs.createFs
|
||||||
import net.sergeych.lyngio.fs.security.PermitAllAccessPolicy
|
import net.sergeych.lyngio.fs.security.PermitAllAccessPolicy
|
||||||
|
|
||||||
suspend fun bootstrapFs() {
|
val scope: Scope = Scope.new()
|
||||||
val session = EvalSession()
|
val installed: Boolean = createFs(PermitAllAccessPolicy, scope)
|
||||||
val scope: Scope = session.getScope()
|
// installed == true on first registration in this ImportManager, false on repeats
|
||||||
val installed: Boolean = createFs(PermitAllAccessPolicy, scope)
|
|
||||||
// installed == true on first registration in this ImportManager, false on repeats
|
|
||||||
|
|
||||||
// In scripts (or via session.eval), import the module to use its symbols:
|
// In scripts (or via scope.eval), import the module to use its symbols:
|
||||||
session.eval("import lyng.io.fs")
|
scope.eval("import lyng.io.fs")
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
You can install with a custom policy too (see Access policy below).
|
You can install with a custom policy too (see Access policy below).
|
||||||
@ -189,7 +185,7 @@ val denyWrites = object : FsAccessPolicy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
createFs(denyWrites, scope)
|
createFs(denyWrites, scope)
|
||||||
session.eval("import lyng.io.fs")
|
scope.eval("import lyng.io.fs")
|
||||||
```
|
```
|
||||||
|
|
||||||
Composite operations like `copy` and `move` are checked as a set of primitives (e.g., `OpenRead(src)` + `Delete(dst)` if overwriting + `CreateFile(dst)` + `OpenWrite(dst)`).
|
Composite operations like `copy` and `move` are checked as a set of primitives (e.g., `OpenRead(src)` + `Delete(dst)` if overwriting + `CreateFile(dst)` + `OpenWrite(dst)`).
|
||||||
|
|||||||
@ -1,164 +0,0 @@
|
|||||||
# lyng.io.html
|
|
||||||
|
|
||||||
`lyng.io.html` provides a pure Lyng HTML builder DSL. It uses Lyng context
|
|
||||||
receiver extensions, so text can be appended with `+"text"` inside tag blocks
|
|
||||||
without global builder state.
|
|
||||||
|
|
||||||
Host code installs the package from `lyngio` with `createHtmlModule(...)`:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
val scope = Script.newScope()
|
|
||||||
createHtmlModule(scope.importManager)
|
|
||||||
```
|
|
||||||
|
|
||||||
Lyng code can then import it:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.html
|
|
||||||
|
|
||||||
val page = html {
|
|
||||||
head {
|
|
||||||
title { +"Demo" }
|
|
||||||
}
|
|
||||||
body {
|
|
||||||
nav {
|
|
||||||
a(href: "/") { +"Home" }
|
|
||||||
}
|
|
||||||
h3 { +"Heading 3" }
|
|
||||||
p {
|
|
||||||
attr("data-id", 123)
|
|
||||||
+"Text is escaped: <safe>"
|
|
||||||
}
|
|
||||||
img(src: "/logo.png", alt: "Logo")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
`html { ... }` returns a `String` beginning with `<!doctype html>`.
|
|
||||||
|
|
||||||
## Escaping
|
|
||||||
|
|
||||||
Text appended with unary `+` is HTML-escaped:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
html {
|
|
||||||
body {
|
|
||||||
p { +"Text & <more>" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
produces:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<!doctype html><html><body><p>Text & <more></p></body></html>
|
|
||||||
```
|
|
||||||
|
|
||||||
Attribute values are escaped with HTML attribute rules:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
p {
|
|
||||||
attr("data-x", "\"quoted\" & <tag>")
|
|
||||||
+"content"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Use `raw(...)` only for trusted markup:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
div {
|
|
||||||
raw("<span>already escaped or trusted</span>")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Tag Helpers
|
|
||||||
|
|
||||||
Current tag helpers cover common structural tags (`head`, `body`, `main`,
|
|
||||||
`section`, `article`, `header`, `footer`, `nav`, `div`, `span`, `p`), headings
|
|
||||||
(`h1` through `h6`), lists (`ul`, `ol`, `li`), and text/code tags (`strong`,
|
|
||||||
`em`, `code`, `pre`, `script`, `style`).
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
body {
|
|
||||||
main {
|
|
||||||
section {
|
|
||||||
h2 { +"News" }
|
|
||||||
p { +"First item" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Common void tags are also available: `meta`, `link`, `img`, `br`, and `input`.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
head {
|
|
||||||
meta { attr("charset", "utf-8") }
|
|
||||||
link {
|
|
||||||
attr("rel", "stylesheet")
|
|
||||||
attr("href", "/site.css")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Attributes
|
|
||||||
|
|
||||||
Use `attr(name, value)` inside a tag block to set an escaped attribute value.
|
|
||||||
`id(...)` and `classes(...)` are small aliases:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
div {
|
|
||||||
id("root")
|
|
||||||
classes("app shell")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Use `flag(name)` for boolean attributes:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
input {
|
|
||||||
attr("type", "checkbox")
|
|
||||||
flag("checked")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Convenience Helpers
|
|
||||||
|
|
||||||
Convenience helpers include `metaCharset()`, `stylesheet(href)`,
|
|
||||||
`a(href) { ... }`, `img(src, alt)`, and `input(type, name, value)`.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
head {
|
|
||||||
metaCharset()
|
|
||||||
stylesheet("/site.css")
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
|
||||||
nav {
|
|
||||||
a(href: "/home") { +"Home" }
|
|
||||||
}
|
|
||||||
img(src: "/logo.png", alt: "Logo & mark")
|
|
||||||
input(type: "hidden", name: "token", value: "abc")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Generic Elements
|
|
||||||
|
|
||||||
Use `tag(name) { ... }` and `voidTag(name) { ... }` for elements that do not
|
|
||||||
have dedicated helpers yet:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
body {
|
|
||||||
tag("custom-element") {
|
|
||||||
flag("hidden")
|
|
||||||
+"Secret"
|
|
||||||
}
|
|
||||||
voidTag("source") {
|
|
||||||
attr("srcset", "/image.webp")
|
|
||||||
attr("type", "image/webp")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
These helpers are intentionally simple escape hatches. Prefer a dedicated helper
|
|
||||||
when one exists because it can encode safer defaults and clearer parameter names.
|
|
||||||
@ -1,181 +0,0 @@
|
|||||||
# lyng.io.http — HTTP/HTTPS client for Lyng scripts
|
|
||||||
|
|
||||||
This module provides a compact HTTP client API for Lyng scripts. It is implemented in `lyngio` and backed by Ktor on supported runtimes.
|
|
||||||
|
|
||||||
> **Note:** `lyngio` is a separate library module. It must be explicitly added as a dependency to your host application and initialized in your Lyng scopes.
|
|
||||||
>
|
|
||||||
> **Shared type note:** `HttpHeaders` is also available from `lyng.io.http.types` when host code wants the reusable value type without relying on the HTTP client module itself.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Add the library to your project (Gradle)
|
|
||||||
|
|
||||||
If you use this repository as a multi-module project, add a dependency on `:lyngio`:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
dependencies {
|
|
||||||
implementation("net.sergeych:lyngio:0.0.1-SNAPSHOT")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
For external projects, ensure you also use the Lyng Maven repository described in `lyng.io.fs`.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Install the module into a Lyng session
|
|
||||||
|
|
||||||
The HTTP module is not installed automatically. Install it into the session scope and provide a policy.
|
|
||||||
|
|
||||||
Kotlin (host) bootstrap example:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
|
||||||
import net.sergeych.lyng.io.http.createHttpModule
|
|
||||||
import net.sergeych.lyngio.http.security.PermitAllHttpAccessPolicy
|
|
||||||
|
|
||||||
suspend fun bootstrapHttp() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope: Scope = session.getScope()
|
|
||||||
createHttpModule(PermitAllHttpAccessPolicy, scope)
|
|
||||||
session.eval("import lyng.io.http")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Using from Lyng scripts
|
|
||||||
|
|
||||||
Simple GET:
|
|
||||||
|
|
||||||
import lyng.io.http
|
|
||||||
|
|
||||||
val r = Http.get(HTTP_TEST_URL + "/hello")
|
|
||||||
[r.status, r.text()]
|
|
||||||
>>> [200,hello from test]
|
|
||||||
|
|
||||||
Headers and response header access:
|
|
||||||
|
|
||||||
import lyng.io.http
|
|
||||||
|
|
||||||
val r = Http.get(HTTP_TEST_URL + "/headers")
|
|
||||||
[r.headers["X-Reply"], r.headers.getAll("X-Reply").size, r.text()]
|
|
||||||
>>> [one,2,header demo]
|
|
||||||
|
|
||||||
Programmatic request object:
|
|
||||||
|
|
||||||
import lyng.io.http
|
|
||||||
|
|
||||||
val q = HttpRequest()
|
|
||||||
q.method = "POST"
|
|
||||||
q.url = HTTP_TEST_URL + "/echo"
|
|
||||||
q.headers = Map("Content-Type" => "text/plain")
|
|
||||||
q.bodyText = "ping"
|
|
||||||
|
|
||||||
val r = Http.request(q)
|
|
||||||
r.text()
|
|
||||||
>>> "POST:ping"
|
|
||||||
|
|
||||||
HTTPS GET:
|
|
||||||
|
|
||||||
import lyng.io.http
|
|
||||||
|
|
||||||
val r = Http.get(HTTPS_TEST_URL + "/hello")
|
|
||||||
[r.status, r.text()]
|
|
||||||
>>> [200,hello from test]
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## API reference
|
|
||||||
|
|
||||||
### `Http` (static methods)
|
|
||||||
|
|
||||||
- `isSupported(): Bool` — Whether HTTP client support is available on the current runtime.
|
|
||||||
- `request(req: HttpRequest): HttpResponse` — Execute a request described by a mutable request object.
|
|
||||||
- `get(url: String, headers...): HttpResponse` — Convenience GET request.
|
|
||||||
- `post(url: String, bodyText: String = "", contentType: String? = null, headers...): HttpResponse` — Convenience text POST request.
|
|
||||||
- `postBytes(url: String, body: Buffer, contentType: String? = null, headers...): HttpResponse` — Convenience binary POST request.
|
|
||||||
|
|
||||||
For convenience methods, `headers...` accepts:
|
|
||||||
|
|
||||||
- `MapEntry`, e.g. `"Accept" => "text/plain"`
|
|
||||||
- 2-item lists, e.g. `["Accept", "text/plain"]`
|
|
||||||
|
|
||||||
### `HttpRequest`
|
|
||||||
|
|
||||||
- `method: String`
|
|
||||||
- `url: String`
|
|
||||||
- `headers: Map<String, String>`
|
|
||||||
- `bodyText: String?`
|
|
||||||
- `bodyBytes: Buffer?`
|
|
||||||
- `timeoutMillis: Int?`
|
|
||||||
|
|
||||||
Only one of `bodyText` and `bodyBytes` should be set.
|
|
||||||
|
|
||||||
### `HttpResponse`
|
|
||||||
|
|
||||||
- `status: Int`
|
|
||||||
- `statusText: String`
|
|
||||||
- `headers: HttpHeaders`
|
|
||||||
- `text(): String`
|
|
||||||
- `bytes(): Buffer`
|
|
||||||
|
|
||||||
Response body decoding is cached inside the response object.
|
|
||||||
|
|
||||||
### `HttpHeaders`
|
|
||||||
|
|
||||||
`HttpHeaders` behaves like `Map<String, String>` for the first value of each header name and additionally exposes:
|
|
||||||
|
|
||||||
- `get(name: String): String?`
|
|
||||||
- `getAll(name: String): List<String>`
|
|
||||||
- `names(): List<String>`
|
|
||||||
|
|
||||||
Header lookup is case-insensitive.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Security policy
|
|
||||||
|
|
||||||
The module uses `HttpAccessPolicy` to authorize requests before they are sent.
|
|
||||||
|
|
||||||
- `HttpAccessPolicy` — interface for custom policies
|
|
||||||
- `PermitAllHttpAccessPolicy` — allows all requests
|
|
||||||
- `HttpAccessOp.Request(method, url)` — operation checked by the policy
|
|
||||||
|
|
||||||
Example restricted policy in Kotlin:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyngio.fs.security.AccessContext
|
|
||||||
import net.sergeych.lyngio.fs.security.AccessDecision
|
|
||||||
import net.sergeych.lyngio.fs.security.Decision
|
|
||||||
import net.sergeych.lyngio.http.security.HttpAccessOp
|
|
||||||
import net.sergeych.lyngio.http.security.HttpAccessPolicy
|
|
||||||
|
|
||||||
val allowLocalOnly = object : HttpAccessPolicy {
|
|
||||||
override suspend fun check(op: HttpAccessOp, ctx: AccessContext): AccessDecision =
|
|
||||||
when (op) {
|
|
||||||
is HttpAccessOp.Request ->
|
|
||||||
if (
|
|
||||||
op.url.startsWith("http://127.0.0.1:") ||
|
|
||||||
op.url.startsWith("https://127.0.0.1:") ||
|
|
||||||
op.url.startsWith("http://localhost:") ||
|
|
||||||
op.url.startsWith("https://localhost:")
|
|
||||||
)
|
|
||||||
AccessDecision(Decision.Allow)
|
|
||||||
else
|
|
||||||
AccessDecision(Decision.Deny, "only local HTTP/HTTPS requests are allowed")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Platform support
|
|
||||||
|
|
||||||
- **JVM:** supported
|
|
||||||
- **Android:** supported via the Ktor CIO client backend
|
|
||||||
- **JS:** supported via the Ktor JS client backend
|
|
||||||
- **Linux native:** supported via the Ktor Curl client backend
|
|
||||||
- **Windows native:** supported via the Ktor WinHttp client backend
|
|
||||||
- **Apple native:** supported via the Ktor Darwin client backend
|
|
||||||
- **Other targets:** may report unsupported; use `Http.isSupported()` before relying on it
|
|
||||||
@ -1,446 +0,0 @@
|
|||||||
# `lyng.io.http.server` - Minimal HTTP/1.1 And WebSocket Server
|
|
||||||
|
|
||||||
This module provides a small server-side HTTP API for Lyng scripts. It is implemented in `lyngio` on top of the existing TCP layer and is intended for embedded tools, local services, test fixtures, and lightweight app backends.
|
|
||||||
|
|
||||||
It supports:
|
|
||||||
- HTTP/1.1 request parsing
|
|
||||||
- keep-alive
|
|
||||||
- exact-path routing
|
|
||||||
- regex routing
|
|
||||||
- path-template routing with named parameters
|
|
||||||
- websocket upgrade and server-side sessions
|
|
||||||
|
|
||||||
It does not aim to replace a full reverse proxy. Typical deployment is behind nginx, Caddy, or another frontend that handles TLS and public-facing edge concerns.
|
|
||||||
|
|
||||||
> **Security note:** this module uses the same `NetAccessPolicy` capability model as raw TCP sockets. If scripts are allowed to listen on TCP, they can host an HTTP server.
|
|
||||||
|
|
||||||
## Install The Module Into A Lyng Session
|
|
||||||
|
|
||||||
Kotlin bootstrap example:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
|
||||||
import net.sergeych.lyng.io.http.server.createHttpServerModule
|
|
||||||
import net.sergeych.lyngio.net.security.PermitAllNetAccessPolicy
|
|
||||||
|
|
||||||
suspend fun bootstrapHttpServer() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope: Scope = session.getScope()
|
|
||||||
createHttpServerModule(PermitAllNetAccessPolicy, scope)
|
|
||||||
session.eval("import lyng.io.http.server")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## RequestContext Sugar
|
|
||||||
|
|
||||||
Route handlers use `RequestContext` as the receiver, so inside handlers you normally write direct calls such as:
|
|
||||||
|
|
||||||
- `jsonBody<T>()`
|
|
||||||
- `respondJson(...)`
|
|
||||||
- `respondHtml { ... }`
|
|
||||||
- `respondText(...)`
|
|
||||||
- `setHeader(...)`
|
|
||||||
- `request.path`
|
|
||||||
- `routeParams["id"]`
|
|
||||||
|
|
||||||
This keeps ordinary HTTP endpoints compact and avoids passing an explicit request or exchange parameter through every route lambda.
|
|
||||||
|
|
||||||
## HTML Response Sugar
|
|
||||||
|
|
||||||
Use `respondHtml { ... }` to render an HTML document with the `lyng.io.html` DSL and send it as `text/html; charset=utf-8`.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http.server
|
|
||||||
import lyng.io.html
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
|
|
||||||
server.get("/") {
|
|
||||||
respondHtml {
|
|
||||||
head {
|
|
||||||
title { +"Lyng status" }
|
|
||||||
}
|
|
||||||
body {
|
|
||||||
h3 { +"Service is running" }
|
|
||||||
p { +"Path: ${request.path}" }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
```
|
|
||||||
|
|
||||||
Pass `code:` when the route should return a non-200 status:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
server.get("/accepted") {
|
|
||||||
respondHtml(code: 202) {
|
|
||||||
body { h3 { +"Accepted" } }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## JSON API Sugar
|
|
||||||
|
|
||||||
For ordinary JSON APIs, `RequestContext` includes two primary helpers:
|
|
||||||
|
|
||||||
- `jsonBody<T>()` decodes the request body with typed `Json.decodeAs(...)`
|
|
||||||
- `respondJson(body, status = 200)` sets JSON content type and responds with plain `toJsonString()`
|
|
||||||
|
|
||||||
These helpers intentionally use ordinary JSON projection for HTTP interop, not canonical `Json.encode(...)`.
|
|
||||||
|
|
||||||
### Typed JSON POST With Route Params
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http.server
|
|
||||||
|
|
||||||
closed class CreateResultRequest(title: String, score: Int)
|
|
||||||
closed class CreateResultResponse(id: String, userId: String, title: String, score: Int)
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
|
|
||||||
server.postPath("/api/users/{userId}/results") {
|
|
||||||
val req = jsonBody<CreateResultRequest>()
|
|
||||||
|
|
||||||
if (req.title.isBlank()) {
|
|
||||||
respondJson({ error: "title must not be empty" }, 400)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
respondJson(
|
|
||||||
CreateResultResponse("r-101", routeParams["userId"], req.title, req.score),
|
|
||||||
201
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
```
|
|
||||||
|
|
||||||
### JSON Response With Route Params
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http.server
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
|
|
||||||
server.getPath("/api/users/{id}") {
|
|
||||||
respondJson({
|
|
||||||
id: routeParams["id"],
|
|
||||||
path: request.path,
|
|
||||||
ok: true
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
```
|
|
||||||
|
|
||||||
## Request And Route Data
|
|
||||||
|
|
||||||
`ServerRequest` exposes parsed HTTP request data:
|
|
||||||
|
|
||||||
- `method: String`
|
|
||||||
- `target: String`
|
|
||||||
- `path: String`
|
|
||||||
- `pathParts: List<String>`
|
|
||||||
- `queryString: String?`
|
|
||||||
- `query: Map<String, String>`
|
|
||||||
- `headers: HttpHeaders`
|
|
||||||
- `body: Buffer`
|
|
||||||
|
|
||||||
`RequestContext` exposes routing context and response controls:
|
|
||||||
|
|
||||||
- `request: ServerRequest`
|
|
||||||
- `routeMatch: RegexMatch?`
|
|
||||||
- `routeParams: Map<String, String>`
|
|
||||||
- `jsonBody<T>()`
|
|
||||||
- `respond(...)`
|
|
||||||
- `respondText(...)`
|
|
||||||
- `respondJson(body, status = 200)`
|
|
||||||
- `respondHtml(code: 200) { ... }`
|
|
||||||
- `setHeader(...)`
|
|
||||||
- `addHeader(...)`
|
|
||||||
- `acceptWebSocket(...)`
|
|
||||||
|
|
||||||
For exact routes, `routeMatch` is `null` and `routeParams` is empty.
|
|
||||||
For regex routes, `routeMatch` is set and `routeParams` is empty.
|
|
||||||
For path-template routes, both `routeMatch` and `routeParams` are set.
|
|
||||||
|
|
||||||
## Reusable Routers
|
|
||||||
|
|
||||||
`Router` collects the same route kinds as `HttpServer`, but does not listen on sockets by itself.
|
|
||||||
Mount it into `HttpServer` or another `Router`.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http.server
|
|
||||||
|
|
||||||
val api = Router()
|
|
||||||
api.get("/health") {
|
|
||||||
respondText(200, "ok")
|
|
||||||
}
|
|
||||||
|
|
||||||
val users = Router()
|
|
||||||
users.getPath("/users/{id}") {
|
|
||||||
respondJson({ id: routeParams["id"] })
|
|
||||||
}
|
|
||||||
|
|
||||||
api.mount(users)
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
server.mount(api)
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
```
|
|
||||||
|
|
||||||
Mounted routers reuse the built-in server router. They are configuration-time composition, not an extra per-request Lyng dispatch layer.
|
|
||||||
|
|
||||||
## WebSocket Routes
|
|
||||||
|
|
||||||
You can route websocket upgrades by exact path, regex, or path template.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
server.ws("/chat") { ws ->
|
|
||||||
ws.sendText("hello")
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
server.wsPath("/ws/{room}") { ws ->
|
|
||||||
ws.sendText("room=" + routeParams["room"])
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
A websocket handler runs only for requests that actually ask for websocket upgrade. Ordinary HTTP requests to the same path are not treated as websocket sessions.
|
|
||||||
|
|
||||||
### Choosing Between `ws(...)` And `acceptWebSocket(...)`
|
|
||||||
|
|
||||||
Use `server.ws(...)` or `server.wsPath(...)` when the route is always a websocket endpoint.
|
|
||||||
|
|
||||||
Use `acceptWebSocket(...)` inside a normal HTTP handler when the same route may inspect the request first and then decide whether to upgrade.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
server.get("/maybe-upgrade") {
|
|
||||||
if (!request.isWebSocketUpgrade()) {
|
|
||||||
respondText(400, "websocket upgrade required")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
acceptWebSocket { ws ->
|
|
||||||
ws.sendText("connected")
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Reading Incoming Messages
|
|
||||||
|
|
||||||
Inside a websocket handler, call `ws.receive()` to wait for the next application message.
|
|
||||||
|
|
||||||
What `receive()` returns:
|
|
||||||
- `WsMessage` for the next text or binary message.
|
|
||||||
- `null` after the client sends a close frame.
|
|
||||||
- `null` after the socket is already closed and no more frames can arrive.
|
|
||||||
|
|
||||||
What reaches Lyng code:
|
|
||||||
- Text frames become `WsMessage(isText = true, text = ...)`.
|
|
||||||
- Binary frames become `WsMessage(isText = false, data = ...)`.
|
|
||||||
- Fragmented websocket messages are reassembled before they are returned.
|
|
||||||
- Ping and pong control frames are handled internally and do not appear in Lyng.
|
|
||||||
- A client close frame is answered by the server close handshake, then `receive()` returns `null`.
|
|
||||||
|
|
||||||
Typical server receive loop:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.buffer
|
|
||||||
|
|
||||||
server.ws("/echo") { ws ->
|
|
||||||
while (true) {
|
|
||||||
val msg = ws.receive() ?: break
|
|
||||||
if (msg.isText) {
|
|
||||||
ws.sendText("echo:" + msg.text)
|
|
||||||
} else {
|
|
||||||
ws.sendBytes(msg.data as Buffer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Sending Outgoing Messages
|
|
||||||
|
|
||||||
Use:
|
|
||||||
- `ws.sendText(text)` for text messages.
|
|
||||||
- `ws.sendBytes(data)` for binary messages.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.buffer
|
|
||||||
|
|
||||||
server.ws("/push") { ws ->
|
|
||||||
ws.sendText("ready")
|
|
||||||
ws.sendBytes(Buffer(1, 2, 3))
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Send behavior:
|
|
||||||
- Each call sends one websocket message.
|
|
||||||
- The server API does not expose frame-by-frame streaming.
|
|
||||||
- Once the session is closed, send calls fail with a websocket error.
|
|
||||||
|
|
||||||
### What Happens When The Connection Closes
|
|
||||||
|
|
||||||
There are three practical cases:
|
|
||||||
|
|
||||||
1. The client closes first.
|
|
||||||
The runtime replies with a close frame, releases the socket, and `receive()` returns `null`.
|
|
||||||
|
|
||||||
2. Your handler closes first with `ws.close(...)`.
|
|
||||||
The runtime sends a close frame and releases the socket locally.
|
|
||||||
|
|
||||||
3. The transport disappears unexpectedly.
|
|
||||||
The session is released and no more messages can be received; subsequent sends fail.
|
|
||||||
|
|
||||||
What Lyng code should do:
|
|
||||||
- Treat `receive() == null` as end-of-session.
|
|
||||||
- Exit the handler or break the receive loop at that point.
|
|
||||||
- Do not keep sending after close has been observed.
|
|
||||||
|
|
||||||
The current server-side API does not expose the peer close code or close reason to Lyng.
|
|
||||||
|
|
||||||
### Closing The Connection Yourself
|
|
||||||
|
|
||||||
Call `ws.close()` when you want to terminate the websocket session.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
server.ws("/chat") { ws ->
|
|
||||||
ws.sendText("server shutting down")
|
|
||||||
ws.close(1000, "done")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Close semantics:
|
|
||||||
- `close()` sends a websocket close frame with the given code and reason.
|
|
||||||
- Defaults are `code = 1000` and `reason = ""`.
|
|
||||||
- `close()` is idempotent; calling it again after close does nothing.
|
|
||||||
- After local close, the session should be treated as unusable.
|
|
||||||
- After close, `isOpen()` becomes false and further sends fail.
|
|
||||||
|
|
||||||
### WebSocket Handler Pattern
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http.server
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
|
|
||||||
server.wsPath("/rooms/{room}") { ws ->
|
|
||||||
val room = routeParams["room"] ?: "<unknown>"
|
|
||||||
ws.sendText("joined:" + room)
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
val msg = ws.receive() ?: break
|
|
||||||
if (msg.isText) {
|
|
||||||
ws.sendText(room + ":" + msg.text)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
```
|
|
||||||
|
|
||||||
## Path-Template Routes
|
|
||||||
|
|
||||||
Path templates are sugar on top of regex routes. Template parameters are exposed as decoded `routeParams`.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
server.getPath("/users/{userId}/posts/{postId}") {
|
|
||||||
respondText(
|
|
||||||
200,
|
|
||||||
routeParams["userId"] + ":" + routeParams["postId"]
|
|
||||||
)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Template rules:
|
|
||||||
- template must start with `/`
|
|
||||||
- a segment is either literal text or `{name}`
|
|
||||||
- parameter names must be valid identifiers
|
|
||||||
- parameter values match one path segment only
|
|
||||||
- parameter values use path decoding rules:
|
|
||||||
- valid percent-encoding is decoded
|
|
||||||
- `+` stays `+`
|
|
||||||
- malformed `%` stays literal
|
|
||||||
|
|
||||||
## Regex Routes
|
|
||||||
|
|
||||||
Regex routes match the whole request path, not a substring.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
server.get("^/users/([0-9]+)/posts/([0-9]+)$".re) {
|
|
||||||
val m = routeMatch!!
|
|
||||||
respondText(200, "user=" + m[1] + ", post=" + m[2])
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Basic Exact Route
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http.server
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
server.get("/hello") {
|
|
||||||
setHeader("Content-Type", "text/plain")
|
|
||||||
respondText(200, "hello")
|
|
||||||
}
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
```
|
|
||||||
|
|
||||||
## Route Precedence
|
|
||||||
|
|
||||||
Dispatch order is:
|
|
||||||
|
|
||||||
1. exact method route
|
|
||||||
2. exact `any` route
|
|
||||||
3. regex method route, registration order
|
|
||||||
4. regex `any` route, registration order
|
|
||||||
5. fallback
|
|
||||||
|
|
||||||
This means exact routes stay fast and always win over template or regex routes for the same path.
|
|
||||||
|
|
||||||
## API Surface
|
|
||||||
|
|
||||||
### `Router` Route Registration Methods
|
|
||||||
|
|
||||||
- `get(path: String|Regex, handler)`
|
|
||||||
- `getPath(pathTemplate: String, handler)`
|
|
||||||
- `post(path: String|Regex, handler)`
|
|
||||||
- `postPath(pathTemplate: String, handler)`
|
|
||||||
- `put(path: String|Regex, handler)`
|
|
||||||
- `putPath(pathTemplate: String, handler)`
|
|
||||||
- `delete(path: String|Regex, handler)`
|
|
||||||
- `deletePath(pathTemplate: String, handler)`
|
|
||||||
- `any(path: String|Regex, handler)`
|
|
||||||
- `anyPath(pathTemplate: String, handler)`
|
|
||||||
- `ws(path: String|Regex, handler)`
|
|
||||||
- `wsPath(pathTemplate: String, handler)`
|
|
||||||
- `fallback(handler)`
|
|
||||||
- `mount(router)`
|
|
||||||
|
|
||||||
### `HttpServer` Route Registration Methods
|
|
||||||
|
|
||||||
- `get(path: String|Regex, handler)`
|
|
||||||
- `getPath(pathTemplate: String, handler)`
|
|
||||||
- `post(path: String|Regex, handler)`
|
|
||||||
- `postPath(pathTemplate: String, handler)`
|
|
||||||
- `put(path: String|Regex, handler)`
|
|
||||||
- `putPath(pathTemplate: String, handler)`
|
|
||||||
- `delete(path: String|Regex, handler)`
|
|
||||||
- `deletePath(pathTemplate: String, handler)`
|
|
||||||
- `any(path: String|Regex, handler)`
|
|
||||||
- `anyPath(pathTemplate: String, handler)`
|
|
||||||
- `ws(path: String|Regex, handler)`
|
|
||||||
- `wsPath(pathTemplate: String, handler)`
|
|
||||||
- `fallback(handler)`
|
|
||||||
- `mount(router)`
|
|
||||||
- `listen(port, host = null, backlog = 128)`
|
|
||||||
@ -1,177 +0,0 @@
|
|||||||
### lyng.io.net — TCP and UDP sockets for Lyng scripts
|
|
||||||
|
|
||||||
This module provides minimal raw transport networking for Lyng scripts. It is implemented in `lyngio` and backed by Ktor sockets on the JVM and Linux Native, and by Node networking APIs on JS/Node runtimes.
|
|
||||||
|
|
||||||
> **Note:** `lyngio` is a separate library module. It must be explicitly added as a dependency to your host application and initialized in your Lyng scopes.
|
|
||||||
>
|
|
||||||
> **Shared type note:** `IpVersion`, `SocketAddress`, and `Datagram` are also available from `lyng.io.net.types` when host code wants reusable transport value types without depending on the `Net` capability object itself.
|
|
||||||
>
|
|
||||||
> **Important native platform limit:** current native TCP/UDP support is backed by a selector with a per-process file descriptor ceiling. On Linux/macOS native targets this makes high-connection-count servers and same-process load tests unsuitable once the process approaches that limit.
|
|
||||||
>
|
|
||||||
> **Recommendation:** for serious HTTP/TCP servers, prefer the JVM target today. On native targets, keep concurrency bounded, batch local load tests in waves, and use multiple worker processes behind a reverse proxy if you need more throughput before the backend is reworked.
|
|
||||||
>
|
|
||||||
> **Need this fixed?** Please open or upvote an issue at <https://github.com/sergeych/lyng/issues> so native high-concurrency networking can be prioritized.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
#### Install the module into a Lyng session
|
|
||||||
|
|
||||||
Kotlin (host) bootstrap example:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
|
||||||
import net.sergeych.lyng.io.net.createNetModule
|
|
||||||
import net.sergeych.lyngio.net.security.PermitAllNetAccessPolicy
|
|
||||||
|
|
||||||
suspend fun bootstrapNet() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope: Scope = session.getScope()
|
|
||||||
createNetModule(PermitAllNetAccessPolicy, scope)
|
|
||||||
session.eval("import lyng.io.net")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
#### Using from Lyng scripts
|
|
||||||
|
|
||||||
Capability checks and address resolution:
|
|
||||||
|
|
||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val a: SocketAddress = Net.resolve("127.0.0.1", 4040)[0]
|
|
||||||
[Net.isSupported(), a.toString(), a.resolved, a.ipVersion == IpVersion.IPV4]
|
|
||||||
>>> [true,127.0.0.1:4040,true,true]
|
|
||||||
|
|
||||||
TCP client connect, write, read, and close:
|
|
||||||
|
|
||||||
import lyng.buffer
|
|
||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val socket = Net.tcpConnect("127.0.0.1", NET_TEST_TCP_PORT)
|
|
||||||
socket.writeUtf8("ping")
|
|
||||||
socket.flush()
|
|
||||||
val reply = (socket.read(16) as Buffer).decodeUtf8()
|
|
||||||
socket.close()
|
|
||||||
reply
|
|
||||||
>>> "reply:ping"
|
|
||||||
|
|
||||||
Lyng TCP server socket operations with `tcpListen()` and `accept()`:
|
|
||||||
|
|
||||||
import lyng.buffer
|
|
||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val server = Net.tcpListen(0, "127.0.0.1")
|
|
||||||
val port = server.localAddress().port
|
|
||||||
val accepted = launch {
|
|
||||||
val client = server.accept()
|
|
||||||
val line = (client.read(4) as Buffer).decodeUtf8()
|
|
||||||
client.writeUtf8("echo:" + line)
|
|
||||||
client.flush()
|
|
||||||
client.close()
|
|
||||||
server.close()
|
|
||||||
line
|
|
||||||
}
|
|
||||||
|
|
||||||
val socket = Net.tcpConnect("127.0.0.1", port)
|
|
||||||
socket.writeUtf8("ping")
|
|
||||||
socket.flush()
|
|
||||||
val reply = (socket.read(16) as Buffer).decodeUtf8()
|
|
||||||
socket.close()
|
|
||||||
[accepted.await(), reply]
|
|
||||||
>>> [ping,echo:ping]
|
|
||||||
|
|
||||||
UDP bind, send, receive, and inspect sender address:
|
|
||||||
|
|
||||||
import lyng.buffer
|
|
||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val server = Net.udpBind(0, "127.0.0.1")
|
|
||||||
val client = Net.udpBind(0, "127.0.0.1")
|
|
||||||
client.send(Buffer("ping"), "127.0.0.1", server.localAddress().port)
|
|
||||||
val d = server.receive()
|
|
||||||
client.close()
|
|
||||||
server.close()
|
|
||||||
[d.data.decodeUtf8(), d.address.port > 0]
|
|
||||||
>>> [ping,true]
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
#### API reference
|
|
||||||
|
|
||||||
##### `Net` (static methods)
|
|
||||||
|
|
||||||
- `isSupported(): Bool` — Whether any raw networking support is available.
|
|
||||||
- `isTcpAvailable(): Bool` — Whether outbound TCP sockets are available.
|
|
||||||
- `isTcpServerAvailable(): Bool` — Whether listening TCP server sockets are available.
|
|
||||||
- `isUdpAvailable(): Bool` — Whether UDP datagram sockets are available.
|
|
||||||
- `resolve(host: String, port: Int): List<SocketAddress>` — Resolve a host and port into concrete addresses.
|
|
||||||
- `tcpConnect(host: String, port: Int, timeoutMillis: Int? = null, noDelay: Bool = true): TcpSocket` — Open an outbound TCP socket.
|
|
||||||
- `tcpListen(port: Int, host: String? = null, backlog: Int = 128, reuseAddress: Bool = true): TcpServer` — Start a listening TCP server socket.
|
|
||||||
- `udpBind(port: Int = 0, host: String? = null, reuseAddress: Bool = true): UdpSocket` — Bind a UDP socket.
|
|
||||||
|
|
||||||
##### `SocketAddress`
|
|
||||||
|
|
||||||
- `host: String`
|
|
||||||
- `port: Int`
|
|
||||||
- `ipVersion: IpVersion`
|
|
||||||
- `resolved: Bool`
|
|
||||||
- `toString(): String`
|
|
||||||
|
|
||||||
##### `TcpSocket`
|
|
||||||
|
|
||||||
- `isOpen(): Bool`
|
|
||||||
- `localAddress(): SocketAddress`
|
|
||||||
- `remoteAddress(): SocketAddress`
|
|
||||||
- `read(maxBytes: Int = 65536): Buffer?`
|
|
||||||
- `readLine(): String?`
|
|
||||||
- `write(data: Buffer): void`
|
|
||||||
- `writeUtf8(text: String): void`
|
|
||||||
- `flush(): void`
|
|
||||||
- `close(): void`
|
|
||||||
|
|
||||||
##### `TcpServer`
|
|
||||||
|
|
||||||
- `isOpen(): Bool`
|
|
||||||
- `localAddress(): SocketAddress`
|
|
||||||
- `accept(): TcpSocket`
|
|
||||||
- `close(): void`
|
|
||||||
|
|
||||||
##### `UdpSocket`
|
|
||||||
|
|
||||||
- `isOpen(): Bool`
|
|
||||||
- `localAddress(): SocketAddress`
|
|
||||||
- `receive(maxBytes: Int = 65536): Datagram?`
|
|
||||||
- `send(data: Buffer, host: String, port: Int): void`
|
|
||||||
- `close(): void`
|
|
||||||
|
|
||||||
##### `Datagram`
|
|
||||||
|
|
||||||
- `data: Buffer`
|
|
||||||
- `address: SocketAddress`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
#### Security policy
|
|
||||||
|
|
||||||
The module uses `NetAccessPolicy` to authorize network operations before they are executed.
|
|
||||||
|
|
||||||
- `NetAccessPolicy` — interface for custom policies
|
|
||||||
- `PermitAllNetAccessPolicy` — allows all network operations
|
|
||||||
- `NetAccessOp.Resolve(host, port)`
|
|
||||||
- `NetAccessOp.TcpConnect(host, port)`
|
|
||||||
- `NetAccessOp.TcpListen(host, port, backlog)`
|
|
||||||
- `NetAccessOp.UdpBind(host, port)`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
#### Platform support
|
|
||||||
|
|
||||||
- **JVM:** supported
|
|
||||||
- **Android:** supported via the Ktor CIO and Ktor sockets backends
|
|
||||||
- **JS/Node:** supported for `resolve`, TCP client/server, and UDP
|
|
||||||
- **JS/browser:** unsupported; capability checks report unavailable
|
|
||||||
- **Linux Native:** supported via Ktor sockets
|
|
||||||
- **Apple Native:** enabled via the shared native Ktor sockets backend; compile-verified, runtime not yet host-verified
|
|
||||||
- **Other native targets:** currently report unsupported; use capability checks before relying on raw sockets
|
|
||||||
@ -20,26 +20,24 @@ For external projects, ensure you have the appropriate Maven repository configur
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
#### Install the module into a Lyng session
|
#### Install the module into a Lyng Scope
|
||||||
|
|
||||||
The process module is not installed automatically. The preferred host runtime is `EvalSession`: create the session, get its underlying scope, install the module there, and execute scripts through the session. You can customize access control via `ProcessAccessPolicy`.
|
The process module is not installed automatically. You must explicitly register it in the scope’s `ImportManager` using `createProcessModule`. You can customize access control via `ProcessAccessPolicy`.
|
||||||
|
|
||||||
Kotlin (host) bootstrap example:
|
Kotlin (host) bootstrap example:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
import net.sergeych.lyng.Scope
|
import net.sergeych.lyng.Scope
|
||||||
import net.sergeych.lyng.EvalSession
|
import net.sergeych.lyng.Script
|
||||||
import net.sergeych.lyng.io.process.createProcessModule
|
import net.sergeych.lyng.io.process.createProcessModule
|
||||||
import net.sergeych.lyngio.process.security.PermitAllProcessAccessPolicy
|
import net.sergeych.lyngio.process.security.PermitAllProcessAccessPolicy
|
||||||
|
|
||||||
suspend fun bootstrapProcess() {
|
// ... inside a suspend function or runBlocking
|
||||||
val session = EvalSession()
|
val scope: Scope = Script.newScope()
|
||||||
val scope: Scope = session.getScope()
|
createProcessModule(PermitAllProcessAccessPolicy, scope)
|
||||||
createProcessModule(PermitAllProcessAccessPolicy, scope)
|
|
||||||
|
|
||||||
// In scripts (or via session.eval), import the module:
|
// In scripts (or via scope.eval), import the module:
|
||||||
session.eval("import lyng.io.process")
|
scope.eval("import lyng.io.process")
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|||||||
@ -1,279 +0,0 @@
|
|||||||
# `lyng.io.ws` - WebSocket client for Lyng scripts
|
|
||||||
|
|
||||||
This module provides a compact WebSocket client API for Lyng scripts. It is implemented in `lyngio` and currently backed by Ktor WebSockets on the JVM.
|
|
||||||
|
|
||||||
> **Note:** `lyngio` is a separate library module. It must be explicitly added as a dependency to your host application and initialized in your Lyng scopes.
|
|
||||||
>
|
|
||||||
> **Shared type note:** `WsMessage` is also available from `lyng.io.ws.types` when host code wants the reusable message type without depending on the WebSocket client module itself.
|
|
||||||
|
|
||||||
## Install The Module Into A Lyng Session
|
|
||||||
|
|
||||||
Kotlin host bootstrap example:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyng.EvalSession
|
|
||||||
import net.sergeych.lyng.Scope
|
|
||||||
import net.sergeych.lyng.io.ws.createWsModule
|
|
||||||
import net.sergeych.lyngio.ws.security.PermitAllWsAccessPolicy
|
|
||||||
|
|
||||||
suspend fun bootstrapWs() {
|
|
||||||
val session = EvalSession()
|
|
||||||
val scope: Scope = session.getScope()
|
|
||||||
createWsModule(PermitAllWsAccessPolicy, scope)
|
|
||||||
session.eval("import lyng.io.ws")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Using From Lyng Scripts
|
|
||||||
|
|
||||||
### Text Exchange
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_TEST_URL)
|
|
||||||
ws.sendText("ping")
|
|
||||||
val m: WsMessage = ws.receive()
|
|
||||||
ws.close()
|
|
||||||
[ws.url() == WS_TEST_URL, m.isText, m.text]
|
|
||||||
>>> [true,true,echo:ping]
|
|
||||||
```
|
|
||||||
|
|
||||||
### Binary Exchange
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.buffer
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_TEST_BINARY_URL)
|
|
||||||
ws.sendBytes(Buffer(9, 8, 7))
|
|
||||||
val m: WsMessage = ws.receive()
|
|
||||||
ws.close()
|
|
||||||
[m.isText, (m.data as Buffer).hex]
|
|
||||||
>>> [false,010203090807]
|
|
||||||
```
|
|
||||||
|
|
||||||
### Secure `wss` Exchange
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WSS_TEST_URL)
|
|
||||||
ws.sendText("ping")
|
|
||||||
val m: WsMessage = ws.receive()
|
|
||||||
ws.close()
|
|
||||||
[ws.url() == WSS_TEST_URL, m.text]
|
|
||||||
>>> [true,secure:ping]
|
|
||||||
```
|
|
||||||
|
|
||||||
## Message Flow And Session Lifecycle
|
|
||||||
|
|
||||||
### Reading Incoming Messages
|
|
||||||
|
|
||||||
Call `ws.receive()` to wait for the next application message.
|
|
||||||
|
|
||||||
What `receive()` returns:
|
|
||||||
- `WsMessage` for the next text or binary message.
|
|
||||||
- `null` after the peer closes the connection cleanly.
|
|
||||||
- `null` after the transport has already been closed and no more messages can arrive.
|
|
||||||
|
|
||||||
What reaches Lyng code:
|
|
||||||
- Text frames are exposed as `WsMessage(isText = true, text = ...)`.
|
|
||||||
- Binary frames are exposed as `WsMessage(isText = false, data = ...)`.
|
|
||||||
- Fragmented websocket messages are reassembled before they are returned.
|
|
||||||
- Ping and pong control frames are handled internally and are not returned by `receive()`.
|
|
||||||
- Incoming close frames are handled internally; after that `receive()` returns `null`.
|
|
||||||
|
|
||||||
Typical receive loop:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.buffer
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_URL)
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
val msg = ws.receive() ?: break
|
|
||||||
|
|
||||||
if (msg.isText) {
|
|
||||||
println("text=" + msg.text)
|
|
||||||
} else {
|
|
||||||
println("bytes=" + ((msg.data as Buffer).size))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
println("peer closed the websocket")
|
|
||||||
```
|
|
||||||
|
|
||||||
### Sending Outgoing Messages
|
|
||||||
|
|
||||||
Use:
|
|
||||||
- `ws.sendText(text)` for UTF-8 text messages.
|
|
||||||
- `ws.sendBytes(data)` for binary messages.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.buffer
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_URL)
|
|
||||||
ws.sendText("hello")
|
|
||||||
ws.sendBytes(Buffer(1, 2, 3, 4))
|
|
||||||
```
|
|
||||||
|
|
||||||
Send behavior:
|
|
||||||
- Each call sends one websocket message.
|
|
||||||
- The API does not expose partial-frame streaming; send the whole message in one call.
|
|
||||||
- If the session is already closed, `sendText(...)` and `sendBytes(...)` fail with a websocket error.
|
|
||||||
- If the transport breaks during send, the session is released and the send call fails.
|
|
||||||
|
|
||||||
### Detecting Closed Connections
|
|
||||||
|
|
||||||
Use both signals together:
|
|
||||||
- `ws.isOpen()` tells you whether the session is still considered open right now.
|
|
||||||
- `ws.receive() == null` tells you the receive side has reached the end of the websocket session.
|
|
||||||
|
|
||||||
Practical rule:
|
|
||||||
- If `receive()` returns `null`, stop reading and treat the session as closed.
|
|
||||||
- After close has been observed, do not attempt further sends.
|
|
||||||
|
|
||||||
The API does not currently expose the peer close code or close reason to Lyng code.
|
|
||||||
|
|
||||||
### Closing The Connection Yourself
|
|
||||||
|
|
||||||
Call `ws.close()` when you are done.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_URL)
|
|
||||||
ws.sendText("bye")
|
|
||||||
ws.close(1000, "done")
|
|
||||||
```
|
|
||||||
|
|
||||||
Close semantics:
|
|
||||||
- `close()` sends a websocket close frame with the given code and reason.
|
|
||||||
- Defaults are `code = 1000` and `reason = ""`.
|
|
||||||
- After `close()`, the session is released locally and should be treated as closed immediately.
|
|
||||||
- Calling `close()` on an already closed session is a no-op.
|
|
||||||
- After local close, `receive()` returns `null` and further sends fail.
|
|
||||||
|
|
||||||
### Recommended Usage Pattern
|
|
||||||
|
|
||||||
For request-response style exchanges:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_URL)
|
|
||||||
try {
|
|
||||||
ws.sendText("ping")
|
|
||||||
val reply = ws.receive() ?: error("socket closed before reply")
|
|
||||||
println(reply.text)
|
|
||||||
} finally {
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
For long-lived consumers:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect(WS_URL)
|
|
||||||
|
|
||||||
try {
|
|
||||||
while (true) {
|
|
||||||
val msg = ws.receive() ?: break
|
|
||||||
if (msg.isText) {
|
|
||||||
println(msg.text)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
ws.close()
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## API Reference
|
|
||||||
|
|
||||||
### `Ws`
|
|
||||||
|
|
||||||
- `isSupported(): Bool` - whether WebSocket client support is available on the current runtime.
|
|
||||||
- `connect(url: String, headers...): WsSession` - open a client websocket session.
|
|
||||||
|
|
||||||
`headers...` accepts:
|
|
||||||
- `MapEntry`, for example `"Authorization" => "Bearer x"`
|
|
||||||
- 2-item lists, for example `["Authorization", "Bearer x"]`
|
|
||||||
|
|
||||||
### `WsSession`
|
|
||||||
|
|
||||||
- `isOpen(): Bool`
|
|
||||||
- `url(): String`
|
|
||||||
- `sendText(text: String): void`
|
|
||||||
- `sendBytes(data: Buffer): void`
|
|
||||||
- `receive(): WsMessage?`
|
|
||||||
- `close(code: Int = 1000, reason: String = ""): void`
|
|
||||||
|
|
||||||
Behavior summary:
|
|
||||||
- `receive()` returns `null` after close.
|
|
||||||
- `close()` is safe to call more than once.
|
|
||||||
- send operations require an open session.
|
|
||||||
|
|
||||||
### `WsMessage`
|
|
||||||
|
|
||||||
- `isText: Bool`
|
|
||||||
- `text: String?`
|
|
||||||
- `data: Buffer?`
|
|
||||||
|
|
||||||
Payload rules:
|
|
||||||
- Text messages populate `text` and leave `data == null`.
|
|
||||||
- Binary messages populate `data` and leave `text == null`.
|
|
||||||
|
|
||||||
## Security Policy
|
|
||||||
|
|
||||||
The module uses `WsAccessPolicy` to authorize websocket operations.
|
|
||||||
|
|
||||||
- `WsAccessPolicy` - interface for custom policies.
|
|
||||||
- `PermitAllWsAccessPolicy` - allows all websocket operations.
|
|
||||||
- `WsAccessOp.Connect(url)`
|
|
||||||
- `WsAccessOp.Send(url, bytes, isText)`
|
|
||||||
- `WsAccessOp.Receive(url)`
|
|
||||||
|
|
||||||
Example restricted policy in Kotlin:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
import net.sergeych.lyngio.fs.security.AccessContext
|
|
||||||
import net.sergeych.lyngio.fs.security.AccessDecision
|
|
||||||
import net.sergeych.lyngio.fs.security.Decision
|
|
||||||
import net.sergeych.lyngio.ws.security.WsAccessOp
|
|
||||||
import net.sergeych.lyngio.ws.security.WsAccessPolicy
|
|
||||||
|
|
||||||
val allowLocalOnly = object : WsAccessPolicy {
|
|
||||||
override suspend fun check(op: WsAccessOp, ctx: AccessContext): AccessDecision =
|
|
||||||
when (op) {
|
|
||||||
is WsAccessOp.Connect ->
|
|
||||||
if (
|
|
||||||
op.url.startsWith("ws://127.0.0.1:") ||
|
|
||||||
op.url.startsWith("wss://127.0.0.1:") ||
|
|
||||||
op.url.startsWith("ws://localhost:") ||
|
|
||||||
op.url.startsWith("wss://localhost:")
|
|
||||||
)
|
|
||||||
AccessDecision(Decision.Allow)
|
|
||||||
else
|
|
||||||
AccessDecision(Decision.Deny, "only local ws/wss connections are allowed")
|
|
||||||
|
|
||||||
else -> AccessDecision(Decision.Allow)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Platform Support
|
|
||||||
|
|
||||||
- **JVM:** supported.
|
|
||||||
- **Android:** supported via the Ktor CIO websocket client backend.
|
|
||||||
- **JS:** supported via the Ktor JS websocket client backend.
|
|
||||||
- **Linux native:** supported via the Ktor Curl websocket client backend.
|
|
||||||
- **Windows native:** supported via the Ktor WinHttp websocket client backend.
|
|
||||||
- **Apple native:** supported via the Ktor Darwin websocket client backend.
|
|
||||||
- **Other targets:** may report unsupported; use `Ws.isSupported()` before relying on websocket client access.
|
|
||||||
122
docs/lyng_cli.md
122
docs/lyng_cli.md
@ -1,15 +1,13 @@
|
|||||||
# Lyng CLI (`lyng`)
|
### Lyng CLI (`lyng`)
|
||||||
|
|
||||||
The Lyng CLI is the reference command-line tool for the Lyng language. It lets you:
|
The Lyng CLI is the reference command-line tool for the Lyng language. It lets you:
|
||||||
|
|
||||||
- Run Lyng scripts from files or inline strings (shebangs accepted)
|
- Run Lyng scripts from files or inline strings (shebangs accepted)
|
||||||
- Use standard argument passing (`ARGV`) to your scripts.
|
- Use standard argument passing (`ARGV`) to your scripts.
|
||||||
- Resolve local file imports from the executed script's directory tree.
|
|
||||||
- Format Lyng source files via the built-in `fmt` subcommand.
|
- Format Lyng source files via the built-in `fmt` subcommand.
|
||||||
- Register synchronous process-exit handlers with `atExit(...)`.
|
|
||||||
|
|
||||||
|
|
||||||
## Building on Linux
|
#### Building on Linux
|
||||||
|
|
||||||
Requirements:
|
Requirements:
|
||||||
- JDK 17+ (for Gradle and the JVM distribution)
|
- JDK 17+ (for Gradle and the JVM distribution)
|
||||||
@ -21,7 +19,7 @@ The repository provides convenience scripts in `bin/` for local builds and insta
|
|||||||
Note: In this repository the scripts are named `bin/local_release` and `bin/local_jrelease`. In some environments these may be aliased as `bin/release` and `bin/jrelease`. The steps below use the actual file names present here.
|
Note: In this repository the scripts are named `bin/local_release` and `bin/local_jrelease`. In some environments these may be aliased as `bin/release` and `bin/jrelease`. The steps below use the actual file names present here.
|
||||||
|
|
||||||
|
|
||||||
### Option A: Native linuxX64 executable (`lyng`)
|
##### Option A: Native linuxX64 executable (`lyng`)
|
||||||
|
|
||||||
1) Build the native binary:
|
1) Build the native binary:
|
||||||
|
|
||||||
@ -40,27 +38,26 @@ What this does:
|
|||||||
- Produces `distributables/lyng-linuxX64.zip` containing the `lyng` executable.
|
- Produces `distributables/lyng-linuxX64.zip` containing the `lyng` executable.
|
||||||
|
|
||||||
|
|
||||||
### Option B: JVM distribution (`jlyng` launcher)
|
##### Option B: JVM distribution (`jlyng` launcher)
|
||||||
|
|
||||||
This creates a JVM distribution with a launcher script, packages it as a downloadable zip, and links it to `~/bin/jlyng`.
|
This creates a JVM distribution with a launcher script and links it to `~/bin/jlyng`.
|
||||||
|
|
||||||
```
|
```
|
||||||
bin/local_jrelease
|
bin/local_jrelease
|
||||||
```
|
```
|
||||||
|
|
||||||
What this does:
|
What this does:
|
||||||
- Runs `./gradlew :lyng:jvmDistZip` to build the JVM app distribution archive at `lyng/build/distributions/lyng-jvm.zip`.
|
- Runs `./gradlew :lyng:installJvmDist` to build the JVM app distribution to `lyng/build/install/lyng-jvm`.
|
||||||
- Copies the archive to `distributables/lyng-jvm.zip`.
|
- Copies the distribution under `~/bin/jlyng-jvm`.
|
||||||
- Unpacks that distribution under `~/bin/jlyng-jvm`.
|
|
||||||
- Creates a symlink `~/bin/jlyng` pointing to the launcher script.
|
- Creates a symlink `~/bin/jlyng` pointing to the launcher script.
|
||||||
|
|
||||||
|
|
||||||
## Usage
|
#### Usage
|
||||||
|
|
||||||
Once installed, ensure `~/bin` is on your `PATH`. You can then use either the native `lyng` or the JVM `jlyng` launcher (both have the same CLI surface).
|
Once installed, ensure `~/bin` is on your `PATH`. You can then use either the native `lyng` or the JVM `jlyng` launcher (both have the same CLI surface).
|
||||||
|
|
||||||
|
|
||||||
### Running scripts
|
##### Running scripts
|
||||||
|
|
||||||
- Run a script by file name and pass arguments to `ARGV`:
|
- Run a script by file name and pass arguments to `ARGV`:
|
||||||
|
|
||||||
@ -75,7 +72,6 @@ lyng -- -my-script.lyng arg1 arg2
|
|||||||
```
|
```
|
||||||
|
|
||||||
- Execute inline code with `-x/--execute` and pass positional args to `ARGV`:
|
- Execute inline code with `-x/--execute` and pass positional args to `ARGV`:
|
||||||
- Inline execution does not scan the filesystem for local modules; only file-based execution does.
|
|
||||||
|
|
||||||
```
|
```
|
||||||
lyng -x "println(\"Hello\")" more args
|
lyng -x "println(\"Hello\")" more args
|
||||||
@ -88,101 +84,7 @@ lyng --version
|
|||||||
lyng --help
|
lyng --help
|
||||||
```
|
```
|
||||||
|
|
||||||
### Exit handlers: `atExit(...)`
|
### Use in shell scripts
|
||||||
|
|
||||||
The CLI exposes a CLI-only builtin:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
extern fun atExit(append: Bool=true, handler: ()->Void)
|
|
||||||
```
|
|
||||||
|
|
||||||
Use it to register synchronous cleanup handlers that should run when the CLI process is leaving.
|
|
||||||
|
|
||||||
Semantics:
|
|
||||||
- `append=true` appends the handler to the end of the queue.
|
|
||||||
- `append=false` inserts the handler at the front of the queue.
|
|
||||||
- Handlers run one by one.
|
|
||||||
- Exceptions thrown by a handler are ignored, and the next handler still runs.
|
|
||||||
- Handlers are best-effort and run on:
|
|
||||||
- normal script completion
|
|
||||||
- script failure
|
|
||||||
- script `exit(code)`
|
|
||||||
- process shutdown such as `SIGTERM`
|
|
||||||
|
|
||||||
Non-goals:
|
|
||||||
- `SIGKILL`, hard crashes, and power loss cannot be intercepted.
|
|
||||||
- `atExit` is currently a CLI feature only; it is not part of the general embedding/runtime surface.
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
atExit {
|
|
||||||
println("closing resources")
|
|
||||||
}
|
|
||||||
|
|
||||||
atExit(false) {
|
|
||||||
println("runs first")
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Local imports for file execution
|
|
||||||
|
|
||||||
When you execute a script file, the CLI builds a temporary local import manager rooted at the directory that contains the entry script.
|
|
||||||
|
|
||||||
Formal structure:
|
|
||||||
|
|
||||||
- Root directory: the parent directory of the script passed to `lyng`.
|
|
||||||
- Scan scope: every `.lyng` file under that root directory, recursively.
|
|
||||||
- Entry script: the executed file itself is not registered as an importable module.
|
|
||||||
- Module name mapping: `relative/path/to/file.lyng` maps to import name `relative.path.to.file`.
|
|
||||||
- Package declaration: if a scanned file starts with `package ...` as its first non-blank line, that package name must exactly match the relative path mapping.
|
|
||||||
- Package omission: if there is no leading `package` declaration, the CLI uses the relative path mapping as the module name.
|
|
||||||
- Duplicates: if two files resolve to the same module name, CLI execution fails before script execution starts.
|
|
||||||
- Import visibility: only files inside the entry root subtree are considered. Parent directories and sibling projects are not searched.
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
|
|
||||||
```
|
|
||||||
project/
|
|
||||||
main.lyng
|
|
||||||
util/answer.lyng
|
|
||||||
math/add.lyng
|
|
||||||
```
|
|
||||||
|
|
||||||
`util/answer.lyng` is imported as `import util.answer`.
|
|
||||||
|
|
||||||
`math/add.lyng` is imported as `import math.add`.
|
|
||||||
|
|
||||||
Example contents:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
// util/answer.lyng
|
|
||||||
package util.answer
|
|
||||||
|
|
||||||
import math.add
|
|
||||||
|
|
||||||
fun answer() = plus(40, 2)
|
|
||||||
```
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
// math/add.lyng
|
|
||||||
fun plus(a, b) = a + b
|
|
||||||
```
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
// main.lyng
|
|
||||||
import util.answer
|
|
||||||
|
|
||||||
println(answer())
|
|
||||||
```
|
|
||||||
|
|
||||||
Rationale:
|
|
||||||
|
|
||||||
- The module name is deterministic from the filesystem layout.
|
|
||||||
- Explicit `package` remains available as a consistency check instead of a second, conflicting naming system.
|
|
||||||
- The import search space stays local to the executed script, which avoids accidental cross-project resolution.
|
|
||||||
|
|
||||||
## Use in shell scripts
|
|
||||||
|
|
||||||
Standard unix shebangs (`#!`) are supported, so you can make Lyng scripts directly executable on Unix-like systems. For example:
|
Standard unix shebangs (`#!`) are supported, so you can make Lyng scripts directly executable on Unix-like systems. For example:
|
||||||
|
|
||||||
@ -190,7 +92,7 @@ Standard unix shebangs (`#!`) are supported, so you can make Lyng scripts direct
|
|||||||
println("Hello, world!")
|
println("Hello, world!")
|
||||||
|
|
||||||
|
|
||||||
### Formatting source: `fmt` subcommand
|
##### Formatting source: `fmt` subcommand
|
||||||
|
|
||||||
Format Lyng files with the built-in formatter.
|
Format Lyng files with the built-in formatter.
|
||||||
|
|
||||||
@ -232,7 +134,7 @@ lyng fmt --spacing --wrap src/file.lyng
|
|||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
## Notes
|
#### Notes
|
||||||
|
|
||||||
- Both native and JVM distributions expose the same CLI interface. Use whichever best fits your environment.
|
- Both native and JVM distributions expose the same CLI interface. Use whichever best fits your environment.
|
||||||
- When executing scripts, all positional arguments after the script name are available in Lyng as `ARGV`.
|
- When executing scripts, all positional arguments after the script name are available in Lyng as `ARGV`.
|
||||||
|
|||||||
@ -1,116 +0,0 @@
|
|||||||
# `.lyng.d` Definition Files
|
|
||||||
|
|
||||||
`.lyng.d` files declare Lyng symbols for tooling without shipping runtime implementations. The IntelliJ IDEA plugin merges
|
|
||||||
all `*.lyng.d` files from the current directory and its parent directories into the active file’s analysis, enabling:
|
|
||||||
|
|
||||||
- completion
|
|
||||||
- navigation
|
|
||||||
- error checking for declared symbols
|
|
||||||
- Quick Docs for declarations defined in `.lyng.d`
|
|
||||||
|
|
||||||
Place `*.lyng.d` files next to the code they describe (or in a parent folder). The plugin will pick them up automatically.
|
|
||||||
|
|
||||||
## Writing `.lyng.d` Files
|
|
||||||
|
|
||||||
You can declare any language-level symbol in a `.lyng.d` file. Use doc comments before declarations to make Quick Docs
|
|
||||||
work in the IDE. The doc parser accepts standard comments (`/** ... */` or `// ...`) and supports tags like `@param`.
|
|
||||||
|
|
||||||
### Full Example
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
/** Library entry point */
|
|
||||||
extern fun connect(url: String, timeoutMs: Int = 5000): Client
|
|
||||||
|
|
||||||
/** Type alias with generics */
|
|
||||||
type NameMap = Map<String, String>
|
|
||||||
|
|
||||||
/** Multiple inheritance via interfaces */
|
|
||||||
interface A { abstract fun a(): Int }
|
|
||||||
interface B { abstract fun b(): Int }
|
|
||||||
|
|
||||||
/** A concrete class implementing both */
|
|
||||||
class Multi(name: String) : A, B {
|
|
||||||
/** Public field */
|
|
||||||
val id: Int = 0
|
|
||||||
|
|
||||||
/** Mutable property with accessors */
|
|
||||||
var size: Int
|
|
||||||
get() = 0
|
|
||||||
set(v) { }
|
|
||||||
|
|
||||||
/** Instance method */
|
|
||||||
fun a(): Int = 1
|
|
||||||
fun b(): Int = 2
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Nullable and dynamic types */
|
|
||||||
extern val dynValue: dynamic
|
|
||||||
extern var dynVar: dynamic?
|
|
||||||
|
|
||||||
/** Delegated property */
|
|
||||||
class LazyBox(val create) {
|
|
||||||
fun getValue(thisRef, name) = create()
|
|
||||||
}
|
|
||||||
|
|
||||||
val cached by LazyBox { 42 }
|
|
||||||
|
|
||||||
/** Delegated function */
|
|
||||||
object RpcDelegate {
|
|
||||||
fun invoke(thisRef, name, args...) = Unset
|
|
||||||
}
|
|
||||||
|
|
||||||
fun remoteCall by RpcDelegate
|
|
||||||
|
|
||||||
/** Singleton object */
|
|
||||||
object Settings {
|
|
||||||
val version: String = "1.0"
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Class with documented members */
|
|
||||||
class Client {
|
|
||||||
/** Returns a greeting. */
|
|
||||||
fun greet(name: String): String = "hi " + name
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
See a runnable sample file in `docs/samples/definitions.lyng.d`.
|
|
||||||
|
|
||||||
Notes:
|
|
||||||
- Use real bodies if the declaration is not `extern` or `abstract`.
|
|
||||||
- If you need purely declarative stubs, prefer `extern` members (see `embedding.md`).
|
|
||||||
|
|
||||||
## Doc Comment Format
|
|
||||||
|
|
||||||
Doc comments are picked up when they immediately precede a declaration.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
/**
|
|
||||||
* A sample function.
|
|
||||||
* @param name user name
|
|
||||||
* @return greeting string
|
|
||||||
*/
|
|
||||||
fun greet(name: String): String = "hi " + name
|
|
||||||
```
|
|
||||||
|
|
||||||
## Generating `.lyng.d` Files
|
|
||||||
|
|
||||||
You can generate `.lyng.d` as part of your build. A common approach is to write a Gradle task that emits a file from a
|
|
||||||
template or a Kotlin data model.
|
|
||||||
|
|
||||||
Example (pseudo-code):
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
tasks.register("generateLyngDefs") {
|
|
||||||
doLast {
|
|
||||||
val out = file("src/main/lyng/api.lyng.d")
|
|
||||||
out.writeText(
|
|
||||||
"""
|
|
||||||
/** Generated API */
|
|
||||||
fun ping(): Int
|
|
||||||
""".trimIndent()
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Place the generated file in your source tree, and the IDE will load it automatically.
|
|
||||||
@ -2,25 +2,16 @@
|
|||||||
|
|
||||||
`lyngio` is a separate library that extends the Lyng core (`lynglib`) with powerful, multiplatform, and secure I/O capabilities.
|
`lyngio` is a separate library that extends the Lyng core (`lynglib`) with powerful, multiplatform, and secure I/O capabilities.
|
||||||
|
|
||||||
> **Important native networking limit:** `lyng.io.net` on current native targets is suitable for modest workloads, local tools, and test servers, but not yet for high-connection-count production servers. For serious HTTP/TCP serving, prefer the JVM target for now. If native high-concurrency networking matters for your use case, please open or upvote an issue at <https://github.com/sergeych/lyng/issues>.
|
|
||||||
|
|
||||||
#### Why a separate module?
|
#### Why a separate module?
|
||||||
|
|
||||||
1. **Security:** I/O and process execution are sensitive operations. By keeping them in a separate module, we ensure that the Lyng core remains 100% safe by default. You only enable what you explicitly need.
|
1. **Security:** I/O and process execution are sensitive operations. By keeping them in a separate module, we ensure that the Lyng core remains 100% safe by default. You only enable what you explicitly need.
|
||||||
2. **Footprint:** Not every script needs filesystem or process access. Keeping these as a separate module helps minimize the dependency footprint for small embedded projects.
|
2. **Footprint:** Not every script needs filesystem or process access. Keeping these as a separate module helps minimize the dependency footprint for small embedded projects.
|
||||||
3. **Control:** `lyngio` provides fine-grained security policies (`FsAccessPolicy`, `ProcessAccessPolicy`, `ConsoleAccessPolicy`) that allow you to control exactly what a script can do.
|
3. **Control:** `lyngio` provides fine-grained security policies (`FsAccessPolicy`, `ProcessAccessPolicy`) that allow you to control exactly what a script can do.
|
||||||
|
|
||||||
#### Included Modules
|
#### Included Modules
|
||||||
|
|
||||||
- **[lyng.io.db](lyng.io.db.md):** Portable SQL database access. Provides `Database`, `SqlTransaction`, `ResultSet`, SQLite support through `lyng.io.db.sqlite`, and JVM JDBC support through `lyng.io.db.jdbc`.
|
|
||||||
- **[lyng.io.fs](lyng.io.fs.md):** Async filesystem access. Provides the `Path` class for file/directory operations, streaming, and globbing.
|
- **[lyng.io.fs](lyng.io.fs.md):** Async filesystem access. Provides the `Path` class for file/directory operations, streaming, and globbing.
|
||||||
- **[lyng.io.process](lyng.io.process.md):** External process execution and shell commands. Provides `Process`, `RunningProcess`, and `Platform` information.
|
- **[lyng.io.process](lyng.io.process.md):** External process execution and shell commands. Provides `Process`, `RunningProcess`, and `Platform` information.
|
||||||
- **[lyng.io.console](lyng.io.console.md):** Rich console/TTY access. Provides `Console` capability detection, geometry, output, and iterable events.
|
|
||||||
- **[lyng.io.http](lyng.io.http.md):** HTTP/HTTPS client access. Provides `Http`, `HttpRequest`, `HttpResponse`, and `HttpHeaders`.
|
|
||||||
- **[lyng.io.http.server](lyng.io.http.server.md):** Minimal HTTP/1.1 and WebSocket server. Provides `HttpServer`, `Router`, `ServerRequest`, `RequestContext`, and `ServerWebSocket`.
|
|
||||||
- **[lyng.io.ws](lyng.io.ws.md):** WebSocket client access. Provides `Ws`, `WsSession`, and `WsMessage`.
|
|
||||||
- **[lyng.io.net](lyng.io.net.md):** Transport networking. Provides `Net`, `TcpSocket`, `TcpServer`, `UdpSocket`, and `SocketAddress`.
|
|
||||||
- **Shared networking type packages:** `lyng.io.http.types`, `lyng.io.ws.types`, and `lyng.io.net.types` expose reusable value types such as `HttpHeaders`, `WsMessage`, `IpVersion`, `SocketAddress`, and `Datagram` when host code wants type-only imports without installing the corresponding capability object module.
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@ -45,58 +36,26 @@ dependencies {
|
|||||||
To use `lyngio` modules in your scripts, you must install them into your Lyng scope and provide a security policy.
|
To use `lyngio` modules in your scripts, you must install them into your Lyng scope and provide a security policy.
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
import net.sergeych.lyng.EvalSession
|
import net.sergeych.lyng.Script
|
||||||
import net.sergeych.lyng.io.db.createDbModule
|
|
||||||
import net.sergeych.lyng.io.db.jdbc.createJdbcModule
|
|
||||||
import net.sergeych.lyng.io.db.sqlite.createSqliteModule
|
|
||||||
import net.sergeych.lyng.io.fs.createFs
|
import net.sergeych.lyng.io.fs.createFs
|
||||||
import net.sergeych.lyng.io.process.createProcessModule
|
import net.sergeych.lyng.io.process.createProcessModule
|
||||||
import net.sergeych.lyng.io.console.createConsoleModule
|
|
||||||
import net.sergeych.lyng.io.http.createHttpModule
|
|
||||||
import net.sergeych.lyng.io.net.createNetModule
|
|
||||||
import net.sergeych.lyng.io.ws.createWsModule
|
|
||||||
import net.sergeych.lyngio.fs.security.PermitAllAccessPolicy
|
import net.sergeych.lyngio.fs.security.PermitAllAccessPolicy
|
||||||
import net.sergeych.lyngio.process.security.PermitAllProcessAccessPolicy
|
import net.sergeych.lyngio.process.security.PermitAllProcessAccessPolicy
|
||||||
import net.sergeych.lyngio.console.security.PermitAllConsoleAccessPolicy
|
|
||||||
import net.sergeych.lyngio.http.security.PermitAllHttpAccessPolicy
|
|
||||||
import net.sergeych.lyngio.net.security.PermitAllNetAccessPolicy
|
|
||||||
import net.sergeych.lyngio.ws.security.PermitAllWsAccessPolicy
|
|
||||||
|
|
||||||
suspend fun runMyScript() {
|
suspend fun runMyScript() {
|
||||||
val session = EvalSession()
|
val scope = Script.newScope()
|
||||||
val scope = session.getScope()
|
|
||||||
|
|
||||||
// Install modules with policies
|
// Install modules with policies
|
||||||
createDbModule(scope)
|
|
||||||
createJdbcModule(scope)
|
|
||||||
createSqliteModule(scope)
|
|
||||||
createFs(PermitAllAccessPolicy, scope)
|
createFs(PermitAllAccessPolicy, scope)
|
||||||
createProcessModule(PermitAllProcessAccessPolicy, scope)
|
createProcessModule(PermitAllProcessAccessPolicy, scope)
|
||||||
createConsoleModule(PermitAllConsoleAccessPolicy, scope)
|
|
||||||
createHttpModule(PermitAllHttpAccessPolicy, scope)
|
|
||||||
createNetModule(PermitAllNetAccessPolicy, scope)
|
|
||||||
createWsModule(PermitAllWsAccessPolicy, scope)
|
|
||||||
|
|
||||||
// Now scripts can import them
|
// Now scripts can import them
|
||||||
session.eval("""
|
scope.eval("""
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.jdbc
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
import lyng.io.fs
|
import lyng.io.fs
|
||||||
import lyng.io.process
|
import lyng.io.process
|
||||||
import lyng.io.console
|
|
||||||
import lyng.io.http
|
|
||||||
import lyng.io.net
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
println("H2 JDBC available: " + (openH2("mem:demo;DB_CLOSE_DELAY=-1") != null))
|
|
||||||
println("SQLite available: " + (openSqlite(":memory:") != null))
|
|
||||||
println("Working dir: " + Path(".").readUtf8())
|
println("Working dir: " + Path(".").readUtf8())
|
||||||
println("OS: " + Platform.details().name)
|
println("OS: " + Platform.details().name)
|
||||||
println("TTY: " + Console.isTty())
|
|
||||||
println("HTTP available: " + Http.isSupported())
|
|
||||||
println("TCP available: " + Net.isTcpAvailable())
|
|
||||||
println("WS available: " + Ws.isSupported())
|
|
||||||
""")
|
""")
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
@ -108,39 +67,21 @@ suspend fun runMyScript() {
|
|||||||
`lyngio` is built with a "Secure by Default" philosophy. Every I/O or process operation is checked against a policy.
|
`lyngio` is built with a "Secure by Default" philosophy. Every I/O or process operation is checked against a policy.
|
||||||
|
|
||||||
- **Filesystem Security:** Implement `FsAccessPolicy` to restrict access to specific paths or operations (e.g., read-only access to a sandbox directory).
|
- **Filesystem Security:** Implement `FsAccessPolicy` to restrict access to specific paths or operations (e.g., read-only access to a sandbox directory).
|
||||||
- **Database Installation:** Database access is still explicit-capability style. The host must install `lyng.io.db` and at least one provider such as `lyng.io.db.sqlite` or `lyng.io.db.jdbc`; otherwise scripts cannot open databases.
|
|
||||||
- **Process Security:** Implement `ProcessAccessPolicy` to restrict which executables can be run or to disable shell execution entirely.
|
- **Process Security:** Implement `ProcessAccessPolicy` to restrict which executables can be run or to disable shell execution entirely.
|
||||||
- **Console Security:** Implement `ConsoleAccessPolicy` to control output writes, event reads, and raw mode switching.
|
|
||||||
- **HTTP Security:** Implement `HttpAccessPolicy` to restrict which requests scripts may send.
|
|
||||||
- **Transport Security:** Implement `NetAccessPolicy` to restrict DNS resolution and TCP/UDP socket operations.
|
|
||||||
- **WebSocket Security:** Implement `WsAccessPolicy` to restrict websocket connects and message flow.
|
|
||||||
|
|
||||||
For more details, see the specific module documentation:
|
For more details, see the specific module documentation:
|
||||||
- [Database Module Details](lyng.io.db.md)
|
|
||||||
- [Filesystem Security Details](lyng.io.fs.md#access-policy-security)
|
- [Filesystem Security Details](lyng.io.fs.md#access-policy-security)
|
||||||
- [Process Security Details](lyng.io.process.md#security-policy)
|
- [Process Security Details](lyng.io.process.md#security-policy)
|
||||||
- [Console Module Details](lyng.io.console.md)
|
|
||||||
- [HTTP Module Details](lyng.io.http.md)
|
|
||||||
- [HTTP Server Module Details](lyng.io.http.server.md)
|
|
||||||
- [Transport Networking Details](lyng.io.net.md)
|
|
||||||
- [WebSocket Module Details](lyng.io.ws.md)
|
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
#### Platform Support Overview
|
#### Platform Support Overview
|
||||||
|
|
||||||
| Platform | lyng.io.db/sqlite | lyng.io.db/jdbc | lyng.io.fs | lyng.io.process | lyng.io.console | lyng.io.http | lyng.io.ws | lyng.io.net |
|
| Platform | lyng.io.fs | lyng.io.process |
|
||||||
| :--- | :---: | :---: | :---: | :---: | :---: | :---: | :---: | :---: |
|
| :--- | :---: | :---: |
|
||||||
| **JVM** | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ |
|
| **JVM** | ✅ | ✅ |
|
||||||
| **Linux Native** | ✅ | ❌ | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ |
|
| **Native (Linux/macOS)** | ✅ | ✅ |
|
||||||
| **Apple Native** | ❌ | ❌ | ✅ | ✅ | ✅ | ✅ | ✅ | ⚠️ |
|
| **Native (Windows)** | ✅ | 🚧 (Planned) |
|
||||||
| **Windows Native** | ❌ | ❌ | ✅ | ❌ | ✅ | ✅ | ✅ | ❌ |
|
| **Android** | ✅ | ❌ |
|
||||||
| **Android** | ⚠️ | ❌ | ✅ | ❌ | ⚠️ | ✅ | ✅ | ✅ |
|
| **NodeJS** | ✅ | ❌ |
|
||||||
| **JS / Node** | ❌ | ❌ | ✅ | ❌ | ⚠️ | ✅ | ✅ | ✅ |
|
| **Browser / Wasm** | ✅ (In-memory) | ❌ |
|
||||||
| **JS / Browser** | ❌ | ❌ | ✅ (in-memory) | ❌ | ⚠️ | ✅ | ✅ | ❌ |
|
|
||||||
| **Wasm** | ❌ | ❌ | ✅ (in-memory) | ❌ | ⚠️ | ❌ | ❌ | ❌ |
|
|
||||||
|
|
||||||
Legend:
|
|
||||||
- `✅` supported
|
|
||||||
- `⚠️` available but environment-dependent or not fully host-verified yet
|
|
||||||
- `❌` unsupported
|
|
||||||
|
|||||||
99
docs/math.md
99
docs/math.md
@ -60,13 +60,8 @@ but:
|
|||||||
|
|
||||||
## Round and range
|
## Round and range
|
||||||
|
|
||||||
The following functions return the argument unchanged if it is `Int`.
|
The following functions return its argument if it is `Int`,
|
||||||
|
or transformed `Real` otherwise.
|
||||||
For `Decimal`:
|
|
||||||
- `floor(x)`, `ceil(x)`, and `round(x)` currently use exact decimal operations
|
|
||||||
- the result stays `Decimal`
|
|
||||||
|
|
||||||
For `Real`, the result is a transformed `Real`.
|
|
||||||
|
|
||||||
| name | description |
|
| name | description |
|
||||||
|----------------|--------------------------------------------------------|
|
|----------------|--------------------------------------------------------|
|
||||||
@ -77,14 +72,6 @@ For `Real`, the result is a transformed `Real`.
|
|||||||
|
|
||||||
## Lyng math functions
|
## Lyng math functions
|
||||||
|
|
||||||
Decimal note:
|
|
||||||
- all scalar math helpers accept `Decimal`
|
|
||||||
- `abs(x)` stays exact for `Decimal`
|
|
||||||
- `pow(x, y)` is exact for `Decimal` when `y` is an integral exponent
|
|
||||||
- the remaining `Decimal` cases currently use a temporary bridge:
|
|
||||||
`Decimal -> Real -> host math -> Decimal`
|
|
||||||
- this is temporary; native decimal implementations are planned
|
|
||||||
|
|
||||||
| name | meaning |
|
| name | meaning |
|
||||||
|-----------|------------------------------------------------------|
|
|-----------|------------------------------------------------------|
|
||||||
| sin(x) | sine |
|
| sin(x) | sine |
|
||||||
@ -104,7 +91,7 @@ Decimal note:
|
|||||||
| log10(x) | $log_{10}(x)$ |
|
| log10(x) | $log_{10}(x)$ |
|
||||||
| pow(x, y) | ${x^y}$ |
|
| pow(x, y) | ${x^y}$ |
|
||||||
| sqrt(x) | $ \sqrt {x}$ |
|
| sqrt(x) | $ \sqrt {x}$ |
|
||||||
| abs(x) | absolute value of x. Int if x is Int, Decimal if x is Decimal, Real otherwise |
|
| abs(x) | absolute value of x. Int if x is Int, Real otherwise |
|
||||||
| clamp(x, range) | limit x to be inside range boundaries |
|
| clamp(x, range) | limit x to be inside range boundaries |
|
||||||
|
|
||||||
For example:
|
For example:
|
||||||
@ -117,92 +104,12 @@ For example:
|
|||||||
assert( abs(-1) is Int)
|
assert( abs(-1) is Int)
|
||||||
assert( abs(-2.21) == 2.21 )
|
assert( abs(-2.21) == 2.21 )
|
||||||
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
// Decimal-aware math works too. Some functions are exact, some still bridge through Real temporarily:
|
|
||||||
assert( (abs("-2.5".d) as Decimal).toStringExpanded() == "2.5" )
|
|
||||||
assert( (floor("2.9".d) as Decimal).toStringExpanded() == "2" )
|
|
||||||
assert( sin("0.5".d) is Decimal )
|
|
||||||
|
|
||||||
// clamp() limits value to the range:
|
// clamp() limits value to the range:
|
||||||
assert( clamp(15, 0..10) == 10 )
|
assert( clamp(15, 0..10) == 10 )
|
||||||
assert( clamp(-5, 0..10) == 0 )
|
assert( clamp(-5, 0..10) == 0 )
|
||||||
assert( 5.clamp(0..10) == 5 )
|
assert( 5.clamp(0..10) == 5 )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
## Linear algebra: `lyng.matrix`
|
|
||||||
|
|
||||||
For vectors and dense matrices, import `lyng.matrix`:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
```
|
|
||||||
|
|
||||||
It provides:
|
|
||||||
|
|
||||||
- `Vector`
|
|
||||||
- `Matrix`
|
|
||||||
- `vector(values)`
|
|
||||||
- `matrix(rows)`
|
|
||||||
|
|
||||||
Core operations include:
|
|
||||||
|
|
||||||
- matrix addition and subtraction
|
|
||||||
- matrix-matrix multiplication
|
|
||||||
- matrix-vector multiplication
|
|
||||||
- transpose
|
|
||||||
- determinant
|
|
||||||
- inverse
|
|
||||||
- linear solve
|
|
||||||
- vector dot, norm, normalize, cross, outer product
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
val b: Matrix = matrix([[7, 8], [9, 10], [11, 12]])
|
|
||||||
val product: Matrix = a * b
|
|
||||||
assertEquals([[58.0, 64.0], [139.0, 154.0]], product.toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
Matrices also support two-axis bracket indexing and slicing:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val m: Matrix = matrix([[1, 2, 3], [4, 5, 6], [7, 8, 9]])
|
|
||||||
assertEquals(6.0, m[1, 2])
|
|
||||||
val sub: Matrix = m[0..1, 1..2]
|
|
||||||
assertEquals([[2.0, 3.0], [5.0, 6.0]], sub.toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
See [Matrix](Matrix.md) for the full API.
|
|
||||||
|
|
||||||
## Random values
|
|
||||||
|
|
||||||
Lyng stdlib provides a global random singleton and deterministic seeded generators:
|
|
||||||
|
|
||||||
| name | meaning |
|
|
||||||
|--------------------------|---------|
|
|
||||||
| Random.nextInt() | random `Int` from full platform range |
|
|
||||||
| Random.nextFloat() | random `Real` in `[0,1)` |
|
|
||||||
| Random.next(range) | random value from the given finite range |
|
|
||||||
| Random.seeded(seed) | creates deterministic generator |
|
|
||||||
| SeededRandom.nextInt() | deterministic random `Int` |
|
|
||||||
| SeededRandom.nextFloat() | deterministic random `Real` in `[0,1)` |
|
|
||||||
| SeededRandom.next(range) | deterministic random value from range |
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
|
|
||||||
val rng = Random.seeded(1234)
|
|
||||||
assert( rng.next(1..10) in 1..10 )
|
|
||||||
assert( rng.next('a'..<'f') in 'a'..<'f' )
|
|
||||||
assert( rng.next(0.0..<1.0) >= 0.0 )
|
|
||||||
assert( rng.next(0.0..<1.0) < 1.0 )
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Scientific constant
|
## Scientific constant
|
||||||
|
|
||||||
| name | meaning |
|
| name | meaning |
|
||||||
|
|||||||
@ -4,7 +4,7 @@
|
|||||||
|
|
||||||
Before kotlin 2.0, there was an excellent library, kotlinx.datetime, which was widely used everywhere, also in Lyng and its dependencies.
|
Before kotlin 2.0, there was an excellent library, kotlinx.datetime, which was widely used everywhere, also in Lyng and its dependencies.
|
||||||
|
|
||||||
When Kotlin 2.0 was released, or soon after, JetBrains made a perplexing decision to remove `Instant` and `Clock` from kotlinx.datetime and replace it with _yet experimental_ analogs in `kotlin.time`.
|
When kotlin 2.0 was released, or soon after, JetBrains made an exptic decision to remove `Instant` and `Clock` from kotlinx.datetime and replace it with _yet experimental_ analogs in `kotlin.time`.
|
||||||
|
|
||||||
The problem is, these were not quite the same (these weren't `@Serializable`!), so people didn't migrate with ease. Okay, then JetBrains decided to not only deprecate it but also make them unusable on Apple targets. It sort of split auditories of many published libraries to those who hate JetBrains and Apple and continue to use 1.9-2.0 compatible versions that no longer work with Kotlin 2.2 on Apple targets (but work pretty well with earlier Kotlin or on other platforms).
|
The problem is, these were not quite the same (these weren't `@Serializable`!), so people didn't migrate with ease. Okay, then JetBrains decided to not only deprecate it but also make them unusable on Apple targets. It sort of split auditories of many published libraries to those who hate JetBrains and Apple and continue to use 1.9-2.0 compatible versions that no longer work with Kotlin 2.2 on Apple targets (but work pretty well with earlier Kotlin or on other platforms).
|
||||||
|
|
||||||
@ -12,14 +12,14 @@ Later JetBrains added serializers for their new `Instant` and `Clock` types, but
|
|||||||
|
|
||||||
## Solution
|
## Solution
|
||||||
|
|
||||||
We hereby publish a new version of Lyng, 1.0.8-SNAPSHOT, which uses `kotlin.time.Instant` and `kotlin.time.Clock` instead of `kotlinx.datetime.Instant` and `kotlinx.datetime.Clock`. It is in other aspects compatible also with Lynon encoded binaries. You might need to migrate your code to use `kotlin.time` types. (LocalDateTime/TimeZone still come from `kotlinx.datetime`.)
|
We hereby publish a new version of Lyng, 1.0.8-SNAPSHOT, which uses `ktlin.time.Instant` and `kotlin.time.Clock` instead of `kotlinx.datetime.Instant` and `kotlinx.datetime.Clock; it is in other aspects compatible also with Lynon encoded binaries. Still you might need to migrate your code to use `kotlinx.datetime` types.
|
||||||
|
|
||||||
So, if you are getting errors with new version, please do:
|
So, if you are getting errors with new version, plase do:
|
||||||
|
|
||||||
- upgrade to Kotlin 2.2
|
- upgrade to Kotlin 2.2
|
||||||
- upgrade to Lyng 1.0.8-SNAPSHOT
|
- upgrade to Lyng 1.0.8-SNAPSHOT
|
||||||
- replace in your code imports (or other uses) of `kotlinx.datetime.Clock` to `kotlin.time.Clock` and `kotlinx.datetime.Instant` to `kotlin.time.Instant`.
|
- replace in your code imports (or other uses) of`kotlinx.datetime.Clock` to `kotlin.time.Clock` and `kotlinx.datetime.Instant` to `kotlin.time.Instant`.
|
||||||
|
|
||||||
This should solve the problem and hopefully we'll see no more such "brilliant" ideas from IDEA ideologspersons.
|
This should solve the problem and hopefully we'll see no more suh a brillant ideas from IDEA ideologspersons.
|
||||||
|
|
||||||
Sorry for inconvenience and send a ray of hate to JetBrains ;)
|
Sorry for inconvenicence and send a ray of hate to JetBrains ;)
|
||||||
@ -32,36 +32,10 @@ Depending on the platform, these coroutines may be executed on different CPU and
|
|||||||
assert(xIsCalled)
|
assert(xIsCalled)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
This example shows how to launch a coroutine with `launch` which returns [Deferred] instance, the latter have ways to await for the coroutine completion, cancel it if it is no longer needed, and retrieve possible result.
|
This example shows how to launch a coroutine with `launch` which returns [Deferred] instance, the latter have ways to await for the coroutine completion and retrieve possible result.
|
||||||
|
|
||||||
Launch has the only argument which should be a callable (lambda usually) that is run in parallel (or cooperatively in parallel), and return anything as the result.
|
Launch has the only argument which should be a callable (lambda usually) that is run in parallel (or cooperatively in parallel), and return anything as the result.
|
||||||
|
|
||||||
When you have an iterable of deferreds, use `joinAll()` to await all of them and collect results in input order:
|
|
||||||
|
|
||||||
val jobs = (1..4).map { n ->
|
|
||||||
launch {
|
|
||||||
delay(1)
|
|
||||||
n * 10
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assertEquals([10, 20, 30, 40], jobs.joinAll())
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
If you no longer need the result, cancel the deferred. Awaiting a cancelled deferred throws `CancellationException`:
|
|
||||||
|
|
||||||
var reached = false
|
|
||||||
val work = launch {
|
|
||||||
delay(100)
|
|
||||||
reached = true
|
|
||||||
"ok"
|
|
||||||
}
|
|
||||||
work.cancel()
|
|
||||||
assertThrows(CancellationException) { work.await() }
|
|
||||||
assert(work.isCancelled)
|
|
||||||
assert(!work.isActive)
|
|
||||||
assert(!reached)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Synchronization: Mutex
|
## Synchronization: Mutex
|
||||||
|
|
||||||
Suppose we have a resource, that could be used concurrently, a counter in our case. If we won't protect it, concurrent usage cause RC, Race Condition, providing wrong result:
|
Suppose we have a resource, that could be used concurrently, a counter in our case. If we won't protect it, concurrent usage cause RC, Race Condition, providing wrong result:
|
||||||
@ -75,7 +49,7 @@ Suppose we have a resource, that could be used concurrently, a counter in our ca
|
|||||||
delay(100)
|
delay(100)
|
||||||
counter = c + 1
|
counter = c + 1
|
||||||
}
|
}
|
||||||
}.forEach { (it as Deferred).await() }
|
}.forEach { it.await() }
|
||||||
assert(counter < 50) { "counter is "+counter }
|
assert(counter < 50) { "counter is "+counter }
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
@ -90,12 +64,13 @@ Using [Mutex] makes it all working:
|
|||||||
launch {
|
launch {
|
||||||
// slow increment:
|
// slow increment:
|
||||||
mutex.withLock {
|
mutex.withLock {
|
||||||
val c = counter ?: 0
|
val c = counter
|
||||||
|
delay(10)
|
||||||
counter = c + 1
|
counter = c + 1
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}.forEach { (it as Deferred).await() }
|
}.forEach { it.await() }
|
||||||
assert(counter in 1..4)
|
assertEquals(4, counter)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
now everything works as expected: `mutex.withLock` makes them all be executed in sequence, not in parallel.
|
now everything works as expected: `mutex.withLock` makes them all be executed in sequence, not in parallel.
|
||||||
@ -230,73 +205,6 @@ Flows allow easy transforming of any [Iterable]. See how the standard Lyng libra
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
## Channel
|
|
||||||
|
|
||||||
A [Channel] is a **hot pipe** between coroutines: values are pushed in by a producer and pulled out by a consumer, with each value consumed exactly once.
|
|
||||||
|
|
||||||
Unlike a `Flow` (which is cold and re-runs its generator on every collection), a `Channel` is stateful — the right tool for classic _producer / consumer_ work.
|
|
||||||
|
|
||||||
val ch = Channel() // rendezvous: sender waits for receiver
|
|
||||||
|
|
||||||
val producer = launch {
|
|
||||||
for (i in 1..5) ch.send(i)
|
|
||||||
ch.close() // signal: no more values
|
|
||||||
}
|
|
||||||
|
|
||||||
var item = ch.receive() // suspends until a value is ready
|
|
||||||
while (item != null) {
|
|
||||||
println(item)
|
|
||||||
item = ch.receive()
|
|
||||||
}
|
|
||||||
// prints 1 2 3 4 5
|
|
||||||
|
|
||||||
`receive()` returns `null` when the channel is both closed and fully drained — that is the idiomatic loop termination condition.
|
|
||||||
|
|
||||||
Channels can also be buffered so the producer can run ahead:
|
|
||||||
|
|
||||||
val ch = Channel(4) // buffer up to 4 items without blocking
|
|
||||||
|
|
||||||
ch.send(10)
|
|
||||||
ch.send(20)
|
|
||||||
ch.send(30)
|
|
||||||
ch.close()
|
|
||||||
|
|
||||||
assertEquals(10, ch.receive())
|
|
||||||
assertEquals(20, ch.receive())
|
|
||||||
assertEquals(30, ch.receive())
|
|
||||||
assertEquals(null, ch.receive()) // drained
|
|
||||||
|
|
||||||
For the full API — including `tryReceive`, `Channel.UNLIMITED`, and the fan-out / ping-pong patterns — see the [Channel] reference page.
|
|
||||||
|
|
||||||
## LaunchPool
|
|
||||||
|
|
||||||
When you need **bounded concurrency** — run at most *N* tasks at the same time without spawning a new coroutine per task — use [LaunchPool]:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val pool = LaunchPool(4) // at most 4 tasks run in parallel
|
|
||||||
|
|
||||||
val jobs = (1..20).map { n ->
|
|
||||||
pool.launch { expensiveCompute(n) }
|
|
||||||
}
|
|
||||||
pool.closeAndJoin() // wait for all tasks to complete
|
|
||||||
|
|
||||||
val results = jobs.joinAll()
|
|
||||||
```
|
|
||||||
|
|
||||||
Exceptions thrown inside a submitted lambda are captured in the returned `Deferred` and do not crash the pool, so other tasks continue running normally.
|
|
||||||
|
|
||||||
See [LaunchPool] for the full API including bounded queues and cancellation.
|
|
||||||
|
|
||||||
[LaunchPool]: LaunchPool.md
|
|
||||||
|
|
||||||
| | Flow | Channel |
|
|
||||||
|---|---|---|
|
|
||||||
| **temperature** | cold (lazy) | hot (eager) |
|
|
||||||
| **replay** | every collector gets a fresh run | each item consumed once |
|
|
||||||
| **consumers** | any number, each gets all items | one receiver per item |
|
|
||||||
| **typical use** | transform pipelines, sequences | producer–consumer, fan-out |
|
|
||||||
|
|
||||||
[Channel]: Channel.md
|
|
||||||
|
|
||||||
[Iterable]: Iterable.md
|
[Iterable]: Iterable.md
|
||||||
|
|
||||||
@ -316,14 +224,17 @@ Future work: introduce thread‑safe pooling (e.g., per‑thread pools or confin
|
|||||||
|
|
||||||
### Closures inside coroutine helpers (launch/flow)
|
### Closures inside coroutine helpers (launch/flow)
|
||||||
|
|
||||||
Closures executed by `launch { ... }` and `flow { ... }` use **compile‑time resolution** just like any other Lyng code:
|
Closures executed by `launch { ... }` and `flow { ... }` resolve names using the `ClosureScope` rules:
|
||||||
|
|
||||||
- **Captured locals are slots**: outer locals are resolved at compile time and captured as frame‑slot references, so they remain visible across suspension points.
|
1. **Current frame locals and arguments**: Variables defined within the current closure execution.
|
||||||
- **Members are statically resolved**: member access requires a statically known receiver type or an explicit cast (except `Object` members).
|
2. **Captured lexical ancestry**: Outer local variables captured at the site where the closure was defined (the "lexical environment").
|
||||||
- **No runtime fallbacks**: there is no dynamic name lookup or “search parent scopes” at runtime for missing symbols.
|
3. **Captured receiver members**: If the closure was defined within a class or explicitly bound to an object, it checks members of that object (`this`), following MRO and respecting visibility.
|
||||||
|
4. **Caller environment**: Falls back to the calling context (e.g., the caller's `this` or local variables).
|
||||||
|
5. **Global/Module fallbacks**: Final check for module-level constants and global functions.
|
||||||
|
|
||||||
Implications:
|
Implications:
|
||||||
- Global helpers like `delay(ms)` and `yield()` must be imported/known at compile time.
|
- Outer locals (e.g., `counter`) stay visible across suspension points.
|
||||||
- If you need dynamic access, use explicit helpers (e.g., `dynamic { ... }`) rather than relying on scope resolution.
|
- Global helpers like `delay(ms)` and `yield()` are available from inside closures.
|
||||||
|
- If you write your own async helpers, execute user lambdas under `ClosureScope(callScope, capturedCreatorScope)` and avoid manual ancestry walking.
|
||||||
|
|
||||||
See also: [Scopes and Closures: compile-time resolution](scopes_and_closures.md)
|
See also: [Scopes and Closures: resolution and safety](scopes_and_closures.md)
|
||||||
|
|||||||
@ -114,12 +114,10 @@ When running end‑to‑end “book” workloads or heavier benches, you can ena
|
|||||||
Flags are mutable at runtime, e.g.:
|
Flags are mutable at runtime, e.g.:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
runTest {
|
PerfFlags.ARG_BUILDER = false
|
||||||
PerfFlags.ARG_BUILDER = false
|
val r1 = (Scope().eval(script) as ObjInt).value
|
||||||
val r1 = (EvalSession(Scope()).eval(script) as ObjInt).value
|
PerfFlags.ARG_BUILDER = true
|
||||||
PerfFlags.ARG_BUILDER = true
|
val r2 = (Scope().eval(script) as ObjInt).value
|
||||||
val r2 = (EvalSession(Scope()).eval(script) as ObjInt).value
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Reset flags at the end of a test to avoid impacting other tests.
|
Reset flags at the end of a test to avoid impacting other tests.
|
||||||
@ -621,3 +619,4 @@ Reproduce
|
|||||||
Notes
|
Notes
|
||||||
- Negative caches are installed only after a real miss throws (cache‑after‑miss), preserving error semantics and invalidation on `layoutVersion` changes.
|
- Negative caches are installed only after a real miss throws (cache‑after‑miss), preserving error semantics and invalidation on `layoutVersion` changes.
|
||||||
- IndexRef PIC augments the existing direct path and uses move‑to‑front promotion; it is keyed on `(classId, layoutVersion)` like other PICs.
|
- IndexRef PIC augments the existing direct path and uses move‑to‑front promotion; it is keyed on `(classId, layoutVersion)` like other PICs.
|
||||||
|
|
||||||
|
|||||||
@ -1,62 +0,0 @@
|
|||||||
## Pi Spigot JVM Baseline
|
|
||||||
|
|
||||||
Saved on April 4, 2026 before the `List<Int>` indexed-access follow-up fix.
|
|
||||||
|
|
||||||
Benchmark target:
|
|
||||||
- [examples/pi-bench.py](/home/sergeych/dev/lyng/examples/pi-bench.py)
|
|
||||||
- [examples/pi-bench.lyng](../examples/pi-bench.lyng)
|
|
||||||
|
|
||||||
Execution path:
|
|
||||||
- Python: `python3 examples/pi-bench.py`
|
|
||||||
- Lyng JVM: `./gradlew :lyng:runJvm --args='/home/sergeych/dev/lyng/examples/pi-bench.lyng'`
|
|
||||||
- Constraint: do not use Kotlin/Native `lyng` CLI for perf comparisons
|
|
||||||
|
|
||||||
Baseline measurements:
|
|
||||||
- Python full script: `167 ms`
|
|
||||||
- Lyng JVM full script: `1.287097604 s`
|
|
||||||
- Python warm function average over 5 runs: `126.126 ms`
|
|
||||||
- Lyng JVM warm function average over 5 runs: about `1071.6 ms`
|
|
||||||
|
|
||||||
Baseline ratio:
|
|
||||||
- Full script: about `7.7x` slower on Lyng JVM
|
|
||||||
- Warm function only: about `8.5x` slower on Lyng JVM
|
|
||||||
|
|
||||||
Primary finding at baseline:
|
|
||||||
- The hot `reminders[j]` accesses in `piSpigot` were still lowered through boxed object index ops and boxed arithmetic.
|
|
||||||
- Newly added `GET_INDEX_INT` and `SET_INDEX_INT` only reached `pi`, not `reminders`.
|
|
||||||
- Root cause: initializer element inference handled list literals, but not `List.fill(boxes) { 2 }`, so `reminders` did not become known `List<Int>` at compile time.
|
|
||||||
|
|
||||||
## After Optimizations 1-4
|
|
||||||
|
|
||||||
Follow-up change:
|
|
||||||
- propagate inferred lambda return class into bytecode compilation
|
|
||||||
- infer `List.fill(...)` element type from the fill lambda
|
|
||||||
- lower `reminders[j]` reads and writes to `GET_INDEX_INT` and `SET_INDEX_INT`
|
|
||||||
- add primitive-backed `ObjList` storage for all-int lists
|
|
||||||
- lower `List.fill(Int) { Int }` to `LIST_FILL_INT`
|
|
||||||
- stop boxing the integer index inside `GET_INDEX_INT` / `SET_INDEX_INT`
|
|
||||||
|
|
||||||
Verification:
|
|
||||||
- `piSpigot` disassembly now contains typed ops for `reminders`, for example:
|
|
||||||
- `GET_INDEX_INT s5(reminders), s10(j), ...`
|
|
||||||
- `SET_INDEX_INT s5(reminders), s10(j), ...`
|
|
||||||
|
|
||||||
Post-change measurements using `jlyng`:
|
|
||||||
- Full script: `655.819559 ms`
|
|
||||||
- Warm 5-run total: `1.430945810 s`
|
|
||||||
- Warm average per run: about `286.2 ms`
|
|
||||||
|
|
||||||
Observed improvement vs baseline:
|
|
||||||
- Full script: about `1.96x` faster (`1.287 s -> 0.656 s`)
|
|
||||||
- Warm function: about `3.74x` faster (`1071.6 ms -> 286.2 ms`)
|
|
||||||
|
|
||||||
Residual gap vs Python baseline:
|
|
||||||
- Full script: Lyng JVM is still about `3.9x` slower than Python (`655.8 ms` vs `167 ms`)
|
|
||||||
- Warm function: Lyng JVM is still about `2.3x` slower than Python (`286.2 ms` vs `126.126 ms`)
|
|
||||||
|
|
||||||
Current benchmark-test snapshot (`n=200`, JVM test harness):
|
|
||||||
- `optimized-int-division-rval-off`: `135 ms`
|
|
||||||
- `optimized-int-division-rval-on`: `125 ms`
|
|
||||||
- `piSpigot` bytecode now contains:
|
|
||||||
- `LIST_FILL_INT` for both `pi` and `reminders`
|
|
||||||
- `GET_INDEX_INT` / `SET_INDEX_INT` for the hot indexed loop
|
|
||||||
@ -1,65 +0,0 @@
|
|||||||
/**
|
|
||||||
* Sample .lyng.d file for IDE support.
|
|
||||||
* Demonstrates declarations and doc comments.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/** Simple function with default and named parameters. */
|
|
||||||
extern fun connect(url: String, timeoutMs: Int = 5000): Client
|
|
||||||
|
|
||||||
/** Type alias with generics. */
|
|
||||||
type NameMap = Map<String, String>
|
|
||||||
|
|
||||||
/** Multiple inheritance via interfaces. */
|
|
||||||
interface A { abstract fun a(): Int }
|
|
||||||
interface B { abstract fun b(): Int }
|
|
||||||
|
|
||||||
/** A concrete class implementing both. */
|
|
||||||
class Multi(name: String) : A, B {
|
|
||||||
/** Public field. */
|
|
||||||
val id: Int = 0
|
|
||||||
|
|
||||||
/** Mutable property with accessors. */
|
|
||||||
var size: Int
|
|
||||||
get() = 0
|
|
||||||
set(v) { }
|
|
||||||
|
|
||||||
/** Instance method. */
|
|
||||||
fun a(): Int = 1
|
|
||||||
fun b(): Int = 2
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Nullable and dynamic types. */
|
|
||||||
extern val dynValue: dynamic
|
|
||||||
extern var dynVar: dynamic?
|
|
||||||
|
|
||||||
/** Delegated property provider. */
|
|
||||||
class LazyBox(val create) {
|
|
||||||
fun getValue(thisRef, name) = create()
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Delegated property using provider. */
|
|
||||||
val cached by LazyBox { 42 }
|
|
||||||
|
|
||||||
/** Delegated function. */
|
|
||||||
object RpcDelegate {
|
|
||||||
fun invoke(thisRef, name, args...) = Unset
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Remote function proxy. */
|
|
||||||
fun remoteCall by RpcDelegate
|
|
||||||
|
|
||||||
/** Singleton object. */
|
|
||||||
object Settings {
|
|
||||||
/** Version string. */
|
|
||||||
val version: String = "1.0"
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Client API entry.
|
|
||||||
* @param name user name
|
|
||||||
* @return greeting string
|
|
||||||
*/
|
|
||||||
class Client {
|
|
||||||
/** Returns a greeting. */
|
|
||||||
fun greet(name: String): String = "hi " + name
|
|
||||||
}
|
|
||||||
@ -1,50 +0,0 @@
|
|||||||
class Tag(name: String) {
|
|
||||||
val name = name
|
|
||||||
var inner = ""
|
|
||||||
|
|
||||||
fun child(tagName: String, block: Tag.()->void) {
|
|
||||||
val child = Tag(tagName)
|
|
||||||
with(child) { block(this) }
|
|
||||||
inner += child.render()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun head(block: Tag.()->void) { child("head", block) }
|
|
||||||
fun body(block: Tag.()->void) { child("body", block) }
|
|
||||||
fun title(block: Tag.()->void) { child("title", block) }
|
|
||||||
fun h1(block: Tag.()->void) { child("h1", block) }
|
|
||||||
|
|
||||||
fun addText(text: String) {
|
|
||||||
inner += text
|
|
||||||
}
|
|
||||||
|
|
||||||
fun render() {
|
|
||||||
"<" + name + ">" + inner + "</" + name + ">"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
context(Tag)
|
|
||||||
fun String.unaryPlus() {
|
|
||||||
this@Tag.addText(this)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun html(block: Tag.()->void) {
|
|
||||||
val root = Tag("html")
|
|
||||||
with(root) { block(this) }
|
|
||||||
root.render()
|
|
||||||
}
|
|
||||||
|
|
||||||
val page = html {
|
|
||||||
head {
|
|
||||||
title {
|
|
||||||
+"Demo"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
body {
|
|
||||||
h1 {
|
|
||||||
+"Heading 1"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
println(page)
|
|
||||||
assertEquals("<html><head><title>Demo</title></head><body><h1>Heading 1</h1></body></html>", page)
|
|
||||||
@ -1,26 +1,23 @@
|
|||||||
// Sample: Operator Overloading in Lyng
|
// Sample: Operator Overloading in Lyng
|
||||||
|
|
||||||
class Vector<T>(val x: T, val y: T) {
|
class Vector(val x, val y) {
|
||||||
// Overload unary +
|
|
||||||
fun unaryPlus() = this
|
|
||||||
|
|
||||||
// Overload +
|
// Overload +
|
||||||
fun plus(other: Vector<U>) = Vector(x + other.x, y + other.y)
|
fun plus(other) = Vector(x + other.x, y + other.y)
|
||||||
|
|
||||||
// Overload -
|
// Overload -
|
||||||
fun minus(other: Vector<U>) = Vector(x - other.x, y - other.y)
|
fun minus(other) = Vector(x - other.x, y - other.y)
|
||||||
|
|
||||||
// Overload unary -
|
// Overload unary -
|
||||||
fun negate() = Vector(-x, -y)
|
fun negate() = Vector(-x, -y)
|
||||||
|
|
||||||
// Overload ==
|
// Overload ==
|
||||||
fun equals(other) {
|
fun equals(other) {
|
||||||
if (other is Vector<U>) x == other.x && y == other.y
|
if (other is Vector) x == other.x && y == other.y
|
||||||
else false
|
else false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Overload * (scalar multiplication)
|
// Overload * (scalar multiplication)
|
||||||
fun mul(scalar: Int | Real) = Vector(x * scalar, y * scalar)
|
fun mul(scalar) = Vector(x * scalar, y * scalar)
|
||||||
|
|
||||||
override fun toString() = "Vector(${x}, ${y})"
|
override fun toString() = "Vector(${x}, ${y})"
|
||||||
}
|
}
|
||||||
@ -31,11 +28,6 @@ val v2 = Vector(5, 5)
|
|||||||
println("v1: " + v1)
|
println("v1: " + v1)
|
||||||
println("v2: " + v2)
|
println("v2: " + v2)
|
||||||
|
|
||||||
// Test unary +
|
|
||||||
val v0 = +v1
|
|
||||||
println("+v1 = " + v0)
|
|
||||||
assertEquals(Vector(10, 20), v0)
|
|
||||||
|
|
||||||
// Test binary +
|
// Test binary +
|
||||||
val v3 = v1 + v2
|
val v3 = v1 + v2
|
||||||
println("v1 + v2 = " + v3)
|
println("v1 + v2 = " + v3)
|
||||||
|
|||||||
@ -13,6 +13,7 @@ fun findSumLimit(f) {
|
|||||||
println("limit reached after "+n+" rounds")
|
println("limit reached after "+n+" rounds")
|
||||||
break sum
|
break sum
|
||||||
}
|
}
|
||||||
|
n++
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
println("limit not reached")
|
println("limit not reached")
|
||||||
|
|||||||
@ -1,10 +0,0 @@
|
|||||||
fun describe<T>(x: T): String = when (T) {
|
|
||||||
nullable -> "nullable"
|
|
||||||
else -> "non-null"
|
|
||||||
}
|
|
||||||
|
|
||||||
type MaybeInt = Int?
|
|
||||||
assert(MaybeInt is nullable)
|
|
||||||
assert(!(Int is nullable))
|
|
||||||
assertEquals("nullable", describe<Int?>(null))
|
|
||||||
assertEquals("non-null", describe<Int>(1))
|
|
||||||
@ -1,19 +1,94 @@
|
|||||||
# Scopes and Closures: compile-time resolution
|
# Scopes and Closures: resolution and safety
|
||||||
|
|
||||||
Attention to AI: name lookup in runtime `Scope` is legacy. The bytecode compiler uses **compile-time name/member resolution only**.
|
Attention to AI: name lookup is ibsolete and must not be used with bytecode compiler
|
||||||
|
|
||||||
This page documents the **current** rules: static name resolution, closure captures, and the limited role of runtime `Scope` in Kotlin interop and explicit dynamic helpers.
|
This page documents how name resolution works with `ClosureScope`, how to avoid recursion pitfalls, and how to safely capture and execute callbacks that need access to outer locals.
|
||||||
|
|
||||||
## Current rules (bytecode compiler)
|
## Why this matters
|
||||||
- **All names resolve at compile time**: locals, parameters, captures, members, imports, and module globals must be known when compiling. Missing symbols are compile-time errors.
|
Name lookup across nested scopes and closures can accidentally form recursive resolution paths or hide expected symbols (outer locals, module/global functions). The rules below ensure predictable resolution and prevent infinite recursion.
|
||||||
- **Exception: `compile if` can skip dead branches**: inside an untaken `compile if (...)` branch, names are not resolved or type-checked at all. This is the supported way to guard optional classes or packages such as `defined(Udp)` or `defined(lyng.io.net)`.
|
|
||||||
- **No runtime fallbacks**: there is no dynamic name lookup, no fallback opcodes, and no “search parent scopes” at runtime for missing names.
|
|
||||||
- **Object members on unknown types only**: `toString`, `toInspectString`, `let`, `also`, `apply`, `run` are allowed on unknown types; all other members require a statically known receiver type or an explicit cast.
|
|
||||||
- **Closures capture slots**: lambdas and nested functions capture **frame slots** directly. Captures are resolved at compile time and compiled to slot references.
|
|
||||||
- **Scope is a reflection facade**: `Scope` is used only for Kotlin interop or explicit dynamic helpers. It must **not** be used for general symbol resolution in compiled Lyng code.
|
|
||||||
|
|
||||||
## Explicit dynamic access (opt-in only)
|
## Resolution order in ClosureScope
|
||||||
Dynamic name access is available only via explicit helpers (e.g., `dynamic { get { name -> ... } }`). It is **not** a fallback for normal member or variable access.
|
When evaluating an identifier `name` inside a closure, `ClosureScope.get(name)` resolves in this order:
|
||||||
|
|
||||||
## Legacy interpreter behavior (reference only)
|
1. **Current frame locals and arguments**: Variables defined within the current closure execution.
|
||||||
The old runtime `Scope`-based resolution order (locals → captured → `this` → caller → globals) is obsolete for bytecode compilation. Keep it only for legacy interpreter paths and tooling that explicitly opts into it.
|
2. **Captured lexical ancestry**: Outer local variables captured at the site where the closure was defined (the "lexical environment").
|
||||||
|
3. **Captured receiver members**: If the closure was defined within a class or explicitly bound to an object, it checks members of that object (`this`). This includes both instance fields/methods and class-level static members, following the MRO (C3) and respecting visibility rules (private members are only visible if the closure was defined in their class).
|
||||||
|
4. **Caller environment**: If not found lexically, it falls back to the calling context (e.g., the DSL's `this` or the caller's local variables).
|
||||||
|
5. **Global/Module fallbacks**: Final check for module-level constants and global functions.
|
||||||
|
|
||||||
|
This ensures that closures primarily interact with their defining environment (lexical capture) while still being able to participate in DSL-style calling contexts.
|
||||||
|
|
||||||
|
## Use raw‑chain helpers for ancestry walks
|
||||||
|
When authoring new scope types or advanced lookups, avoid calling virtual `get` while walking parents. Instead, use the non‑dispatch helpers on `Scope`:
|
||||||
|
|
||||||
|
- `chainLookupIgnoreClosure(name)`
|
||||||
|
- Walk raw `parent` chain and check only per‑frame locals/bindings/slots.
|
||||||
|
- Ignores overridden `get` (e.g., in `ClosureScope`). Cycle‑safe.
|
||||||
|
- `chainLookupWithMembers(name)`
|
||||||
|
- Like above, but after locals/bindings it also checks each frame’s `thisObj` members.
|
||||||
|
- Ignores overridden `get`. Cycle‑safe.
|
||||||
|
- `baseGetIgnoreClosure(name)`
|
||||||
|
- For the current frame only: check locals/bindings, then walk raw parents (locals/bindings), then fallback to this frame’s `thisObj` members.
|
||||||
|
|
||||||
|
These helpers avoid ping‑pong recursion and make structural cycles harmless (lookups terminate).
|
||||||
|
|
||||||
|
## Preventing structural cycles
|
||||||
|
- Don’t construct parent chains that can point back to a descendant.
|
||||||
|
- A debug‑time guard throws if assigning a parent would create a cycle; keep it enabled for development builds.
|
||||||
|
- Even with a cycle, chain helpers break out via a small `visited` set keyed by `frameId`.
|
||||||
|
|
||||||
|
## Capturing lexical environments for callbacks
|
||||||
|
For dynamic objects or custom builders, capture the creator’s lexical scope so callbacks can see outer locals/parameters:
|
||||||
|
|
||||||
|
1. Use `snapshotForClosure()` on the caller scope to capture locals/bindings/slots and parent.
|
||||||
|
2. Store this snapshot and run callbacks under `ClosureScope(callScope, captured)`.
|
||||||
|
|
||||||
|
Kotlin sketch:
|
||||||
|
```kotlin
|
||||||
|
val captured = scope.snapshotForClosure()
|
||||||
|
val execScope = ClosureScope(currentCallScope, captured)
|
||||||
|
callback.execute(execScope)
|
||||||
|
```
|
||||||
|
|
||||||
|
This ensures expressions like `contractName` used inside dynamic `get { name -> ... }` resolve to outer variables defined at the creation site.
|
||||||
|
|
||||||
|
## Closures in coroutines (launch/flow)
|
||||||
|
- The closure frame still prioritizes its own locals/args.
|
||||||
|
- Outer locals declared before suspension points remain visible through slot‑aware ancestry lookups.
|
||||||
|
- Global functions like `delay(ms)` and `yield()` are resolved via module/root fallbacks from within closures.
|
||||||
|
|
||||||
|
Tip: If a closure unexpectedly cannot see an outer local, check whether an intermediate runtime helper introduced an extra call frame; the built‑in lookup already traverses caller ancestry, so prefer the standard helpers rather than custom dispatch.
|
||||||
|
|
||||||
|
## Local variable references and missing symbols
|
||||||
|
- Unqualified identifier resolution first prefers locals/bindings/slots before falling back to `this` members.
|
||||||
|
- If neither locals nor members contain the symbol, missing field lookups map to `SymbolNotFound` (compatibility alias for `SymbolNotDefinedException`).
|
||||||
|
|
||||||
|
## Performance notes
|
||||||
|
- The `visited` sets used for cycle detection are tiny and short‑lived; in typical scripts the overhead is negligible.
|
||||||
|
- If profiling shows hotspots, consider limiting ancestry depth in your custom helpers or using small fixed arrays instead of hash sets—only for extremely hot code paths.
|
||||||
|
|
||||||
|
## Practical Example: `cached`
|
||||||
|
|
||||||
|
The `cached` function (defined in `lyng.stdlib`) is a classic example of using closures to maintain state. It wraps a builder into a zero-argument function that computes once and remembers the result:
|
||||||
|
|
||||||
|
```lyng
|
||||||
|
fun cached(builder) {
|
||||||
|
var calculated = false
|
||||||
|
var value = null
|
||||||
|
{ // This lambda captures `calculated`, `value`, and `builder`
|
||||||
|
if( !calculated ) {
|
||||||
|
value = builder()
|
||||||
|
calculated = true
|
||||||
|
}
|
||||||
|
value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Because Lyng now correctly isolates closures for each evaluation of a lambda literal, using `cached` inside a class instance works as expected: each instance maintains its own private `calculated` and `value` state, even if they share the same property declaration.
|
||||||
|
|
||||||
|
## Dos and Don’ts
|
||||||
|
- Do use `chainLookupIgnoreClosure` / `chainLookupWithMembers` for ancestry traversals.
|
||||||
|
- Do maintain the resolution order above for predictable behavior.
|
||||||
|
- Don’t call virtual `get` while walking parents; it risks recursion across scope types.
|
||||||
|
- Don’t attach instance scopes to transient/pool frames; bind to a stable parent scope instead.
|
||||||
|
|||||||
@ -1,40 +1,6 @@
|
|||||||
# Lyng serialization
|
# Lyng serialization
|
||||||
|
|
||||||
Lyng has a built-in serialization module, `lyng.serialization`.
|
Lyng has builting binary bit-effective serialization format, called Lynon for LYng Object Notation. It is typed, binary, implements caching, automatic compression, variable-length ints, one-bit Booleans an many nice features.
|
||||||
|
|
||||||
There are now two built-in formats with different goals:
|
|
||||||
|
|
||||||
- `Lynon`: the canonical binary format for Lyng values.
|
|
||||||
- `Json`: the canonical JSON-based round-trip format for Lyng values.
|
|
||||||
|
|
||||||
In addition, `Obj.toJson()` / `toJsonString()` remain available as a plain JSON projection for interoperability with
|
|
||||||
regular JSON tools and Kotlin `kotlinx.serialization`.
|
|
||||||
|
|
||||||
## Canonical formats
|
|
||||||
|
|
||||||
`Lynon` and `Json` are both exposed as format objects with the same surface:
|
|
||||||
|
|
||||||
- `Format.encode(value)`
|
|
||||||
- `Format.decode(encodedValue)`
|
|
||||||
|
|
||||||
For the built-in formats:
|
|
||||||
|
|
||||||
- `Lynon.encode(x)` returns `BitBuffer`
|
|
||||||
- `Lynon.decode(bitBuffer)` returns the original Lyng value
|
|
||||||
- `Json.encode(x)` returns `String`
|
|
||||||
- `Json.decode(jsonString)` returns the original Lyng value
|
|
||||||
|
|
||||||
`Json` also provides a typed canonical mode:
|
|
||||||
|
|
||||||
- `Json.encodeAs(Type, value)` returns `String`
|
|
||||||
- `Json.decodeAs(Type, jsonString)` returns the original Lyng value of the specified type
|
|
||||||
|
|
||||||
This is still canonical JSON, but it is schema-driven instead of fully self-describing.
|
|
||||||
|
|
||||||
## Lynon
|
|
||||||
|
|
||||||
Lynon is LYng Object Notation. It is typed, binary, bit-effective, implements caching, automatic compression,
|
|
||||||
variable-length integers, one-bit booleans, and preserves Lyng runtime structure well.
|
|
||||||
|
|
||||||
It is as simple as:
|
It is as simple as:
|
||||||
|
|
||||||
@ -51,11 +17,10 @@ It is as simple as:
|
|||||||
assertEquals( text, Lynon.decode(encodedBits) )
|
assertEquals( text, Lynon.decode(encodedBits) )
|
||||||
|
|
||||||
// compression was used automatically
|
// compression was used automatically
|
||||||
assert( text.length > (encodedBits.toBuffer() as Buffer).size )
|
assert( text.length > encodedBits.toBuffer().size )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Any class you create is serializable by default; Lynon serializes first constructor fields, then any `var` member
|
Any class you create is serializable by default; lynon serializes first constructor fields, then any `var` member fields.
|
||||||
fields.
|
|
||||||
|
|
||||||
## Transient Fields
|
## Transient Fields
|
||||||
|
|
||||||
@ -75,7 +40,7 @@ class MyData(@Transient val tempSecret, val publicData) {
|
|||||||
|
|
||||||
Transient fields:
|
Transient fields:
|
||||||
- Are **omitted** from Lynon binary streams.
|
- Are **omitted** from Lynon binary streams.
|
||||||
- Are **omitted** from JSON output (`toJson`) and canonical `Json.encode(...)`.
|
- Are **omitted** from JSON output (via `toJson`).
|
||||||
- Are **ignored** during structural equality checks (`==`).
|
- Are **ignored** during structural equality checks (`==`).
|
||||||
- If a transient constructor parameter has a **default value**, it will be restored to that default value during deserialization. Otherwise, it will be `null`.
|
- If a transient constructor parameter has a **default value**, it will be restored to that default value during deserialization. Otherwise, it will be `null`.
|
||||||
- Class body fields marked as `@Transient` will keep their initial values (or values assigned in `init`) after deserialization.
|
- Class body fields marked as `@Transient` will keep their initial values (or values assigned in `init`) after deserialization.
|
||||||
@ -84,131 +49,8 @@ Transient fields:
|
|||||||
|
|
||||||
- **Singleton Objects**: `object` declarations are serializable by name. Their state (mutable fields) is also serialized and restored, respecting `@Transient`.
|
- **Singleton Objects**: `object` declarations are serializable by name. Their state (mutable fields) is also serialized and restored, respecting `@Transient`.
|
||||||
- **Classes**: Class objects themselves can be serialized. They are serialized by their full qualified name. When converted to JSON, a class object includes its public static fields (excluding those marked `@Transient`).
|
- **Classes**: Class objects themselves can be serialized. They are serialized by their full qualified name. When converted to JSON, a class object includes its public static fields (excluding those marked `@Transient`).
|
||||||
- **Exceptions**: canonical formats preserve exception class, message, extra data, and captured stack trace.
|
|
||||||
|
|
||||||
## Plain JSON projection vs canonical Json format
|
## Custom Serialization
|
||||||
|
|
||||||
There are two JSON-related APIs and they serve different purposes:
|
|
||||||
|
|
||||||
- `Obj.toJson()` / `toJsonString()`
|
|
||||||
- produce ordinary JSON values
|
|
||||||
- best for interop with external JSON systems
|
|
||||||
- best for `Obj.decodeSerializable()` / `decodeSerializableWith()`
|
|
||||||
- may be lossy for Lyng-specific structures
|
|
||||||
|
|
||||||
- `Json.encode()` / `Json.decode()`
|
|
||||||
- produce JSON text too
|
|
||||||
- use Lyng-specific type tags so the payload is self-describing
|
|
||||||
- intended for round-tripping Lyng values
|
|
||||||
- intended to match Lynon semantics where JSON can carry them
|
|
||||||
- still keep ordinary string-key maps in traditional JSON object form
|
|
||||||
- can preserve values that plain JSON cannot represent directly, such as:
|
|
||||||
- maps with non-string keys
|
|
||||||
- sets
|
|
||||||
- buffers and bit buffers
|
|
||||||
- class instances
|
|
||||||
- singleton objects
|
|
||||||
- enums
|
|
||||||
- exceptions
|
|
||||||
- date/time objects
|
|
||||||
- non-finite reals
|
|
||||||
- `void`
|
|
||||||
|
|
||||||
- `Json.encodeAs(Type, value)` / `Json.decodeAs(Type, text)`
|
|
||||||
- also round-trip Lyng values through JSON text
|
|
||||||
- use the declared or requested type as decoding schema
|
|
||||||
- recursively omit type tags when the declared type is already exact enough
|
|
||||||
- keep canonical tags when the runtime value is more specific than the declared type
|
|
||||||
- produce less noisy JSON for closed and otherwise precisely-typed object graphs
|
|
||||||
- still keep ordinary `Map<String, ...>` values in traditional JSON object form
|
|
||||||
|
|
||||||
Why this split exists:
|
|
||||||
|
|
||||||
- plain `toJson()` must remain ordinary JSON so it stays convenient for external JSON systems and Kotlin
|
|
||||||
`kotlinx.serialization`
|
|
||||||
- canonical `Json.encode()` is for Lyng-to-Lyng transport through JSON text without any external schema, so it must
|
|
||||||
remain self-describing and preserve Lyng runtime
|
|
||||||
distinctions whenever possible
|
|
||||||
- `Json.encodeAs()` exists for the cases where a schema is known on both sides and we want canonical round-trip
|
|
||||||
behavior with fewer tags
|
|
||||||
- one API cannot optimize for both goals at once: either you get too many Lyng tags for ordinary JSON interop, or you
|
|
||||||
get lossy round-trips
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
import lyng.serialization
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
enum Color { Red, Green }
|
|
||||||
class Point(x,y) { var z = 42 }
|
|
||||||
|
|
||||||
val p = Point(1,2)
|
|
||||||
p.z = 99
|
|
||||||
val x = List(
|
|
||||||
p,
|
|
||||||
Map([1, "one"], ["two", 2]),
|
|
||||||
Set(1,2,3),
|
|
||||||
"hello".encodeUtf8(),
|
|
||||||
Date(2026,4,15),
|
|
||||||
Color.Green
|
|
||||||
)
|
|
||||||
|
|
||||||
assertEquals(x, Json.decode(Json.encode(x)))
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Typed canonical example:
|
|
||||||
|
|
||||||
import lyng.serialization
|
|
||||||
|
|
||||||
closed class Point(x: Int, y: Int)
|
|
||||||
closed class Segment(a: Point, b: Point)
|
|
||||||
|
|
||||||
val value = Segment(Point(0,1), Point(2,3))
|
|
||||||
val json = Json.encodeAs(Segment, value)
|
|
||||||
|
|
||||||
assertEquals("{\"a\":{\"x\":0,\"y\":1},\"b\":{\"x\":2,\"y\":3}}", json)
|
|
||||||
assertEquals(value, Json.decodeAs(Segment, json))
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
## Adding more formats from Kotlin modules
|
|
||||||
|
|
||||||
External modules can add new formats on the Kotlin side.
|
|
||||||
|
|
||||||
The common base class is:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
abstract class ObjSerializationFormatClass(className: String) : ObjClass(className) {
|
|
||||||
abstract suspend fun encodeValue(scope: Scope, value: Obj): Obj
|
|
||||||
abstract suspend fun decodeValue(scope: Scope, encoded: Obj): Obj
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
To export a new format from a module:
|
|
||||||
|
|
||||||
```kotlin
|
|
||||||
im.addPackage("test.formats") { module ->
|
|
||||||
module.bindSerializationFormat(
|
|
||||||
object : ObjSerializationFormatClass("Reverse") {
|
|
||||||
override suspend fun encodeValue(scope: Scope, value: Obj): Obj =
|
|
||||||
ObjString(value.toString(scope).value.reversed())
|
|
||||||
|
|
||||||
override suspend fun decodeValue(scope: Scope, encoded: Obj): Obj =
|
|
||||||
ObjString((encoded as ObjString).value.reversed())
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Then from Lyng, after importing the Kotlin module above, usage looks like:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import test.formats
|
|
||||||
|
|
||||||
assertEquals("cba", Reverse.encode("abc"))
|
|
||||||
assertEquals("abc", Reverse.decode("cba"))
|
|
||||||
```
|
|
||||||
|
|
||||||
## Notes
|
|
||||||
|
|
||||||
Important is to understand that normally `Lynon.decode` wants [BitBuffer], as `Lynon.encode` produces. If you have the regular [Buffer], be sure to convert it:
|
Important is to understand that normally `Lynon.decode` wants [BitBuffer], as `Lynon.encode` produces. If you have the regular [Buffer], be sure to convert it:
|
||||||
|
|
||||||
@ -217,3 +59,5 @@ Important is to understand that normally `Lynon.decode` wants [BitBuffer], as `L
|
|||||||
this possibly creates extra zero bits at the end, as bit content could be shorter than byte-grained but for the Lynon format it does not make sense. Note that when you serialize [BitBuffer], exact number of bits is written. To convert bit buffer to bytes:
|
this possibly creates extra zero bits at the end, as bit content could be shorter than byte-grained but for the Lynon format it does not make sense. Note that when you serialize [BitBuffer], exact number of bits is written. To convert bit buffer to bytes:
|
||||||
|
|
||||||
Lynon.encode("hello").toBuffer()
|
Lynon.encode("hello").toBuffer()
|
||||||
|
|
||||||
|
(topic is incomplete and under construction)
|
||||||
|
|||||||
185
docs/time.md
185
docs/time.md
@ -1,135 +1,74 @@
|
|||||||
# Lyng time functions
|
# Lyng time functions
|
||||||
|
|
||||||
Lyng date and time support requires importing `lyng.time`. The module provides four related types:
|
Lyng date and time support requires importing `lyng.time` packages. Lyng uses simple yet modern time object models:
|
||||||
|
|
||||||
- `Instant` for absolute timestamps.
|
- `Instant` class for absolute time stamps with platform-dependent resolution.
|
||||||
- `Date` for calendar dates without time-of-day or timezone.
|
- `DateTime` class for calendar-aware points in time within a specific time zone.
|
||||||
- `DateTime` for calendar-aware points in time in a specific timezone.
|
- `Duration` to represent amount of time not depending on the calendar (e.g., milliseconds, seconds).
|
||||||
- `Duration` for absolute elapsed time.
|
|
||||||
|
|
||||||
## Time instant: `Instant`
|
## Time instant: `Instant`
|
||||||
|
|
||||||
`Instant` represents some moment of time independently of the calendar. It is similar to SQL `TIMESTAMP`
|
Represent some moment of time not depending on the calendar. It is similar to `TIMESTAMP` in SQL or `Instant` in Kotlin.
|
||||||
or Kotlin `Instant`.
|
|
||||||
|
|
||||||
### Constructing and converting
|
### Constructing and converting
|
||||||
|
|
||||||
import lyng.time
|
import lyng.time
|
||||||
|
|
||||||
|
// default constructor returns time now:
|
||||||
val t1 = Instant()
|
val t1 = Instant()
|
||||||
val t2 = Instant(1704110400)
|
|
||||||
|
// constructing from a number is treated as seconds since unix epoch:
|
||||||
|
val t2 = Instant(1704110400) // 2024-01-01T12:00:00Z
|
||||||
|
|
||||||
|
// from RFC3339 string:
|
||||||
val t3 = Instant("2024-01-01T12:00:00.123456Z")
|
val t3 = Instant("2024-01-01T12:00:00.123456Z")
|
||||||
|
|
||||||
val t4 = t3.truncateToMinute()
|
// truncation:
|
||||||
assertEquals("2024-01-01T12:00:00Z", t4.toRFC3339())
|
val t4 = t3.truncateToMinute
|
||||||
|
assertEquals(t4.toRFC3339(), "2024-01-01T12:00:00Z")
|
||||||
|
|
||||||
|
// to localized DateTime (uses system default TZ if not specified):
|
||||||
val dt = t3.toDateTime("+02:00")
|
val dt = t3.toDateTime("+02:00")
|
||||||
assertEquals(14, dt.hour)
|
assertEquals(dt.hour, 14)
|
||||||
|
|
||||||
val d = t3.toDate("Z")
|
|
||||||
assertEquals(Date(2024, 1, 1), d)
|
|
||||||
|
|
||||||
### Instant members
|
### Instant members
|
||||||
|
|
||||||
| member | description |
|
| member | description |
|
||||||
|--------------------------------|------------------------------------------------------|
|
|--------------------------------|---------------------------------------------------------|
|
||||||
| epochSeconds: Real | offset in seconds since Unix epoch |
|
| epochSeconds: Real | positive or negative offset in seconds since Unix epoch |
|
||||||
| epochWholeSeconds: Int | whole seconds since Unix epoch |
|
| epochWholeSeconds: Int | same, but in _whole seconds_. Slightly faster |
|
||||||
| nanosecondsOfSecond: Int | nanoseconds within the current second |
|
| nanosecondsOfSecond: Int | offset from epochWholeSeconds in nanos |
|
||||||
| isDistantFuture: Bool | true if it is `Instant.distantFuture` |
|
| isDistantFuture: Bool | true if it `Instant.distantFuture` |
|
||||||
| isDistantPast: Bool | true if it is `Instant.distantPast` |
|
| isDistantPast: Bool | true if it `Instant.distantPast` |
|
||||||
| truncateToMinute(): Instant | truncate to minute precision |
|
| truncateToMinute: Instant | create new instance truncated to minute |
|
||||||
| truncateToSecond(): Instant | truncate to second precision |
|
| truncateToSecond: Instant | create new instance truncated to second |
|
||||||
| truncateToMillisecond(): Instant | truncate to millisecond precision |
|
| truncateToMillisecond: Instant | truncate new instance to millisecond |
|
||||||
| truncateToMicrosecond(): Instant | truncate to microsecond precision |
|
| truncateToMicrosecond: Instant | truncate new instance to microsecond |
|
||||||
| toRFC3339(): String | format as RFC3339 string in UTC |
|
| toRFC3339(): String | format as RFC3339 string (UTC) |
|
||||||
| toDateTime(tz?): DateTime | localize to a timezone |
|
| toDateTime(tz?): DateTime | localize to a TimeZone (ID string or offset seconds) |
|
||||||
| toDate(tz?): Date | convert to a calendar date in a timezone |
|
|
||||||
|
|
||||||
## Calendar date: `Date`
|
|
||||||
|
|
||||||
`Date` represents a pure calendar date. It has no time-of-day and no attached timezone. Use it for values
|
|
||||||
like birthdays, due dates, invoice dates, and SQL `DATE` columns.
|
|
||||||
|
|
||||||
### Constructing
|
|
||||||
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
val today = Date()
|
|
||||||
val d1 = Date(2026, 4, 15)
|
|
||||||
val d2 = Date("2024-02-29")
|
|
||||||
val d3 = Date.parseIso("2024-02-29")
|
|
||||||
val d4 = Date(DateTime(2024, 5, 20, 15, 30, 45, "+02:00"))
|
|
||||||
val d5 = Date(Instant("2024-01-01T23:30:00Z"), "+02:00")
|
|
||||||
|
|
||||||
### Date members
|
|
||||||
|
|
||||||
| member | description |
|
|
||||||
|--------------------------------|------------------------------------------------------------|
|
|
||||||
| year: Int | year component |
|
|
||||||
| month: Int | month component (1..12) |
|
|
||||||
| day: Int | day of month (alias `dayOfMonth`) |
|
|
||||||
| dayOfWeek: Int | day of week (1=Monday, 7=Sunday) |
|
|
||||||
| dayOfYear: Int | day of year (1..365/366) |
|
|
||||||
| isLeapYear: Bool | whether this date is in a leap year |
|
|
||||||
| lengthOfMonth: Int | number of days in this month |
|
|
||||||
| lengthOfYear: Int | 365 or 366 |
|
|
||||||
| toIsoString(): String | ISO `YYYY-MM-DD` string |
|
|
||||||
| toSortableString(): String | alias to `toIsoString()` |
|
|
||||||
| toDateTime(tz="Z"): DateTime | start-of-day `DateTime` in the specified timezone |
|
|
||||||
| atStartOfDay(tz="Z"): DateTime | alias to `toDateTime()` |
|
|
||||||
| addDays(n): Date | add or subtract calendar days |
|
|
||||||
| addMonths(n): Date | add or subtract months, normalizing end-of-month |
|
|
||||||
| addYears(n): Date | add or subtract years |
|
|
||||||
| daysUntil(other): Int | calendar days until `other` |
|
|
||||||
| daysSince(other): Int | calendar days since `other` |
|
|
||||||
| static today(tz?): Date | today in the specified timezone |
|
|
||||||
| static parseIso(s): Date | parse ISO `YYYY-MM-DD` |
|
|
||||||
|
|
||||||
### Date arithmetic
|
|
||||||
|
|
||||||
`Date` supports only whole-day arithmetic. This is deliberate: calendar dates should not silently accept
|
|
||||||
sub-day durations.
|
|
||||||
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
val d1 = Date(2026, 4, 15)
|
|
||||||
val d2 = d1.addDays(10)
|
|
||||||
|
|
||||||
assertEquals(Date(2026, 4, 25), d2)
|
|
||||||
assertEquals(Date(2026, 4, 18), d1 + 3.days)
|
|
||||||
assertEquals(Date(2026, 4, 12), d1 - 3.days)
|
|
||||||
assertEquals(10, d1.daysUntil(d2))
|
|
||||||
assertEquals(10, d2.daysSince(d1))
|
|
||||||
assertEquals(10, d2 - d1)
|
|
||||||
|
|
||||||
### Date conversions
|
|
||||||
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
val i = Instant("2024-01-01T23:30:00Z")
|
|
||||||
assertEquals(Date(2024, 1, 1), i.toDate("Z"))
|
|
||||||
assertEquals(Date(2024, 1, 2), i.toDate("+02:00"))
|
|
||||||
|
|
||||||
val dt = DateTime(2024, 5, 20, 15, 30, 45, "+02:00")
|
|
||||||
assertEquals(Date(2024, 5, 20), dt.date)
|
|
||||||
assertEquals(Date(2024, 5, 20), dt.toDate())
|
|
||||||
assertEquals(DateTime(2024, 5, 20, 0, 0, 0, "Z"), Date(2024, 5, 20).toDateTime("Z"))
|
|
||||||
assertEquals(DateTime(2024, 5, 20, 0, 0, 0, "+02:00"), Date(2024, 5, 20).atStartOfDay("+02:00"))
|
|
||||||
|
|
||||||
## Calendar time: `DateTime`
|
## Calendar time: `DateTime`
|
||||||
|
|
||||||
`DateTime` represents a point in time in a specific timezone. It provides access to calendar components
|
`DateTime` represents a point in time in a specific timezone. It provides access to calendar components like year,
|
||||||
such as year, month, day, and hour.
|
month, and day.
|
||||||
|
|
||||||
### Constructing
|
### Constructing
|
||||||
|
|
||||||
import lyng.time
|
import lyng.time
|
||||||
|
|
||||||
|
// Current time in system default timezone
|
||||||
val now = DateTime.now()
|
val now = DateTime.now()
|
||||||
|
|
||||||
|
// Specific timezone
|
||||||
val offsetTime = DateTime.now("+02:00")
|
val offsetTime = DateTime.now("+02:00")
|
||||||
|
|
||||||
|
// From Instant
|
||||||
val dt = Instant().toDateTime("Z")
|
val dt = Instant().toDateTime("Z")
|
||||||
|
|
||||||
|
// By components (year, month, day, hour=0, minute=0, second=0, timeZone="UTC")
|
||||||
val dt2 = DateTime(2024, 1, 1, 12, 0, 0, "Z")
|
val dt2 = DateTime(2024, 1, 1, 12, 0, 0, "Z")
|
||||||
|
|
||||||
|
// From RFC3339 string
|
||||||
val dt3 = DateTime.parseRFC3339("2024-01-01T12:00:00+02:00")
|
val dt3 = DateTime.parseRFC3339("2024-01-01T12:00:00+02:00")
|
||||||
|
|
||||||
### DateTime members
|
### DateTime members
|
||||||
@ -144,9 +83,7 @@ such as year, month, day, and hour.
|
|||||||
| second: Int | second component (0..59) |
|
| second: Int | second component (0..59) |
|
||||||
| dayOfWeek: Int | day of week (1=Monday, 7=Sunday) |
|
| dayOfWeek: Int | day of week (1=Monday, 7=Sunday) |
|
||||||
| timeZone: String | timezone ID string |
|
| timeZone: String | timezone ID string |
|
||||||
| date: Date | calendar date component |
|
|
||||||
| toInstant(): Instant | convert back to absolute Instant |
|
| toInstant(): Instant | convert back to absolute Instant |
|
||||||
| toDate(): Date | extract the calendar date in this timezone |
|
|
||||||
| toUTC(): DateTime | shortcut to convert to UTC |
|
| toUTC(): DateTime | shortcut to convert to UTC |
|
||||||
| toTimeZone(tz): DateTime | convert to another timezone |
|
| toTimeZone(tz): DateTime | convert to another timezone |
|
||||||
| addMonths(n): DateTime | add/subtract months (normalizes end of month) |
|
| addMonths(n): DateTime | add/subtract months (normalizes end of month) |
|
||||||
@ -159,27 +96,28 @@ such as year, month, day, and hour.
|
|||||||
|
|
||||||
`DateTime` handles calendar arithmetic correctly:
|
`DateTime` handles calendar arithmetic correctly:
|
||||||
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
val leapDay = Instant("2024-02-29T12:00:00Z").toDateTime("Z")
|
val leapDay = Instant("2024-02-29T12:00:00Z").toDateTime("Z")
|
||||||
val nextYear = leapDay.addYears(1)
|
val nextYear = leapDay.addYears(1)
|
||||||
assertEquals(28, nextYear.day)
|
assertEquals(nextYear.day, 28) // Feb 29, 2024 -> Feb 28, 2025
|
||||||
|
|
||||||
# `Duration` class
|
# `Duration` class
|
||||||
|
|
||||||
`Duration` represents absolute elapsed time between two instants.
|
Represent absolute time distance between two `Instant`.
|
||||||
|
|
||||||
import lyng.time
|
import lyng.time
|
||||||
|
|
||||||
val t1 = Instant()
|
val t1 = Instant()
|
||||||
delay(1.millisecond)
|
|
||||||
val t2 = Instant()
|
|
||||||
|
|
||||||
assert(t2 - t1 >= 1.millisecond)
|
// yes we can delay to period, and it is not blocking. is suspends!
|
||||||
assert(t2 - t1 < 100.millisecond)
|
delay(1.millisecond)
|
||||||
|
|
||||||
|
val t2 = Instant()
|
||||||
|
// be suspend, so actual time may vary:
|
||||||
|
assert( t2 - t1 >= 1.millisecond)
|
||||||
|
assert( t2 - t1 < 100.millisecond)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Duration values can be created from numbers using extensions on `Int` and `Real`:
|
Duration can be converted from numbers, like `5.minutes` and so on. Extensions are created for
|
||||||
|
`Int` and `Real`, so for n as Real or Int it is possible to create durations::
|
||||||
|
|
||||||
- `n.millisecond`, `n.milliseconds`
|
- `n.millisecond`, `n.milliseconds`
|
||||||
- `n.second`, `n.seconds`
|
- `n.second`, `n.seconds`
|
||||||
@ -187,9 +125,10 @@ Duration values can be created from numbers using extensions on `Int` and `Real`
|
|||||||
- `n.hour`, `n.hours`
|
- `n.hour`, `n.hours`
|
||||||
- `n.day`, `n.days`
|
- `n.day`, `n.days`
|
||||||
|
|
||||||
Larger units like months or years are calendar-dependent and are intentionally not part of `Duration`.
|
The bigger time units like months or years are calendar-dependent and can't be used with `Duration`.
|
||||||
|
|
||||||
Each duration instance can be converted to numbers in these units:
|
Each duration instance can be converted to number of any of these time units, as `Real` number, if `d` is a `Duration`
|
||||||
|
instance:
|
||||||
|
|
||||||
- `d.microseconds`
|
- `d.microseconds`
|
||||||
- `d.milliseconds`
|
- `d.milliseconds`
|
||||||
@ -198,16 +137,18 @@ Each duration instance can be converted to numbers in these units:
|
|||||||
- `d.hours`
|
- `d.hours`
|
||||||
- `d.days`
|
- `d.days`
|
||||||
|
|
||||||
Example:
|
for example
|
||||||
|
|
||||||
import lyng.time
|
import lyng.time
|
||||||
|
assertEquals( 60, 1.minute.seconds )
|
||||||
|
assertEquals( 10.milliseconds, 0.01.seconds )
|
||||||
|
|
||||||
assertEquals(60, 1.minute.seconds)
|
|
||||||
assertEquals(10.milliseconds, 0.01.seconds)
|
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
# Utility functions
|
# Utility functions
|
||||||
|
|
||||||
## `delay(duration: Duration)`
|
## delay(duration: Duration)
|
||||||
|
|
||||||
|
Suspends current coroutine for at least the specified duration.
|
||||||
|
|
||||||
|
|
||||||
Suspends the current coroutine for at least the specified duration.
|
|
||||||
|
|||||||
354
docs/tutorial.md
354
docs/tutorial.md
@ -14,7 +14,7 @@ __Other documents to read__ maybe after this one:
|
|||||||
- [time](time.md) and [parallelism](parallelism.md)
|
- [time](time.md) and [parallelism](parallelism.md)
|
||||||
- [parallelism] - multithreaded code, coroutines, etc.
|
- [parallelism] - multithreaded code, coroutines, etc.
|
||||||
- Some class
|
- Some class
|
||||||
references: [List], [ImmutableList], [Set], [ImmutableSet], [Map], [ImmutableMap], [Real], [Range], [Iterable], [Iterator], [time manipulation](time.md), [Array], [RingBuffer], [Buffer].
|
references: [List], [Set], [Map], [Real], [Range], [Iterable], [Iterator], [time manipulation](time.md), [Array], [RingBuffer], [Buffer].
|
||||||
- Some samples: [combinatorics](samples/combinatorics.lyng.md), national vars and
|
- Some samples: [combinatorics](samples/combinatorics.lyng.md), national vars and
|
||||||
loops: [сумма ряда](samples/сумма_ряда.lyng.md). More at [samples folder](samples)
|
loops: [сумма ряда](samples/сумма_ряда.lyng.md). More at [samples folder](samples)
|
||||||
|
|
||||||
@ -229,8 +229,9 @@ Naturally, assignment returns its value:
|
|||||||
rvalue means you cant assign the result if the assignment
|
rvalue means you cant assign the result if the assignment
|
||||||
|
|
||||||
var x
|
var x
|
||||||
// compile-time error: can't assign to rvalue
|
assertThrows { (x = 11) = 5 }
|
||||||
(x = 11) = 5
|
void
|
||||||
|
>>> void
|
||||||
|
|
||||||
This also prevents chain assignments so use parentheses:
|
This also prevents chain assignments so use parentheses:
|
||||||
|
|
||||||
@ -248,24 +249,18 @@ When the value is `null`, it might throws `NullReferenceException`, the name is
|
|||||||
one can check it against null or use _null coalescing_. The null coalescing means, if the operand (left) is null,
|
one can check it against null or use _null coalescing_. The null coalescing means, if the operand (left) is null,
|
||||||
the operation won't be performed and the result will be null. Here is the difference:
|
the operation won't be performed and the result will be null. Here is the difference:
|
||||||
|
|
||||||
class Sample {
|
val ref = null
|
||||||
var field = 1
|
assertThrows { ref.field }
|
||||||
fun method() { 2 }
|
assertThrows { ref.method() }
|
||||||
var list = [1, 2, 3]
|
assertThrows { ref.array[1] }
|
||||||
}
|
assertThrows { ref[1] }
|
||||||
|
assertThrows { ref() }
|
||||||
val ref: Sample? = null
|
|
||||||
val list: List<Int>? = null
|
|
||||||
// direct access throws NullReferenceException:
|
|
||||||
// ref.field
|
|
||||||
// ref.method()
|
|
||||||
// ref.list[1]
|
|
||||||
// list[1]
|
|
||||||
|
|
||||||
assert( ref?.field == null )
|
assert( ref?.field == null )
|
||||||
assert( ref?.method() == null )
|
assert( ref?.method() == null )
|
||||||
assert( ref?.list?[1] == null )
|
assert( ref?.array?[1] == null )
|
||||||
assert( list?[1] == null )
|
assert( ref?[1] == null )
|
||||||
|
assert( ref?() == null )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Note: `?.` is still a typed operation. The receiver must have a compile-time type that declares the member; if the
|
Note: `?.` is still a typed operation. The receiver must have a compile-time type that declares the member; if the
|
||||||
@ -327,8 +322,8 @@ Much like let, but it does not alter returned value:
|
|||||||
|
|
||||||
While it is not altering return value, the source object could be changed:
|
While it is not altering return value, the source object could be changed:
|
||||||
also
|
also
|
||||||
class Point(var x: Int, var y: Int)
|
class Point(x,y)
|
||||||
val p: Point = Point(1,2).also { it.x++ }
|
val p = Point(1,2).also { it.x++ }
|
||||||
assertEquals(p.x, 2)
|
assertEquals(p.x, 2)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
@ -336,9 +331,9 @@ also
|
|||||||
|
|
||||||
It works much like `also`, but is executed in the context of the source object:
|
It works much like `also`, but is executed in the context of the source object:
|
||||||
|
|
||||||
class Point(var x: Int, var y: Int)
|
class Point(x,y)
|
||||||
// see the difference: apply changes this to newly created Point:
|
// see the difference: apply changes this to newly created Point:
|
||||||
val p = Point(1,2).apply { this@Point.x++; this@Point.y++ }
|
val p = Point(1,2).apply { x++; y++ }
|
||||||
assertEquals(p, Point(2,3))
|
assertEquals(p, Point(2,3))
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
@ -346,46 +341,12 @@ It works much like `also`, but is executed in the context of the source object:
|
|||||||
|
|
||||||
Sets `this` to the first argument and executes the block. Returns the value returned by the block:
|
Sets `this` to the first argument and executes the block. Returns the value returned by the block:
|
||||||
|
|
||||||
class Point(var x: Int, var y: Int)
|
class Point(x,y)
|
||||||
val p = Point(1,2)
|
val p = Point(1,2)
|
||||||
val sum = with(p) { x + y }
|
val sum = with(p) { x + y }
|
||||||
assertEquals(3, sum)
|
assertEquals(3, sum)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Receiver lambdas can also keep outer receivers in scope. The primary receiver wins for unqualified lookup, and `this@Type`
|
|
||||||
selects an outer receiver explicitly:
|
|
||||||
|
|
||||||
class Html { fun lang() = "en" }
|
|
||||||
class Body { fun lang() = "body" }
|
|
||||||
|
|
||||||
fun html(block: Html.()->String) = with(Html()) { block(this) }
|
|
||||||
fun body(block: Body.()->String) = with(Body()) { block(this) }
|
|
||||||
|
|
||||||
val result = html {
|
|
||||||
body {
|
|
||||||
lang() + ":" + this@Html.lang()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assertEquals("body:en", result)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
You can declare the same requirement in a function type:
|
|
||||||
|
|
||||||
val block: context(Html) Body.()->String = {
|
|
||||||
lang() + ":" + this@Html.lang()
|
|
||||||
}
|
|
||||||
|
|
||||||
If the primary receiver does not define a member and multiple outer/context receivers do, Lyng reports an ambiguity instead of picking one silently:
|
|
||||||
|
|
||||||
class A { fun title() = "a" }
|
|
||||||
class B { fun title() = "b" }
|
|
||||||
class C
|
|
||||||
|
|
||||||
val block: context(A, B) C.()->String = {
|
|
||||||
// title() // compile-time ambiguity
|
|
||||||
this@A.title()
|
|
||||||
}
|
|
||||||
|
|
||||||
## run
|
## run
|
||||||
|
|
||||||
Executes a block after it returning the value passed by the block. for example, can be used with elvis operator:
|
Executes a block after it returning the value passed by the block. for example, can be used with elvis operator:
|
||||||
@ -409,18 +370,6 @@ It is rather simple, like everywhere else:
|
|||||||
|
|
||||||
See [math](math.md) for more on it. Notice using Greek as identifier, all languages are allowed.
|
See [math](math.md) for more on it. Notice using Greek as identifier, all languages are allowed.
|
||||||
|
|
||||||
For linear algebra, import `lyng.matrix`:
|
|
||||||
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[1, 2], [3, 4]])
|
|
||||||
val i: Matrix = Matrix.identity(2)
|
|
||||||
val sum: Matrix = a + i
|
|
||||||
assertEquals([[2.0, 2.0], [3.0, 5.0]], sum.toList())
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
See [Matrix](Matrix.md) for vectors, matrix multiplication, inversion, and slicing such as `m[0..2, 1]`.
|
|
||||||
|
|
||||||
Logical operation could be used the same
|
Logical operation could be used the same
|
||||||
|
|
||||||
var x = 10
|
var x = 10
|
||||||
@ -548,18 +497,6 @@ Aliases expand to their underlying type expressions. See `docs/generics.md` for
|
|||||||
|
|
||||||
`Null` is the class of `null`. It is a singleton type and mostly useful for type inference results.
|
`Null` is the class of `null`. It is a singleton type and mostly useful for type inference results.
|
||||||
|
|
||||||
For type expressions, you can check nullability directly:
|
|
||||||
|
|
||||||
T is nullable
|
|
||||||
T !is nullable
|
|
||||||
|
|
||||||
This is especially useful in generic code and in `when` over a type parameter:
|
|
||||||
|
|
||||||
fun describe<T>(x: T): String = when (T) {
|
|
||||||
nullable -> "nullable"
|
|
||||||
else -> "non-null"
|
|
||||||
}
|
|
||||||
|
|
||||||
## Type inference
|
## Type inference
|
||||||
|
|
||||||
The compiler infers types from:
|
The compiler infers types from:
|
||||||
@ -576,13 +513,6 @@ Examples:
|
|||||||
fun inc(x=0) = x + 1 // (Int)->Int
|
fun inc(x=0) = x + 1 // (Int)->Int
|
||||||
fun maybe(flag) { if(flag) 1 else null } // ()->Int?
|
fun maybe(flag) { if(flag) 1 else null } // ()->Int?
|
||||||
|
|
||||||
Function types are written as `(T1, T2, ...)->R`. You can include ellipsis in function *types* to
|
|
||||||
express a variadic position:
|
|
||||||
|
|
||||||
var fmt: (String, Object...)->String
|
|
||||||
var f: (Int, Object..., String)->Real
|
|
||||||
var anyArgs: (...)->Int // shorthand for (Object...)->Int
|
|
||||||
|
|
||||||
Untyped locals are allowed, but their type is fixed on the first assignment:
|
Untyped locals are allowed, but their type is fixed on the first assignment:
|
||||||
|
|
||||||
var x
|
var x
|
||||||
@ -705,9 +635,8 @@ There are default parameters in Lyng:
|
|||||||
It is possible to define also vararg using ellipsis:
|
It is possible to define also vararg using ellipsis:
|
||||||
|
|
||||||
fun sum(args...) {
|
fun sum(args...) {
|
||||||
val list = args as List
|
var result = args[0]
|
||||||
var result = list[0]
|
for( i in 1 ..< args.size ) result += args[i]
|
||||||
for( i in 1 ..< list.size ) result += list[i]
|
|
||||||
}
|
}
|
||||||
sum(10,20,30)
|
sum(10,20,30)
|
||||||
>>> 60
|
>>> 60
|
||||||
@ -800,11 +729,6 @@ one could be with ellipsis that means "the rest pf arguments as List":
|
|||||||
assert( { a, b...-> [a,...b] }(100, 1, 2, 3) == [100, 1, 2, 3])
|
assert( { a, b...-> [a,...b] }(100, 1, 2, 3) == [100, 1, 2, 3])
|
||||||
void
|
void
|
||||||
|
|
||||||
Type-annotated lambdas can use variadic *function types* as well:
|
|
||||||
|
|
||||||
val f: (Int, Object..., String)->Real = { a, rest..., b -> 0.0 }
|
|
||||||
val anyArgs: (...)->Int = { -> 0 }
|
|
||||||
|
|
||||||
### Using lambda as the parameter
|
### Using lambda as the parameter
|
||||||
|
|
||||||
See also: [Testing and Assertions](Testing.md)
|
See also: [Testing and Assertions](Testing.md)
|
||||||
@ -843,13 +767,6 @@ Lyng has built-in mutable array class `List` with simple literals:
|
|||||||
|
|
||||||
[List] is an implementation of the type `Array`, and through it `Collection` and [Iterable]. Please read [Iterable],
|
[List] is an implementation of the type `Array`, and through it `Collection` and [Iterable]. Please read [Iterable],
|
||||||
many collection based methods are implemented there.
|
many collection based methods are implemented there.
|
||||||
For immutable list values, use `list.toImmutable()` and [ImmutableList].
|
|
||||||
|
|
||||||
To construct a list programmatically, use the static helper `List.fill`:
|
|
||||||
|
|
||||||
val tens = List.fill(5) { index -> index * 10 }
|
|
||||||
assertEquals([0, 10, 20, 30, 40], tens)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Lists can contain any type of objects, lists too:
|
Lists can contain any type of objects, lists too:
|
||||||
|
|
||||||
@ -858,19 +775,11 @@ Lists can contain any type of objects, lists too:
|
|||||||
assert( list is Array ) // general interface
|
assert( list is Array ) // general interface
|
||||||
assert(list.size == 3)
|
assert(list.size == 3)
|
||||||
// second element is a list too:
|
// second element is a list too:
|
||||||
assert((list[1] as List).size == 2)
|
assert(list[1].size == 2)
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Notice usage of indexing. You can use negative indexes to offset from the end of the list; see more in [Lists](List.md).
|
Notice usage of indexing. You can use negative indexes to offset from the end of the list; see more in [Lists](List.md).
|
||||||
|
|
||||||
In general, bracket indexing may contain more than one selector:
|
|
||||||
|
|
||||||
value[i]
|
|
||||||
value[i, j]
|
|
||||||
|
|
||||||
For built-in lists, strings, maps, and buffers, the selector is usually a single value such as an `Int`, `Range`, or `Regex`.
|
|
||||||
For types with custom indexers, multiple selectors are packed into one list-like index object and passed to `getAt` / `putAt`.
|
|
||||||
|
|
||||||
When you want to "flatten" it to single array, you can use splat syntax:
|
When you want to "flatten" it to single array, you can use splat syntax:
|
||||||
|
|
||||||
[1, ...[2,3], 4]
|
[1, ...[2,3], 4]
|
||||||
@ -1040,7 +949,6 @@ Set are unordered collection of unique elements, see [Set]. Sets are [Iterable]
|
|||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Please see [Set] for detailed description.
|
Please see [Set] for detailed description.
|
||||||
For immutable set values, use `set.toImmutable()` and [ImmutableSet].
|
|
||||||
|
|
||||||
# Maps
|
# Maps
|
||||||
|
|
||||||
@ -1101,7 +1009,6 @@ Notes:
|
|||||||
- When you need computed (expression) keys or non-string keys, use `Map(...)` constructor with entries, e.g. `Map( ("a" + "b") => 1 )`, then merge with a literal if needed: `{ base: } + (computedKey => 42)`.
|
- When you need computed (expression) keys or non-string keys, use `Map(...)` constructor with entries, e.g. `Map( ("a" + "b") => 1 )`, then merge with a literal if needed: `{ base: } + (computedKey => 42)`.
|
||||||
|
|
||||||
Please see the [Map] reference for a deeper guide.
|
Please see the [Map] reference for a deeper guide.
|
||||||
For immutable map values, use `map.toImmutable()` and [ImmutableMap].
|
|
||||||
|
|
||||||
# Flow control operators
|
# Flow control operators
|
||||||
|
|
||||||
@ -1128,37 +1035,6 @@ Or, more neat:
|
|||||||
>>> just 3
|
>>> just 3
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
## compile if
|
|
||||||
|
|
||||||
`compile if` is a compile-time conditional. Unlike normal `if`, the compiler evaluates its condition while compiling
|
|
||||||
the file and completely skips the untaken branch. This is useful when some class or package may or may not be
|
|
||||||
available:
|
|
||||||
|
|
||||||
compile if (defined(Udp)) {
|
|
||||||
val socket = Udp()
|
|
||||||
println("udp is available")
|
|
||||||
} else {
|
|
||||||
println("udp is not available")
|
|
||||||
}
|
|
||||||
|
|
||||||
`compile if` also supports single-statement branches:
|
|
||||||
|
|
||||||
compile if (defined(lyng.io.net) && !defined(Udp))
|
|
||||||
println("network module exists, but Udp is not visible here")
|
|
||||||
else
|
|
||||||
println("either Udp exists or the module is unavailable")
|
|
||||||
|
|
||||||
Current condition syntax is intentionally limited to compile-time symbol checks:
|
|
||||||
|
|
||||||
- `defined(Name)`
|
|
||||||
- `defined(package.name)`
|
|
||||||
- `!`, `&&`, `||`
|
|
||||||
- parentheses
|
|
||||||
|
|
||||||
Examples:
|
|
||||||
compile if (defined(Udp) && defined(Tcp))
|
|
||||||
println("both transports are available")
|
|
||||||
|
|
||||||
## When
|
## When
|
||||||
|
|
||||||
See also: [Comprehensive guide to `when`](when.md)
|
See also: [Comprehensive guide to `when`](when.md)
|
||||||
@ -1347,8 +1223,8 @@ ends normally, without breaks. It allows override loop result value, for example
|
|||||||
to not calculate it in every iteration. For example, consider this naive prime number
|
to not calculate it in every iteration. For example, consider this naive prime number
|
||||||
test function (remember function return it's last expression result):
|
test function (remember function return it's last expression result):
|
||||||
|
|
||||||
fun naive_is_prime(candidate: Int) {
|
fun naive_is_prime(candidate) {
|
||||||
val x = candidate
|
val x = if( candidate !is Int) candidate.toInt() else candidate
|
||||||
var divisor = 1
|
var divisor = 1
|
||||||
while( ++divisor < x/2 || divisor == 2 ) {
|
while( ++divisor < x/2 || divisor == 2 ) {
|
||||||
if( x % divisor == 0 ) break false
|
if( x % divisor == 0 ) break false
|
||||||
@ -1423,48 +1299,12 @@ For loop are intended to traverse collections, and all other objects that suppor
|
|||||||
size and index access, like lists:
|
size and index access, like lists:
|
||||||
|
|
||||||
var letters = 0
|
var letters = 0
|
||||||
val words: List<String> = ["hello", "world"]
|
for( w in ["hello", "wolrd"]) {
|
||||||
for( w in words) {
|
letters += w.length
|
||||||
letters += (w as String).length
|
|
||||||
}
|
}
|
||||||
"total letters: "+letters
|
"total letters: "+letters
|
||||||
>>> "total letters: 10"
|
>>> "total letters: 10"
|
||||||
|
|
||||||
When you need a counting loop that goes backwards, use an explicit descending
|
|
||||||
range:
|
|
||||||
|
|
||||||
var sum = 0
|
|
||||||
for( i in 5 downTo 1 ) {
|
|
||||||
sum += i
|
|
||||||
}
|
|
||||||
sum
|
|
||||||
>>> 15
|
|
||||||
|
|
||||||
If the lower bound should be excluded, use `downUntil`:
|
|
||||||
|
|
||||||
val xs = []
|
|
||||||
for( i in 5 downUntil 1 ) {
|
|
||||||
xs.add(i)
|
|
||||||
}
|
|
||||||
xs
|
|
||||||
>>> [5,4,3,2]
|
|
||||||
|
|
||||||
This is intentionally explicit: `5..1` is an empty ascending range, not an
|
|
||||||
implicit reverse loop.
|
|
||||||
|
|
||||||
Descending loops also support `step`:
|
|
||||||
|
|
||||||
val xs = []
|
|
||||||
for( i in 10 downTo 1 step 3 ) {
|
|
||||||
xs.add(i)
|
|
||||||
}
|
|
||||||
xs
|
|
||||||
>>> [10,7,4,1]
|
|
||||||
|
|
||||||
For descending ranges, `step` stays positive. The direction comes from
|
|
||||||
`downTo` / `downUntil`, so `10 downTo 1 step 3` is valid, while
|
|
||||||
`10 downTo 1 step -3` is an error.
|
|
||||||
|
|
||||||
For loop support breaks the same as while loops above:
|
For loop support breaks the same as while loops above:
|
||||||
|
|
||||||
fun search(haystack, needle) {
|
fun search(haystack, needle) {
|
||||||
@ -1594,12 +1434,6 @@ It could be open and closed:
|
|||||||
assert( 5 !in (1..<5) )
|
assert( 5 !in (1..<5) )
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Descending ranges are explicit too:
|
|
||||||
|
|
||||||
assertEquals([5,4,3,2,1], (5 downTo 1).toList())
|
|
||||||
assertEquals([5,4,3,2], (5 downUntil 1).toList())
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Ranges could be inside other ranges:
|
Ranges could be inside other ranges:
|
||||||
|
|
||||||
assert( (2..3) in (1..10) )
|
assert( (2..3) in (1..10) )
|
||||||
@ -1612,19 +1446,11 @@ There are character ranges too:
|
|||||||
|
|
||||||
and you can use ranges in for-loops:
|
and you can use ranges in for-loops:
|
||||||
|
|
||||||
for( x in 'a'..<'c' ) println(x)
|
for( x in 'a' ..< 'c' ) println(x)
|
||||||
>>> a
|
>>> a
|
||||||
>>> b
|
>>> b
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Descending character ranges work the same way:
|
|
||||||
|
|
||||||
for( ch in 'e' downTo 'a' step 2 ) println(ch)
|
|
||||||
>>> e
|
|
||||||
>>> c
|
|
||||||
>>> a
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
See [Ranges](Range.md) for detailed documentation on it.
|
See [Ranges](Range.md) for detailed documentation on it.
|
||||||
|
|
||||||
# Time routines
|
# Time routines
|
||||||
@ -1688,94 +1514,28 @@ The type for the character objects is `Char`.
|
|||||||
|
|
||||||
### String literal escapes
|
### String literal escapes
|
||||||
|
|
||||||
Lyng string literals can use either double quotes or backticks:
|
|
||||||
|
|
||||||
val a = "hello"
|
|
||||||
val b = `hello`
|
|
||||||
assert(a == b)
|
|
||||||
|
|
||||||
| escape | ASCII value |
|
| escape | ASCII value |
|
||||||
|--------|-----------------------|
|
|--------|-----------------------|
|
||||||
| \n | 0x10, newline |
|
| \n | 0x10, newline |
|
||||||
| \r | 0x13, carriage return |
|
| \r | 0x13, carriage return |
|
||||||
| \t | 0x07, tabulation |
|
| \t | 0x07, tabulation |
|
||||||
| \\ | \ slash character |
|
| \\ | \ slash character |
|
||||||
| \uXXXX | unicode code point |
|
| \" | " double quote |
|
||||||
|
|
||||||
Delimiter-specific escapes:
|
|
||||||
|
|
||||||
| form | escape | value |
|
|
||||||
|--------|--------|------------------|
|
|
||||||
| `"..."` | \" | " double quote |
|
|
||||||
| `` `...` `` | \` | ` backtick |
|
|
||||||
|
|
||||||
Unicode escape form is exactly 4 hex digits, e.g. `"\u263A"` -> `☺`.
|
|
||||||
|
|
||||||
Other `\c` combinations, where c is any char except mentioned above, are left intact, e.g.:
|
Other `\c` combinations, where c is any char except mentioned above, are left intact, e.g.:
|
||||||
|
|
||||||
val s = "\a"
|
val s = "\a"
|
||||||
assert(s[0] == '\\')
|
assert(s[0] == '\')
|
||||||
assert(s[1] == 'a')
|
assert(s[1] == 'a')
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
same as:
|
same as:
|
||||||
|
|
||||||
val s = "\\a"
|
val s = "\\a"
|
||||||
assert(s[0] == '\\')
|
assert(s[0] == '\')
|
||||||
assert(s[1] == 'a')
|
assert(s[1] == 'a')
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
### String interpolation
|
|
||||||
|
|
||||||
Supported forms:
|
|
||||||
|
|
||||||
- `$name`
|
|
||||||
- `${expr}`
|
|
||||||
|
|
||||||
Literal dollar forms:
|
|
||||||
|
|
||||||
- `\$` -> `$`
|
|
||||||
- `$$` -> `$`
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
val name = "Lyng"
|
|
||||||
assertEquals("hello, Lyng!", "hello, $name!")
|
|
||||||
assertEquals("hello, Lyng!", `hello, $name!`)
|
|
||||||
assertEquals("sum=3", "sum=${1+2}")
|
|
||||||
assertEquals("sum=3", `sum=${1+2}`)
|
|
||||||
assertEquals("\$name", "\$name")
|
|
||||||
assertEquals("\$name", "$$name")
|
|
||||||
assertEquals("\$name", `\$name`)
|
|
||||||
assertEquals("\$name", `$$name`)
|
|
||||||
assertEquals("\\Lyng", "\\$name")
|
|
||||||
assertEquals("\\Lyng", `\\$name`)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Interpolation and `printf`-style formatting can be combined when needed:
|
|
||||||
|
|
||||||
val method = "transfer"
|
|
||||||
val argc = 2
|
|
||||||
val compact = "%s:%d"(method, argc)
|
|
||||||
assertEquals("call=transfer:2", "call=$compact")
|
|
||||||
assertEquals("[transfer:2] ok", "[${"%s:%d"(method, argc)}] ok")
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
Interpolation also works well with regex patterns. To keep a literal `$` in the
|
|
||||||
regex, escape it in the resulting pattern:
|
|
||||||
|
|
||||||
val currency = "USD"
|
|
||||||
val amount = 15
|
|
||||||
val escapedDollar = "\\$"
|
|
||||||
val re = Regex("^${currency}${escapedDollar}${amount}$")
|
|
||||||
assert("USD$15" =~ re)
|
|
||||||
assert("USD15" !~ re)
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
If you need old literal behavior in a file, add a leading directive comment:
|
|
||||||
|
|
||||||
// feature: interpolation: off
|
|
||||||
|
|
||||||
### Char literal escapes
|
### Char literal escapes
|
||||||
|
|
||||||
Are the same as in string literals with little difference:
|
Are the same as in string literals with little difference:
|
||||||
@ -1787,9 +1547,6 @@ Are the same as in string literals with little difference:
|
|||||||
| \t | 0x07, tabulation |
|
| \t | 0x07, tabulation |
|
||||||
| \\ | \ slash character |
|
| \\ | \ slash character |
|
||||||
| \' | ' apostrophe |
|
| \' | ' apostrophe |
|
||||||
| \uXXXX | unicode code point |
|
|
||||||
|
|
||||||
For char literals, use `'\\'` to represent a single backslash character; `'\'` is invalid.
|
|
||||||
|
|
||||||
### Char instance members
|
### Char instance members
|
||||||
|
|
||||||
@ -1856,14 +1613,6 @@ Open-ended ranges could be used to get start and end too:
|
|||||||
assertEquals( "pult", "catapult"[ 4.. ])
|
assertEquals( "pult", "catapult"[ 4.. ])
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
The same bracket syntax is also used by imported numeric modules such as `lyng.matrix`, where indexing can be multi-axis:
|
|
||||||
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val m: Matrix = matrix([[1, 2, 3], [4, 5, 6]])
|
|
||||||
assertEquals(6.0, m[1, 2])
|
|
||||||
>>> void
|
|
||||||
|
|
||||||
### String operations
|
### String operations
|
||||||
|
|
||||||
Concatenation is a `+`: `"hello " + name` works as expected. No confusion. There is also
|
Concatenation is a `+`: `"hello " + name` works as expected. No confusion. There is also
|
||||||
@ -1875,21 +1624,13 @@ Concatenation is a `+`: `"hello " + name` works as expected. No confusion. There
|
|||||||
|
|
||||||
Extraction:
|
Extraction:
|
||||||
|
|
||||||
("abcd42def"[ "\d+".re ] as RegexMatch).value
|
"abcd42def"[ "\d+".re ].value
|
||||||
>>> "42"
|
>>> "42"
|
||||||
|
|
||||||
Part match:
|
Part match:
|
||||||
|
|
||||||
assert( "abc foo def" =~ "f[oO]+".re )
|
assert( "abc foo def" =~ "f[oO]+".re )
|
||||||
assert( "foo" == ($~ as RegexMatch).value )
|
assert( "foo" == $~.value )
|
||||||
>>> void
|
|
||||||
|
|
||||||
Replacing text:
|
|
||||||
|
|
||||||
assertEquals("bonono", "banana".replace('a', 'o'))
|
|
||||||
assertEquals("a-b-c", "a.b.c".replace(".", "-")) // string patterns are literal
|
|
||||||
assertEquals("v#.#.#", "v1.2.3".replace("\d+".re, "#"))
|
|
||||||
assertEquals("v[1].[2].[3]", "v1.2.3".replace("(\d+)".re) { m -> "[" + m[1] + "]" })
|
|
||||||
>>> void
|
>>> void
|
||||||
|
|
||||||
Repeating the fragment:
|
Repeating the fragment:
|
||||||
@ -1927,8 +1668,6 @@ A typical set of String functions includes:
|
|||||||
| characters | create [List] of characters (1) |
|
| characters | create [List] of characters (1) |
|
||||||
| encodeUtf8() | returns [Buffer] with characters encoded to utf8 |
|
| encodeUtf8() | returns [Buffer] with characters encoded to utf8 |
|
||||||
| matches(re) | matches the regular expression (2) |
|
| matches(re) | matches the regular expression (2) |
|
||||||
| replace(old, new) | replace all literal or regex matches; regex needs [Regex] |
|
|
||||||
| replaceFirst(old,new)| replace the first literal or regex match |
|
|
||||||
| | |
|
| | |
|
||||||
|
|
||||||
(1)
|
(1)
|
||||||
@ -1992,7 +1731,6 @@ Lambda avoid unnecessary execution if assertion is not failed. for example:
|
|||||||
| π | See [math](math.md) |
|
| π | See [math](math.md) |
|
||||||
|
|
||||||
[List]: List.md
|
[List]: List.md
|
||||||
[ImmutableList]: ImmutableList.md
|
|
||||||
|
|
||||||
[Testing]: Testing.md
|
[Testing]: Testing.md
|
||||||
|
|
||||||
@ -2009,10 +1747,8 @@ Lambda avoid unnecessary execution if assertion is not failed. for example:
|
|||||||
[string formatting]: https://github.com/sergeych/mp_stools?tab=readme-ov-file#sprintf-syntax-summary
|
[string formatting]: https://github.com/sergeych/mp_stools?tab=readme-ov-file#sprintf-syntax-summary
|
||||||
|
|
||||||
[Set]: Set.md
|
[Set]: Set.md
|
||||||
[ImmutableSet]: ImmutableSet.md
|
|
||||||
|
|
||||||
[Map]: Map.md
|
[Map]: Map.md
|
||||||
[ImmutableMap]: ImmutableMap.md
|
|
||||||
|
|
||||||
[Buffer]: Buffer.md
|
[Buffer]: Buffer.md
|
||||||
|
|
||||||
@ -2145,7 +1881,7 @@ You can add new methods and properties to existing classes without modifying the
|
|||||||
|
|
||||||
### Extension properties
|
### Extension properties
|
||||||
|
|
||||||
val Int.isEven get() = this % 2 == 0
|
val Int.isEven = this % 2 == 0
|
||||||
4.isEven
|
4.isEven
|
||||||
>>> true
|
>>> true
|
||||||
|
|
||||||
@ -2155,30 +1891,6 @@ Example with custom accessors:
|
|||||||
"abc".firstChar
|
"abc".firstChar
|
||||||
>>> 'a'
|
>>> 'a'
|
||||||
|
|
||||||
### Extension indexers
|
|
||||||
|
|
||||||
Indexers can also be extended by overriding `getAt` and `putAt` on the receiver:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
object Storage
|
|
||||||
|
|
||||||
var storageData = {}
|
|
||||||
|
|
||||||
override fun Storage.getAt(key: String): Object? {
|
|
||||||
storageData[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun Storage.putAt(key: String, value: Object) {
|
|
||||||
storageData[key] = value
|
|
||||||
}
|
|
||||||
|
|
||||||
Storage["answer"] = 42
|
|
||||||
val answer: Int? = Storage["answer"]
|
|
||||||
assertEquals(42, answer)
|
|
||||||
```
|
|
||||||
|
|
||||||
This works for classes and named singleton `object` declarations. Bracket syntax is lowered to `getAt` / `putAt`, and multiple selectors are packed into one list-like index object the same way as other custom indexers.
|
|
||||||
|
|
||||||
Extension members are **scope-isolated**: they are visible only in the scope where they are defined and its children. This prevents name collisions and improves security.
|
Extension members are **scope-isolated**: they are visible only in the scope where they are defined and its children. This prevents name collisions and improves security.
|
||||||
|
|
||||||
To get details on OOP in Lyng, see [OOP notes](OOP.md).
|
To get details on OOP in Lyng, see [OOP notes](OOP.md).
|
||||||
|
|||||||
@ -1,235 +1,10 @@
|
|||||||
# What's New in Lyng
|
# What's New in Lyng
|
||||||
|
|
||||||
This document highlights the current Lyng release, **1.5.5**, and the broader additions from the 1.5 cycle.
|
This document highlights the latest additions and improvements to the Lyng language and its ecosystem.
|
||||||
It is intentionally user-facing: new language features, new modules, new tools, and the practical things you can build with them.
|
For a programmer-focused migration summary, see `docs/whats_new_1_5.md`.
|
||||||
For a programmer-focused migration summary across 1.5.x, see `docs/whats_new_1_5.md`.
|
|
||||||
|
|
||||||
## Release 1.5.5 Highlights
|
|
||||||
|
|
||||||
- `1.5.5` extends the 1.5 line with practical database APIs, first-class calendar dates, and better coroutine building blocks.
|
|
||||||
- The 1.5 line now brings together richer ranges and loops, interpolation, math modules, immutable and observable collections, richer `lyngio`, and much better CLI/IDE support.
|
|
||||||
- `1.5.5` adds `Channel`, `LaunchPool`, and `joinAll()` so coroutine-heavy scripts can coordinate work more directly.
|
|
||||||
- `1.5.5` adds `Date`, the portable `lyng.io.db` layer, SQLite/JDBC providers, and a compatibility `lyng.legacy_digest` module.
|
|
||||||
- `1.5.5` also continues runtime/compiler hardening with better import dispatch, faster exact lambda calls, and correct `val +=`/`-=` behavior for mutating types versus real reassignment.
|
|
||||||
- The docs, homepage samples, and release metadata now point at the current stable version.
|
|
||||||
|
|
||||||
## User Highlights Across 1.5.x
|
|
||||||
|
|
||||||
- Descending ranges and loops with `downTo` / `downUntil`
|
|
||||||
- String interpolation with `$name` and `${expr}`
|
|
||||||
- Backtick string literals for raw-ish string text
|
|
||||||
- Decimal arithmetic, matrices/vectors, and complex numbers
|
|
||||||
- Calendar `Date` support in `lyng.time`
|
|
||||||
- `Channel`, `LaunchPool`, and `joinAll()` for coroutine workflows
|
|
||||||
- Immutable collections and opt-in `ObservableList`
|
|
||||||
- Rich `lyngio` modules for SQLite/JDBC databases, console, HTTP, WebSocket, TCP, and UDP
|
|
||||||
- Legacy SHA-1 compatibility helpers in `lyng.legacy_digest`
|
|
||||||
- CLI improvements including the built-in formatter `lyng fmt`
|
|
||||||
- Better IDE support and stronger docs around the released feature set
|
|
||||||
|
|
||||||
## Language Features
|
## Language Features
|
||||||
|
|
||||||
### Descending Ranges and Loops
|
|
||||||
Lyng ranges are no longer just ascending. You can now write explicit descending ranges with inclusive or exclusive lower bounds.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
assertEquals([5,4,3,2,1], (5 downTo 1).toList())
|
|
||||||
assertEquals([5,4,3,2], (5 downUntil 1).toList())
|
|
||||||
|
|
||||||
for (i in 10 downTo 1 step 3) {
|
|
||||||
println(i)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
This also works for characters:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
assertEquals(['e','c','a'], ('e' downTo 'a' step 2).toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
See [Range](Range.md).
|
|
||||||
|
|
||||||
### String Interpolation
|
|
||||||
Lyng 1.5.1 added built-in string interpolation:
|
|
||||||
|
|
||||||
- `$name`
|
|
||||||
- `${expr}`
|
|
||||||
|
|
||||||
Literal dollar forms are explicit too:
|
|
||||||
|
|
||||||
- `\$` -> `$`
|
|
||||||
- `$$` -> `$`
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val name = "Lyng"
|
|
||||||
assertEquals("hello, Lyng!", "hello, $name!")
|
|
||||||
assertEquals("sum=3", "sum=${1+2}")
|
|
||||||
assertEquals("\$name", "\$name")
|
|
||||||
assertEquals("\$name", "$$name")
|
|
||||||
```
|
|
||||||
|
|
||||||
If you need legacy literal-dollar behavior in a file, add:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
// feature: interpolation: off
|
|
||||||
```
|
|
||||||
|
|
||||||
See [Tutorial](tutorial.md).
|
|
||||||
|
|
||||||
### Matrix and Vector Module (`lyng.matrix`)
|
|
||||||
Lyng now ships a dense linear algebra module with immutable double-precision `Matrix` and `Vector` types.
|
|
||||||
|
|
||||||
It provides:
|
|
||||||
|
|
||||||
- `matrix([[...]])` and `vector([...])`
|
|
||||||
- matrix multiplication
|
|
||||||
- matrix inversion
|
|
||||||
- determinant, trace, rank
|
|
||||||
- solving `A * x = b`
|
|
||||||
- vector operations such as `dot`, `normalize`, `cross`, and `outer`
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val a: Matrix = matrix([[4, 7], [2, 6]])
|
|
||||||
val inv: Matrix = a.inverse()
|
|
||||||
assert(abs(inv.get(0, 0) - 0.6) < 1e-9)
|
|
||||||
```
|
|
||||||
|
|
||||||
Matrices also support Lyng-style slicing:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.matrix
|
|
||||||
|
|
||||||
val m: Matrix = matrix([[1, 2, 3], [4, 5, 6], [7, 8, 9]])
|
|
||||||
assertEquals(6.0, m[1, 2])
|
|
||||||
val column: Matrix = m[0..2, 2]
|
|
||||||
val tail: Matrix = m[1.., 1..]
|
|
||||||
assertEquals([[3.0], [6.0], [9.0]], column.toList())
|
|
||||||
assertEquals([[5.0, 6.0], [8.0, 9.0]], tail.toList())
|
|
||||||
```
|
|
||||||
|
|
||||||
See [Matrix](Matrix.md).
|
|
||||||
|
|
||||||
### Multiple Selectors in Bracket Indexing
|
|
||||||
Bracket indexing now accepts more than one selector:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
value[i]
|
|
||||||
value[i, j]
|
|
||||||
value[i, j, k]
|
|
||||||
```
|
|
||||||
|
|
||||||
For custom indexers, multiple selectors are packed into one list-like index object and dispatched through `getAt` / `putAt`.
|
|
||||||
This is the rule used by `lyng.matrix` and by embedding APIs for Kotlin-backed indexers.
|
|
||||||
|
|
||||||
### Decimal Arithmetic Module (`lyng.decimal`)
|
|
||||||
Lyng now ships a first-class decimal module built as a regular extension library rather than a deep core special case.
|
|
||||||
|
|
||||||
It provides:
|
|
||||||
|
|
||||||
- `Decimal`
|
|
||||||
- convenient `.d` conversions from `Int`, `Real`, and `String`
|
|
||||||
- mixed arithmetic with `Int` and `Real`
|
|
||||||
- local division precision and rounding control via `withDecimalContext(...)`
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.decimal
|
|
||||||
|
|
||||||
assertEquals("3", (1 + 2.d).toStringExpanded())
|
|
||||||
assertEquals("0.30000000000000004", (0.1 + 0.2).d.toStringExpanded())
|
|
||||||
assertEquals("0.3", "0.3".d.toStringExpanded())
|
|
||||||
|
|
||||||
assertEquals(
|
|
||||||
"0.3333333333",
|
|
||||||
withDecimalContext(10) { (1.d / 3.d).toStringExpanded() }
|
|
||||||
)
|
|
||||||
```
|
|
||||||
|
|
||||||
The distinction between `Real -> Decimal` and exact decimal parsing is explicit by design:
|
|
||||||
|
|
||||||
- `2.2.d` converts the current `Real` value
|
|
||||||
- `"2.2".d` parses exact decimal text
|
|
||||||
|
|
||||||
See [Decimal](Decimal.md).
|
|
||||||
|
|
||||||
### Complex Numbers (`lyng.complex`)
|
|
||||||
Lyng also ships a complex-number module for ordinary arithmetic in the complex plane.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.complex
|
|
||||||
|
|
||||||
assertEquals(Complex(1.0, 2.0), 1 + 2.i)
|
|
||||||
assertEquals(Complex(2.0, 2.0), 2.i + 2)
|
|
||||||
|
|
||||||
val z = 1 + π.i
|
|
||||||
println(z.exp())
|
|
||||||
```
|
|
||||||
|
|
||||||
See [Complex](Complex.md).
|
|
||||||
|
|
||||||
### Legacy Digest Module (`lyng.legacy_digest`)
|
|
||||||
|
|
||||||
For situations where an external protocol or file format requires a SHA-1 value,
|
|
||||||
Lyng now ships a `lyng.legacy_digest` module backed by a pure Kotlin/KMP
|
|
||||||
implementation with no extra dependencies.
|
|
||||||
|
|
||||||
> ⚠️ SHA-1 is cryptographically broken. Use only for legacy-compatibility work.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.legacy_digest
|
|
||||||
|
|
||||||
val hex = LegacyDigest.sha1("abc")
|
|
||||||
// → "a9993e364706816aba3e25717850c26c9cd0d89d"
|
|
||||||
|
|
||||||
// Also accepts raw bytes:
|
|
||||||
import lyng.buffer
|
|
||||||
val buf = Buffer.decodeHex("616263")
|
|
||||||
assertEquals(hex, LegacyDigest.sha1(buf))
|
|
||||||
```
|
|
||||||
|
|
||||||
The name `LegacyDigest` is intentional: it signals that these algorithms belong
|
|
||||||
to a compatibility layer, not to a current security toolkit.
|
|
||||||
|
|
||||||
See [LegacyDigest](LegacyDigest.md).
|
|
||||||
|
|
||||||
### Binary Operator Interop Registry
|
|
||||||
Lyng now provides a general mechanism for mixed binary operators through `lyng.operators`.
|
|
||||||
|
|
||||||
This solves cases like:
|
|
||||||
|
|
||||||
- `Int + MyType`
|
|
||||||
- `Real < MyType`
|
|
||||||
- `Int == MyType`
|
|
||||||
|
|
||||||
without requiring changes to built-in classes.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.operators
|
|
||||||
|
|
||||||
class DecimalBox(val value: Int) {
|
|
||||||
fun plus(other: DecimalBox) = DecimalBox(value + other.value)
|
|
||||||
fun compareTo(other: DecimalBox) = value <=> other.value
|
|
||||||
}
|
|
||||||
|
|
||||||
OperatorInterop.register(
|
|
||||||
Int,
|
|
||||||
DecimalBox,
|
|
||||||
DecimalBox,
|
|
||||||
[BinaryOperator.Plus, BinaryOperator.Compare, BinaryOperator.Equals],
|
|
||||||
{ x: Int -> DecimalBox(x) },
|
|
||||||
{ x: DecimalBox -> x }
|
|
||||||
)
|
|
||||||
|
|
||||||
assertEquals(DecimalBox(3), 1 + DecimalBox(2))
|
|
||||||
assert(1 < DecimalBox(2))
|
|
||||||
assert(2 == DecimalBox(2))
|
|
||||||
```
|
|
||||||
|
|
||||||
`lyng.decimal` uses this same mechanism internally to interoperate with `Int` and `Real`.
|
|
||||||
|
|
||||||
See [Operator Interop Registry](OperatorInterop.md).
|
|
||||||
|
|
||||||
### Class Properties with Accessors
|
### Class Properties with Accessors
|
||||||
Classes now support properties with custom `get()` and `set()` accessors. Properties in Lyng do **not** have automatic backing fields; they are pure accessors.
|
Classes now support properties with custom `get()` and `set()` accessors. Properties in Lyng do **not** have automatic backing fields; they are pure accessors.
|
||||||
|
|
||||||
@ -327,30 +102,13 @@ Singleton objects are declared using the `object` keyword. They provide a conven
|
|||||||
|
|
||||||
```lyng
|
```lyng
|
||||||
object Config {
|
object Config {
|
||||||
val version = "1.5.6-SNAPSHOT"
|
val version = "1.5.0-SNAPSHOT"
|
||||||
fun show() = println("Config version: " + version)
|
fun show() = println("Config version: " + version)
|
||||||
}
|
}
|
||||||
|
|
||||||
Config.show()
|
Config.show()
|
||||||
```
|
```
|
||||||
|
|
||||||
Named singleton objects can also be used as extension receivers:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
object X {
|
|
||||||
fun base() = "base"
|
|
||||||
}
|
|
||||||
|
|
||||||
fun X.decorate(value): String {
|
|
||||||
this.base() + ":" + value.toString()
|
|
||||||
}
|
|
||||||
|
|
||||||
val X.tag get() = this.base() + ":tag"
|
|
||||||
|
|
||||||
assertEquals("base:42", X.decorate(42))
|
|
||||||
assertEquals("base:tag", X.tag)
|
|
||||||
```
|
|
||||||
|
|
||||||
### Nested Declarations and Lifted Enums
|
### Nested Declarations and Lifted Enums
|
||||||
You can now declare classes, objects, enums, and type aliases inside another class. These nested declarations live in the class namespace (no outer instance capture) and are accessed with a qualifier.
|
You can now declare classes, objects, enums, and type aliases inside another class. These nested declarations live in the class namespace (no outer instance capture) and are accessed with a qualifier.
|
||||||
|
|
||||||
@ -467,124 +225,8 @@ x.clamp(0..10) // returns 10
|
|||||||
|
|
||||||
`clamp()` correctly handles inclusive (`..`) and exclusive (`..<`) ranges. For discrete types like `Int` and `Char`, clamping to an exclusive upper bound returns the previous value.
|
`clamp()` correctly handles inclusive (`..`) and exclusive (`..<`) ranges. For discrete types like `Int` and `Char`, clamping to an exclusive upper bound returns the previous value.
|
||||||
|
|
||||||
### Immutable Collections
|
|
||||||
Lyng 1.5 adds immutable collection types for APIs that should not expose mutable state through aliases:
|
|
||||||
|
|
||||||
- `ImmutableList`
|
|
||||||
- `ImmutableSet`
|
|
||||||
- `ImmutableMap`
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val a = ImmutableList(1,2,3)
|
|
||||||
val b = a + 4
|
|
||||||
|
|
||||||
assertEquals(ImmutableList(1,2,3), a)
|
|
||||||
assertEquals(ImmutableList(1,2,3,4), b)
|
|
||||||
```
|
|
||||||
|
|
||||||
See [ImmutableList](ImmutableList.md), [ImmutableSet](ImmutableSet.md), and [ImmutableMap](ImmutableMap.md).
|
|
||||||
|
|
||||||
### Observable Mutable Lists
|
|
||||||
For reactive-style code, `lyng.observable` provides `ObservableList` with hooks and change streams.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.observable
|
|
||||||
|
|
||||||
val xs = [1,2].observable()
|
|
||||||
xs.onChange { println("changed") }
|
|
||||||
xs += 3
|
|
||||||
```
|
|
||||||
|
|
||||||
You can validate or reject mutations in `beforeChange`, listen in `onChange`, and consume structured change events from `changes()`.
|
|
||||||
|
|
||||||
See [ObservableList](ObservableList.md).
|
|
||||||
|
|
||||||
### Random API
|
|
||||||
The standard library now includes a built-in random API plus deterministic seeded generators.
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
val rng = Random.seeded(1234)
|
|
||||||
assert(rng.next(1..10) in 1..10)
|
|
||||||
assert(rng.next('a'..<'f') in 'a'..<'f')
|
|
||||||
```
|
|
||||||
|
|
||||||
Use:
|
|
||||||
|
|
||||||
- `Random.nextInt()`
|
|
||||||
- `Random.nextFloat()`
|
|
||||||
- `Random.next(range)`
|
|
||||||
- `Random.seeded(seed)`
|
|
||||||
|
|
||||||
## Tooling and Infrastructure
|
## Tooling and Infrastructure
|
||||||
|
|
||||||
### Rich Console Apps with `lyng.io.console`
|
|
||||||
`lyngio` now includes a real console module for terminal applications:
|
|
||||||
|
|
||||||
- TTY detection
|
|
||||||
- screen clearing and cursor movement
|
|
||||||
- alternate screen buffer
|
|
||||||
- raw input mode
|
|
||||||
- typed key and resize events
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.console
|
|
||||||
|
|
||||||
Console.enterAltScreen()
|
|
||||||
Console.clear()
|
|
||||||
Console.moveTo(1, 1)
|
|
||||||
Console.write("Hello from Lyng console app")
|
|
||||||
Console.flush()
|
|
||||||
Console.leaveAltScreen()
|
|
||||||
```
|
|
||||||
|
|
||||||
The repository includes a full interactive Tetris sample built on this API.
|
|
||||||
|
|
||||||
See [lyng.io.console](lyng.io.console.md).
|
|
||||||
|
|
||||||
### HTTP, WebSocket, TCP, and UDP in `lyngio`
|
|
||||||
`lyngio` grew from filesystem/process support into a broader application-facing I/O library. In 1.5.x it includes:
|
|
||||||
|
|
||||||
- `lyng.io.http` for HTTP/HTTPS client calls
|
|
||||||
- `lyng.io.ws` for WebSocket clients
|
|
||||||
- `lyng.io.net` for raw TCP/UDP transport
|
|
||||||
|
|
||||||
HTTP example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.http
|
|
||||||
|
|
||||||
val r = Http.get("https://example.com")
|
|
||||||
println(r.status)
|
|
||||||
println(r.text())
|
|
||||||
```
|
|
||||||
|
|
||||||
TCP example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val socket = Net.tcpConnect("127.0.0.1", 4040)
|
|
||||||
socket.writeUtf8("ping")
|
|
||||||
socket.flush()
|
|
||||||
println(socket.readLine())
|
|
||||||
socket.close()
|
|
||||||
```
|
|
||||||
|
|
||||||
WebSocket example:
|
|
||||||
|
|
||||||
```lyng
|
|
||||||
import lyng.io.ws
|
|
||||||
|
|
||||||
val ws = Ws.connect("wss://example.com/socket")
|
|
||||||
ws.sendText("hello")
|
|
||||||
println(ws.receive())
|
|
||||||
ws.close()
|
|
||||||
```
|
|
||||||
|
|
||||||
These modules are capability-gated and host-installed, keeping Lyng safe by default while making networked scripts practical when enabled.
|
|
||||||
|
|
||||||
See [lyngio overview](lyngio.md), [lyng.io.db](lyng.io.db.md), [lyng.io.http](lyng.io.http.md), [lyng.io.ws](lyng.io.ws.md), and [lyng.io.net](lyng.io.net.md).
|
|
||||||
|
|
||||||
### CLI: Formatting Command
|
### CLI: Formatting Command
|
||||||
A new `fmt` subcommand has been added to the Lyng CLI.
|
A new `fmt` subcommand has been added to the Lyng CLI.
|
||||||
|
|
||||||
@ -594,15 +236,6 @@ lyng fmt --in-place MyFile.lyng # Format file in-place
|
|||||||
lyng fmt --check MyFile.lyng # Check if file needs formatting
|
lyng fmt --check MyFile.lyng # Check if file needs formatting
|
||||||
```
|
```
|
||||||
|
|
||||||
### CLI: Better Terminal Workflows
|
|
||||||
The CLI is no longer just a script launcher. In the 1.5 line it also gained:
|
|
||||||
|
|
||||||
- built-in formatter support
|
|
||||||
- integrated `lyng.io.console` support for terminal programs
|
|
||||||
- downloadable packaged distributions for easier local use
|
|
||||||
|
|
||||||
This makes CLI-first scripting and console applications much more practical than in earlier releases.
|
|
||||||
|
|
||||||
### IDEA Plugin: Autocompletion
|
### IDEA Plugin: Autocompletion
|
||||||
Experimental lightweight autocompletion is now available in the IntelliJ plugin. It features type-aware member suggestions and inheritance-aware completion.
|
Experimental lightweight autocompletion is now available in the IntelliJ plugin. It features type-aware member suggestions and inheritance-aware completion.
|
||||||
|
|
||||||
@ -615,7 +248,6 @@ The `Obj.getLyngExceptionMessageWithStackTrace()` extension method has been adde
|
|||||||
Lyng now provides a public Kotlin reflection bridge and a Lyng‑first class binding workflow. This is the **preferred** way to write Kotlin extensions and library integrations:
|
Lyng now provides a public Kotlin reflection bridge and a Lyng‑first class binding workflow. This is the **preferred** way to write Kotlin extensions and library integrations:
|
||||||
|
|
||||||
- **Bridge resolver**: explicit handles for values, vars, and callables with predictable lookup rules.
|
- **Bridge resolver**: explicit handles for values, vars, and callables with predictable lookup rules.
|
||||||
- **Class bridge binding**: declare extern surfaces in Lyng (`extern` members, or members inside `extern class/object`) and bind the implementations in Kotlin before the first instance is created.
|
- **Class bridge binding**: declare classes/members in Lyng (marked `extern`) and bind the implementations in Kotlin before the first instance is created.
|
||||||
- **Extern declaration rule**: `extern class` / `extern object` are declaration-only; all members in their bodies are implicitly extern.
|
|
||||||
|
|
||||||
See **Embedding Lyng** for full samples and usage details.
|
See **Embedding Lyng** for full samples and usage details.
|
||||||
|
|||||||
@ -18,16 +18,11 @@ In particular, it means no slow and flaky runtime lookups. Once compiled, code g
|
|||||||
|
|
||||||
The API is fixed and will be kept with further Lyng core changes. It is now the recommended way to write Lyng extensions in Kotlin. It is much simpler and more elegant than the internal one. See [Kotlin Bridge Binding](../notes/kotlin_bridge_binding.md).
|
The API is fixed and will be kept with further Lyng core changes. It is now the recommended way to write Lyng extensions in Kotlin. It is much simpler and more elegant than the internal one. See [Kotlin Bridge Binding](../notes/kotlin_bridge_binding.md).
|
||||||
|
|
||||||
Extern declaration clarification:
|
|
||||||
- `extern class` / `extern object` are pure extern surfaces.
|
|
||||||
- Members inside them are implicitly extern (`extern` on a member is optional/redundant).
|
|
||||||
- Lyng method/property bodies for these declarations should be implemented as extensions instead.
|
|
||||||
|
|
||||||
### Smart types system
|
### Smart types system
|
||||||
|
|
||||||
- **Deep inference**: The compiler analyzes types of symbols along the execution path and in many cases eliminates unnecessary casts or type specifications.
|
- **Deep inference**: The compiler analyzes types of symbols along the execution path and in many cases eliminates unnecessary casts or type specifications.
|
||||||
- **Union and intersection types**: `A & B`, `A | B`.
|
- **Union and intersection types**: `A & B`, `A | B`.
|
||||||
- **Generics**: Generic types are first-class citizens with support for [bounds and variance](generics.md). Type params are erased by default and are reified only when needed (e.g., `T::class`, `T is ...`, `as T`, or in extern-facing APIs), which enables checks like `A in T` when `T` is reified.
|
- **Generics**: Generic types are first-class citizens with support for [bounds and variance](generics.md). No type erasure: in a generic function you can, for example, check `A in T`, where T is the generic type.
|
||||||
- **Inner classes and enums**: Full support for nested declarations, including [Enums with lifting](OOP.md#lifted-enum-entries).
|
- **Inner classes and enums**: Full support for nested declarations, including [Enums with lifting](OOP.md#lifted-enum-entries).
|
||||||
|
|
||||||
## Other highlights
|
## Other highlights
|
||||||
|
|||||||
@ -1,325 +0,0 @@
|
|||||||
#!/usr/bin/env lyng
|
|
||||||
|
|
||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
import lyng.io.fs
|
|
||||||
|
|
||||||
val DB_FILE_NAME = "contents.db"
|
|
||||||
val ANSI_ESC = "\u001b["
|
|
||||||
val NEWLINE = "\n"
|
|
||||||
val WINDOWS_SEPARATOR = "\\"
|
|
||||||
val SQLITE_JOURNAL_SUFFIXES = ["-wal", "-shm", "-journal"]
|
|
||||||
|
|
||||||
val USAGE_TEXT = "
|
|
||||||
Lyng content index
|
|
||||||
Scan a directory tree, diff it against a SQLite snapshot, and optionally refresh the snapshot.
|
|
||||||
|
|
||||||
usage:
|
|
||||||
lyng examples/content_index_db.lyng <root> [-u|--update]
|
|
||||||
|
|
||||||
options:
|
|
||||||
-u, --update write the current scan back to $DB_FILE_NAME
|
|
||||||
|
|
||||||
notes:
|
|
||||||
- the database lives inside <root>/$DB_FILE_NAME
|
|
||||||
- on first run the snapshot is created automatically
|
|
||||||
- the script ignores its own SQLite sidecar files
|
|
||||||
"
|
|
||||||
|
|
||||||
val CREATE_FILE_INDEX_SQL = "
|
|
||||||
create table if not exists file_index(
|
|
||||||
path text primary key not null,
|
|
||||||
size integer not null,
|
|
||||||
mtime integer not null
|
|
||||||
)
|
|
||||||
"
|
|
||||||
|
|
||||||
val CREATE_CURRENT_SCAN_SQL = "
|
|
||||||
create temp table current_scan(
|
|
||||||
path text primary key not null,
|
|
||||||
size integer not null,
|
|
||||||
mtime integer not null
|
|
||||||
)
|
|
||||||
"
|
|
||||||
|
|
||||||
val SELECT_ADDED_SQL = "
|
|
||||||
select
|
|
||||||
c.path,
|
|
||||||
c.size,
|
|
||||||
c.mtime
|
|
||||||
from current_scan c
|
|
||||||
left join file_index f on f.path = c.path
|
|
||||||
where f.path is null
|
|
||||||
order by c.path
|
|
||||||
"
|
|
||||||
|
|
||||||
val SELECT_REMOVED_SQL = "
|
|
||||||
select f.path, f.size, f.mtime
|
|
||||||
from file_index f
|
|
||||||
left join current_scan c on c.path = f.path
|
|
||||||
where c.path is null
|
|
||||||
order by f.path
|
|
||||||
"
|
|
||||||
|
|
||||||
val SELECT_CHANGED_SQL = "
|
|
||||||
select c.path, f.size as old_size, c.size as new_size, f.mtime as old_mtime,
|
|
||||||
c.mtime as new_mtime
|
|
||||||
from current_scan c
|
|
||||||
join file_index f on f.path = c.path
|
|
||||||
where c.size != f.size or c.mtime != f.mtime
|
|
||||||
order by c.path
|
|
||||||
"
|
|
||||||
|
|
||||||
val DELETE_MISSING_SQL = "
|
|
||||||
delete from file_index
|
|
||||||
where not exists (
|
|
||||||
select 1
|
|
||||||
from current_scan c
|
|
||||||
where c.path = file_index.path
|
|
||||||
)
|
|
||||||
"
|
|
||||||
|
|
||||||
val UPSERT_SCAN_SQL = "
|
|
||||||
insert or replace into file_index(path, size, mtime)
|
|
||||||
select path, size, mtime
|
|
||||||
from current_scan
|
|
||||||
"
|
|
||||||
|
|
||||||
val INSERT_SCAN_ROW_SQL = "
|
|
||||||
insert into current_scan(path, size, mtime)
|
|
||||||
values(?, ?, ?)
|
|
||||||
"
|
|
||||||
|
|
||||||
val USE_COLOR = true
|
|
||||||
|
|
||||||
class CliOptions(val rootText: String, val updateSnapshot: Bool) {}
|
|
||||||
|
|
||||||
fun out(text: String? = null): Void {
|
|
||||||
if (text == null) {
|
|
||||||
print(NEWLINE)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
print(text + NEWLINE)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun paint(code: String, text: String): String {
|
|
||||||
if (!USE_COLOR) return text
|
|
||||||
ANSI_ESC + code + "m" + text + ANSI_ESC + "0m"
|
|
||||||
}
|
|
||||||
|
|
||||||
fun bold(text: String): String = paint("1", text)
|
|
||||||
fun dim(text: String): String = paint("2", text)
|
|
||||||
fun cyan(text: String): String = paint("36", text)
|
|
||||||
fun green(text: String): String = paint("32", text)
|
|
||||||
fun yellow(text: String): String = paint("33", text)
|
|
||||||
fun red(text: String): String = paint("31", text)
|
|
||||||
|
|
||||||
fun signed(value: Int): String = if (value > 0) "+" + value else value.toString()
|
|
||||||
|
|
||||||
fun plural(count: Int, one: String, many: String): String {
|
|
||||||
if (count == 1) return one
|
|
||||||
many
|
|
||||||
}
|
|
||||||
|
|
||||||
fun childPath(parent: Path, name: String): Path {
|
|
||||||
val base = parent.toString()
|
|
||||||
if (base.endsWith("/") || base.endsWith(WINDOWS_SEPARATOR)) {
|
|
||||||
return Path(base + name)
|
|
||||||
}
|
|
||||||
Path(base + "/" + name)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun relativePath(root: Path, file: Path): String {
|
|
||||||
val parts: List<String> = []
|
|
||||||
for (i in root.segments.size..<file.segments.size) {
|
|
||||||
parts.add(file.segments[i] as String)
|
|
||||||
}
|
|
||||||
parts.joinToString("/")
|
|
||||||
}
|
|
||||||
|
|
||||||
fun isDatabaseArtifact(relative: String): Bool {
|
|
||||||
relative == DB_FILE_NAME || SQLITE_JOURNAL_SUFFIXES.any { relative == DB_FILE_NAME + (it as String) }
|
|
||||||
}
|
|
||||||
|
|
||||||
fun printUsage(message: String? = null): Void {
|
|
||||||
if (message != null && message.trim().isNotEmpty()) {
|
|
||||||
out(red("error: ") + message)
|
|
||||||
out()
|
|
||||||
}
|
|
||||||
|
|
||||||
out(bold(USAGE_TEXT))
|
|
||||||
}
|
|
||||||
|
|
||||||
fun parseArgs(argv: List<String>): CliOptions? {
|
|
||||||
var rootText: String? = null
|
|
||||||
var updateSnapshot = false
|
|
||||||
|
|
||||||
for (arg in argv) {
|
|
||||||
when (arg) {
|
|
||||||
"-u", "--update" -> updateSnapshot = true
|
|
||||||
"-h", "--help" -> {
|
|
||||||
printUsage()
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
else -> {
|
|
||||||
if (arg.startsWith("-")) {
|
|
||||||
printUsage("unknown option: " + arg)
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
if (rootText != null) {
|
|
||||||
printUsage("only one root path is allowed")
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
rootText = arg
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rootText == null) {
|
|
||||||
printUsage("missing required <root> argument")
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
CliOptions(rootText as String, updateSnapshot)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun printBanner(root: Path, dbFile: Path, dbWasCreated: Bool, updateSnapshot: Bool): Void {
|
|
||||||
val mode =
|
|
||||||
if (dbWasCreated) "bootstrap snapshot"
|
|
||||||
else if (updateSnapshot) "scan + refresh snapshot"
|
|
||||||
else "scan only"
|
|
||||||
|
|
||||||
out(cyan("== Lyng content index =="))
|
|
||||||
out(dim("root: " + root))
|
|
||||||
out(dim("db: " + dbFile))
|
|
||||||
out(dim("mode: " + mode))
|
|
||||||
out()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun printSection(title: String, accent: (String)->String, rows: List<SqlRow>, render: (SqlRow)->String): Void {
|
|
||||||
out(accent(title + " (" + rows.size + ")"))
|
|
||||||
if (rows.isEmpty()) {
|
|
||||||
out(dim(" none"))
|
|
||||||
out()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
for (row in rows) {
|
|
||||||
out(render(row))
|
|
||||||
}
|
|
||||||
out()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun renderAdded(row: SqlRow): String {
|
|
||||||
val path = row["path"] as String
|
|
||||||
val size = row["size"] as Int
|
|
||||||
val mtime = row["mtime"] as Int
|
|
||||||
" " + green("+") + " " + bold(path) + dim(" %12d B mtime %d"(size, mtime))
|
|
||||||
}
|
|
||||||
|
|
||||||
fun renderRemoved(row: SqlRow): String {
|
|
||||||
val path = row["path"] as String
|
|
||||||
val size = row["size"] as Int
|
|
||||||
val mtime = row["mtime"] as Int
|
|
||||||
" " + red("-") + " " + bold(path) + dim(" %12d B mtime %d"(size, mtime))
|
|
||||||
}
|
|
||||||
|
|
||||||
fun renderChanged(row: SqlRow): String {
|
|
||||||
val path = row["path"] as String
|
|
||||||
val oldSize = row["old_size"] as Int
|
|
||||||
val newSize = row["new_size"] as Int
|
|
||||||
val oldMtime = row["old_mtime"] as Int
|
|
||||||
val newMtime = row["new_mtime"] as Int
|
|
||||||
val sizeDelta = newSize - oldSize
|
|
||||||
val mtimeDelta = newMtime - oldMtime
|
|
||||||
|
|
||||||
" " + yellow("~") + " " + bold(path) +
|
|
||||||
dim(
|
|
||||||
" size %d -> %d (%s B), mtime %d -> %d (%s ms)"(
|
|
||||||
oldSize,
|
|
||||||
newSize,
|
|
||||||
signed(sizeDelta),
|
|
||||||
oldMtime,
|
|
||||||
newMtime,
|
|
||||||
signed(mtimeDelta)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun loadRows(tx: SqlTransaction, query: String): List<SqlRow> = tx.select(query).toList()
|
|
||||||
|
|
||||||
fun main() {
|
|
||||||
val argv: List<String> = []
|
|
||||||
for (raw in ARGV as List) {
|
|
||||||
argv.add(raw as String)
|
|
||||||
}
|
|
||||||
val options = parseArgs(argv)
|
|
||||||
if (options == null) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
val root = Path(options.rootText)
|
|
||||||
if (!root.exists()) {
|
|
||||||
printUsage("root does not exist: " + root)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (!root.isDirectory()) {
|
|
||||||
printUsage("root is not a directory: " + root)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
val dbFile = childPath(root, DB_FILE_NAME)
|
|
||||||
val dbWasCreated = !dbFile.exists()
|
|
||||||
val shouldUpdateSnapshot = dbWasCreated || options.updateSnapshot
|
|
||||||
|
|
||||||
printBanner(root, dbFile, dbWasCreated, shouldUpdateSnapshot)
|
|
||||||
|
|
||||||
val db = openSqlite(dbFile.toString())
|
|
||||||
|
|
||||||
db.transaction { tx ->
|
|
||||||
tx.execute(CREATE_FILE_INDEX_SQL)
|
|
||||||
|
|
||||||
tx.execute("drop table if exists temp.current_scan")
|
|
||||||
tx.execute(CREATE_CURRENT_SCAN_SQL)
|
|
||||||
|
|
||||||
var scannedFiles = 0
|
|
||||||
for (rawEntry in root.glob("**")) {
|
|
||||||
val entry = rawEntry as Path
|
|
||||||
if (!entry.isFile()) continue
|
|
||||||
|
|
||||||
val relative = relativePath(root, entry)
|
|
||||||
if (isDatabaseArtifact(relative)) continue
|
|
||||||
|
|
||||||
val size = entry.size() ?: 0
|
|
||||||
val mtime = entry.modifiedAtMillis() ?: 0
|
|
||||||
tx.execute(INSERT_SCAN_ROW_SQL, relative, size, mtime)
|
|
||||||
scannedFiles++
|
|
||||||
}
|
|
||||||
|
|
||||||
val added = loadRows(tx, SELECT_ADDED_SQL)
|
|
||||||
val removed = loadRows(tx, SELECT_REMOVED_SQL)
|
|
||||||
val changed = loadRows(tx, SELECT_CHANGED_SQL)
|
|
||||||
|
|
||||||
val totalChanges = added.size + removed.size + changed.size
|
|
||||||
|
|
||||||
out(dim("scanned %d %s under %s"(scannedFiles, plural(scannedFiles, "file", "files"), root.toString())))
|
|
||||||
out(dim("detected %d %s"(totalChanges, plural(totalChanges, "change", "changes"))))
|
|
||||||
out()
|
|
||||||
|
|
||||||
printSection("Added", { green(it) }, added) { renderAdded(it) }
|
|
||||||
printSection("Removed", { red(it) }, removed) { renderRemoved(it) }
|
|
||||||
printSection("Changed", { yellow(it) }, changed) { renderChanged(it) }
|
|
||||||
|
|
||||||
if (shouldUpdateSnapshot) {
|
|
||||||
tx.execute(DELETE_MISSING_SQL)
|
|
||||||
tx.execute(UPSERT_SCAN_SQL)
|
|
||||||
|
|
||||||
val action = if (dbWasCreated) "created" else "updated"
|
|
||||||
out(cyan("snapshot " + action + " in " + dbFile.name))
|
|
||||||
} else {
|
|
||||||
out(dim("snapshot unchanged; re-run with -u or --update to persist the scan"))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main()
|
|
||||||
@ -1,23 +0,0 @@
|
|||||||
#!/env/bin lyng
|
|
||||||
|
|
||||||
import lyng.io.http
|
|
||||||
|
|
||||||
// Step 1: download the main lynglang.com page.
|
|
||||||
val home = Http.get("https://lynglang.com").text()
|
|
||||||
|
|
||||||
// Step 2: find the version-script reference in the page HTML.
|
|
||||||
val jsRef = "src=\"([^\"]*lyng-version\\.js)\"".re.find(home)
|
|
||||||
require(jsRef != null, "lyng-version.js reference not found on the homepage")
|
|
||||||
|
|
||||||
// Step 3: extract the referenced script path from the first regex capture.
|
|
||||||
val versionJsPath = jsRef[1]
|
|
||||||
|
|
||||||
// Step 4: download the script that exposes `window.LYNG_VERSION`.
|
|
||||||
val versionJs = Http.get("https://lynglang.com/" + versionJsPath).text()
|
|
||||||
|
|
||||||
// Step 5: pull the actual version string from the JavaScript source.
|
|
||||||
val versionMatch = "LYNG_VERSION\\s*=\\s*\"([^\"]+)\"".re.find(versionJs)
|
|
||||||
require(versionMatch != null, "LYNG_VERSION assignment not found")
|
|
||||||
|
|
||||||
// Step 6: print the discovered version for the user.
|
|
||||||
println("Lynglang.com version: " + ((versionMatch as RegexMatch)[1]))
|
|
||||||
@ -1,15 +0,0 @@
|
|||||||
import lyng.time
|
|
||||||
|
|
||||||
val n = 700_000
|
|
||||||
|
|
||||||
fun tm<T>(block: ()->T): T {
|
|
||||||
val t = Instant()
|
|
||||||
block().also {
|
|
||||||
println("tm: ${Instant() - t}")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val x = tm { List.fill(n) { it * 10 + 1 } }
|
|
||||||
val y = tm { List.fill(n, n + 10) { it * 10 + 1 } }
|
|
||||||
tm { x.add(-1) }
|
|
||||||
tm { y.add(-2) }
|
|
||||||
@ -1,76 +0,0 @@
|
|||||||
fun calculateDepth(
|
|
||||||
T: Real,
|
|
||||||
m: Real,
|
|
||||||
d: Real,
|
|
||||||
rho: Real = 1.2,
|
|
||||||
c: Real = 340.0,
|
|
||||||
g: Real = 9.81,
|
|
||||||
Cd: Real = 0.5,
|
|
||||||
eps: Real = 1e-3,
|
|
||||||
maxIter: Int = 100
|
|
||||||
): Real? {
|
|
||||||
// Площадь миделя
|
|
||||||
val r = d / 2.0
|
|
||||||
val A = π * r * r
|
|
||||||
|
|
||||||
// Коэффициент сопротивления
|
|
||||||
val k = 0.5 * Cd * rho * A
|
|
||||||
|
|
||||||
// Предельная скорость
|
|
||||||
val vTerm = sqrt(m * g / k)
|
|
||||||
|
|
||||||
// Функция времени падения с высоты h
|
|
||||||
fun tFall(h: Real): Real {
|
|
||||||
// Для численной стабильности при больших h используем логарифмическую форму
|
|
||||||
val arg = exp(g * h / (vTerm * vTerm))
|
|
||||||
// arcosh(x) = ln(x + sqrt(x^2 - 1))
|
|
||||||
val arcosh = ln(arg + sqrt(arg * arg - 1.0))
|
|
||||||
return vTerm / g * arcosh
|
|
||||||
}
|
|
||||||
|
|
||||||
// Полное расчётное время
|
|
||||||
fun Tcalc(h: Real): Real = tFall(h) + h / c
|
|
||||||
|
|
||||||
// Находим интервал, содержащий корень
|
|
||||||
// Нижняя граница: глубина не может быть отрицательной
|
|
||||||
var lo = 0.0
|
|
||||||
// Верхняя граница: сначала попробуем оценку по свободному падению (без звука)
|
|
||||||
var hi = 0.5 * g * T * T // максимальная глубина, если бы не было сопротивления и звука
|
|
||||||
// Уточним hi, чтобы Tcalc(hi) было заведомо больше T
|
|
||||||
while (Tcalc(hi) < T && hi < 1e4) {
|
|
||||||
hi *= 2.0
|
|
||||||
}
|
|
||||||
// Проверка, что hi достаточно велико
|
|
||||||
if (Tcalc(hi) < T) return null // слишком большая глубина, не укладываемся в разумное
|
|
||||||
|
|
||||||
// Бисекция
|
|
||||||
var iter = 0
|
|
||||||
var h = (lo + hi) / 2.0
|
|
||||||
while (iter < maxIter && (hi - lo) > eps) {
|
|
||||||
val f = Tcalc(h) - T
|
|
||||||
if (abs(f) < eps) break
|
|
||||||
if (f > 0) {
|
|
||||||
hi = h
|
|
||||||
} else {
|
|
||||||
lo = h
|
|
||||||
}
|
|
||||||
h = (lo + hi) / 2.0
|
|
||||||
iter++
|
|
||||||
}
|
|
||||||
|
|
||||||
return h
|
|
||||||
}
|
|
||||||
|
|
||||||
// Пример: T=12 секунд
|
|
||||||
val T = 26.0
|
|
||||||
val m = 1.0 // кг
|
|
||||||
val d = 0.1 // м
|
|
||||||
|
|
||||||
val depth = calculateDepth(T, m, d)
|
|
||||||
if (depth != null) {
|
|
||||||
println("Глубина: %.2f м"(depth))
|
|
||||||
// Для проверки выведем теоретическое время при найденной глубине
|
|
||||||
// (можно добавить функцию для самопроверки)
|
|
||||||
} else {
|
|
||||||
println("Расчёт не сошёлся")
|
|
||||||
}
|
|
||||||
@ -1,43 +0,0 @@
|
|||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.jdbc
|
|
||||||
|
|
||||||
println("H2 JDBC demo: typed open, generic open, generated keys")
|
|
||||||
|
|
||||||
val db = openH2("mem:lyng_h2_demo;DB_CLOSE_DELAY=-1")
|
|
||||||
|
|
||||||
db.transaction { tx ->
|
|
||||||
tx.execute("create table if not exists person(id bigint auto_increment primary key, name varchar(120) not null, active boolean not null)")
|
|
||||||
tx.execute("delete from person")
|
|
||||||
|
|
||||||
val firstInsert = tx.execute(
|
|
||||||
"insert into person(name, active) values(?, ?)",
|
|
||||||
"Ada",
|
|
||||||
true
|
|
||||||
)
|
|
||||||
val firstId = firstInsert.getGeneratedKeys().toList()[0][0]
|
|
||||||
assertEquals(1, firstId)
|
|
||||||
|
|
||||||
tx.execute(
|
|
||||||
"insert into person(name, active) values(?, ?)",
|
|
||||||
"Linus",
|
|
||||||
false
|
|
||||||
)
|
|
||||||
|
|
||||||
val rows = tx.select("select id, name, active from person order by id").toList()
|
|
||||||
assertEquals(2, rows.size)
|
|
||||||
println("#" + rows[0]["id"] + " " + rows[0]["name"] + " active=" + rows[0]["active"])
|
|
||||||
println("#" + rows[1]["id"] + " " + rows[1]["name"] + " active=" + rows[1]["active"])
|
|
||||||
}
|
|
||||||
|
|
||||||
val genericDb = openDatabase(
|
|
||||||
"jdbc:h2:mem:lyng_h2_generic;DB_CLOSE_DELAY=-1",
|
|
||||||
Map()
|
|
||||||
)
|
|
||||||
|
|
||||||
val answer = genericDb.transaction { tx ->
|
|
||||||
tx.select("select 42 as answer").toList()[0]["answer"]
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(42, answer)
|
|
||||||
println("Generic JDBC openDatabase(...) also works: answer=$answer")
|
|
||||||
println("OK")
|
|
||||||
@ -1,19 +0,0 @@
|
|||||||
import lyng.io.http.server
|
|
||||||
|
|
||||||
closed class CreateUserRequest(name: String, age: Int)
|
|
||||||
closed class CreateUserResponse(id: Int, name: String, age: Int)
|
|
||||||
|
|
||||||
val server = HttpServer()
|
|
||||||
|
|
||||||
server.postPath("/api/users") {
|
|
||||||
val req = jsonBody<CreateUserRequest>()
|
|
||||||
|
|
||||||
if (req.name.isBlank()) {
|
|
||||||
respondJson({ error: "name must not be empty" }, 400)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
respondJson(CreateUserResponse(101, req.name, req.age), 201)
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(8080, "127.0.0.1")
|
|
||||||
@ -1,67 +0,0 @@
|
|||||||
import lyng.time
|
|
||||||
|
|
||||||
val WORK_SIZE = 500
|
|
||||||
val THREADS = 1
|
|
||||||
|
|
||||||
fn piSpigot(iThread: Int, n: Int) {
|
|
||||||
var piIter = 0
|
|
||||||
var pi = List.fill(n) { 0 }
|
|
||||||
val boxes = n * 10 / 3
|
|
||||||
var reminders = List.fill(boxes) { 2 }
|
|
||||||
var heldDigits = 0
|
|
||||||
for (i in 0..<n) {
|
|
||||||
var carriedOver = 0
|
|
||||||
var sum = 0
|
|
||||||
for (j in (boxes - 1) downTo 0) {
|
|
||||||
val denom = j * 2 + 1
|
|
||||||
reminders[j] *= 10
|
|
||||||
sum = reminders[j] + carriedOver
|
|
||||||
val quotient = sum / denom
|
|
||||||
reminders[j] = sum % denom
|
|
||||||
carriedOver = quotient * j
|
|
||||||
}
|
|
||||||
reminders[0] = sum % 10
|
|
||||||
var q = sum / 10
|
|
||||||
if (q == 9) {
|
|
||||||
++heldDigits
|
|
||||||
} else if (q == 10) {
|
|
||||||
q = 0
|
|
||||||
for (k in 1..heldDigits) {
|
|
||||||
var replaced = pi[i - k]
|
|
||||||
if (replaced == 9) {
|
|
||||||
replaced = 0
|
|
||||||
} else {
|
|
||||||
++replaced
|
|
||||||
}
|
|
||||||
pi[i - k] = replaced
|
|
||||||
}
|
|
||||||
heldDigits = 1
|
|
||||||
} else {
|
|
||||||
heldDigits = 1
|
|
||||||
}
|
|
||||||
pi[piIter] = q
|
|
||||||
++piIter
|
|
||||||
}
|
|
||||||
|
|
||||||
var res = ""
|
|
||||||
for (i in (n - 8)..<n) {
|
|
||||||
res += pi[i]
|
|
||||||
}
|
|
||||||
|
|
||||||
println(iThread.toString() + ": " + res)
|
|
||||||
res
|
|
||||||
}
|
|
||||||
|
|
||||||
for( r in 0..100 ) {
|
|
||||||
val t0 = Instant()
|
|
||||||
|
|
||||||
println("piBench (lyng): THREADS = " + THREADS + ", WORK_SIZE = " + WORK_SIZE)
|
|
||||||
for (i in 0..<THREADS) {
|
|
||||||
piSpigot(i, WORK_SIZE)
|
|
||||||
}
|
|
||||||
|
|
||||||
val dt = Instant() - t0
|
|
||||||
|
|
||||||
println("all done, dt = ", dt)
|
|
||||||
delay(800)
|
|
||||||
}
|
|
||||||
@ -1,83 +0,0 @@
|
|||||||
# Copyright 2026 Sergey S. Chernov real.sergeych@gmail.com
|
|
||||||
#
|
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
# you may not use this file except in compliance with the License.
|
|
||||||
# You may obtain a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
# See the License for the specific language governing permissions and
|
|
||||||
# limitations under the License.
|
|
||||||
#
|
|
||||||
|
|
||||||
import time
|
|
||||||
from multiprocessing import Process
|
|
||||||
|
|
||||||
def piSpigot(iThread, nx):
|
|
||||||
piIter = 0
|
|
||||||
pi = [None] * nx
|
|
||||||
boxes = nx * 10 // 3
|
|
||||||
reminders = [None]*boxes
|
|
||||||
i = 0
|
|
||||||
while i < boxes:
|
|
||||||
reminders[i] = 2
|
|
||||||
i += 1
|
|
||||||
heldDigits = 0
|
|
||||||
i = 0
|
|
||||||
while i < nx:
|
|
||||||
carriedOver = 0
|
|
||||||
sum = 0
|
|
||||||
j = boxes - 1
|
|
||||||
while j >= 0:
|
|
||||||
reminders[j] *= 10
|
|
||||||
sum = reminders[j] + carriedOver
|
|
||||||
quotient = sum // (j * 2 + 1)
|
|
||||||
reminders[j] = sum % (j * 2 + 1)
|
|
||||||
carriedOver = quotient * j
|
|
||||||
j -= 1
|
|
||||||
reminders[0] = sum % 10
|
|
||||||
q = sum // 10
|
|
||||||
if q == 9:
|
|
||||||
heldDigits += 1
|
|
||||||
elif q == 10:
|
|
||||||
q = 0
|
|
||||||
k = 1
|
|
||||||
while k <= heldDigits:
|
|
||||||
replaced = pi[i - k]
|
|
||||||
if replaced == 9:
|
|
||||||
replaced = 0
|
|
||||||
else:
|
|
||||||
replaced += 1
|
|
||||||
pi[i - k] = replaced
|
|
||||||
k += 1
|
|
||||||
heldDigits = 1
|
|
||||||
else:
|
|
||||||
heldDigits = 1
|
|
||||||
pi[piIter] = q
|
|
||||||
piIter += 1
|
|
||||||
i += 1
|
|
||||||
res = ""
|
|
||||||
for i in range(len(pi)-8, len(pi), 1):
|
|
||||||
res += str(pi[i])
|
|
||||||
print(str(iThread) + ": " + res)
|
|
||||||
|
|
||||||
def createProcesses():
|
|
||||||
THREADS = 1
|
|
||||||
WORK_SIZE = 500
|
|
||||||
print("piBench (python3): THREADS = " + str(THREADS) + ", WORK_SIZE = " + str(WORK_SIZE))
|
|
||||||
pa = []
|
|
||||||
for i in range(THREADS):
|
|
||||||
p = Process(target=piSpigot, args=(i, WORK_SIZE))
|
|
||||||
p.start()
|
|
||||||
pa.append(p)
|
|
||||||
for p in pa:
|
|
||||||
p.join()
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
t1 = time.time()
|
|
||||||
createProcesses()
|
|
||||||
dt = time.time() - t1
|
|
||||||
print("total time: %i ms" % (dt*1000))
|
|
||||||
@ -1,71 +0,0 @@
|
|||||||
import lyng.io.db.jdbc
|
|
||||||
|
|
||||||
/*
|
|
||||||
PostgreSQL JDBC demo.
|
|
||||||
|
|
||||||
Usage:
|
|
||||||
lyng examples/postgres_basic.lyng [jdbc-url] [user] [password]
|
|
||||||
|
|
||||||
Typical local URL:
|
|
||||||
jdbc:postgresql://127.0.0.1/postgres
|
|
||||||
*/
|
|
||||||
|
|
||||||
fun cliArgs(): List<String> {
|
|
||||||
val result: List<String> = []
|
|
||||||
for (raw in ARGV as List) {
|
|
||||||
result.add(raw as String)
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
val argv = cliArgs()
|
|
||||||
val URL = if (argv.size > 0) argv[0] else "jdbc:postgresql://127.0.0.1/postgres"
|
|
||||||
val USER = if (argv.size > 1) argv[1] else ""
|
|
||||||
val PASSWORD = if (argv.size > 2) argv[2] else ""
|
|
||||||
|
|
||||||
println("PostgreSQL JDBC demo: typed open, generated keys, nested transaction")
|
|
||||||
|
|
||||||
val db = openPostgres(URL, USER, PASSWORD)
|
|
||||||
|
|
||||||
db.transaction { tx ->
|
|
||||||
tx.execute("create table if not exists lyng_pg_demo(id bigserial primary key, title text not null, done boolean not null)")
|
|
||||||
tx.execute("delete from lyng_pg_demo")
|
|
||||||
|
|
||||||
val firstInsert = tx.execute(
|
|
||||||
"insert into lyng_pg_demo(title, done) values(?, ?)",
|
|
||||||
"Verify PostgreSQL JDBC support",
|
|
||||||
false
|
|
||||||
)
|
|
||||||
val firstId = firstInsert.getGeneratedKeys().toList()[0][0]
|
|
||||||
println("First generated id=" + firstId)
|
|
||||||
|
|
||||||
tx.execute(
|
|
||||||
"insert into lyng_pg_demo(title, done) values(?, ?)",
|
|
||||||
"Review documentation",
|
|
||||||
true
|
|
||||||
)
|
|
||||||
|
|
||||||
try {
|
|
||||||
tx.transaction { inner ->
|
|
||||||
inner.execute(
|
|
||||||
"insert into lyng_pg_demo(title, done) values(?, ?)",
|
|
||||||
"This row is rolled back",
|
|
||||||
false
|
|
||||||
)
|
|
||||||
throw IllegalStateException("rollback nested")
|
|
||||||
}
|
|
||||||
} catch (_: IllegalStateException) {
|
|
||||||
println("Nested transaction rolled back as expected")
|
|
||||||
}
|
|
||||||
|
|
||||||
val rows = tx.select("select id, title, done from lyng_pg_demo order by id").toList()
|
|
||||||
for (row in rows) {
|
|
||||||
println("#" + row["id"] + " " + row["title"] + " done=" + row["done"])
|
|
||||||
}
|
|
||||||
|
|
||||||
val count = tx.select("select count(*) as count from lyng_pg_demo").toList()[0]["count"]
|
|
||||||
assertEquals(2, count)
|
|
||||||
println("Visible rows after nested rollback: " + count)
|
|
||||||
}
|
|
||||||
|
|
||||||
println("OK")
|
|
||||||
@ -1,89 +0,0 @@
|
|||||||
import lyng.io.db
|
|
||||||
import lyng.io.db.sqlite
|
|
||||||
import lyng.time
|
|
||||||
|
|
||||||
println("SQLite demo: typed open, generic open, result sets, generated keys, nested rollback")
|
|
||||||
|
|
||||||
// The typed helper is the simplest entry point when you know you want SQLite.
|
|
||||||
val db = openSqlite(":memory:")
|
|
||||||
|
|
||||||
db.transaction { tx ->
|
|
||||||
// Keep schema creation and data changes inside one transaction block.
|
|
||||||
tx.execute("create table task(id integer primary key autoincrement, title text not null, done integer not null, due_date date not null)")
|
|
||||||
|
|
||||||
// execute(...) is for side-effect statements. Generated keys are read from
|
|
||||||
// ExecutionResult rather than from a synthetic row-returning INSERT.
|
|
||||||
val firstInsert = tx.execute(
|
|
||||||
"insert into task(title, done, due_date) values(?, ?, ?)",
|
|
||||||
"Write a SQLite example",
|
|
||||||
false,
|
|
||||||
Date(2026, 4, 15)
|
|
||||||
)
|
|
||||||
val firstGeneratedKeys = firstInsert.getGeneratedKeys()
|
|
||||||
val firstId = firstGeneratedKeys.toList()[0][0]
|
|
||||||
assertEquals(1, firstId)
|
|
||||||
|
|
||||||
tx.execute(
|
|
||||||
"insert into task(title, done, due_date) values(?, ?, ?)",
|
|
||||||
"Review the DB API",
|
|
||||||
true,
|
|
||||||
Date(2026, 4, 16)
|
|
||||||
)
|
|
||||||
|
|
||||||
// Nested transactions are real savepoints. If the inner block fails,
|
|
||||||
// only the nested work is rolled back.
|
|
||||||
try {
|
|
||||||
tx.transaction { inner ->
|
|
||||||
inner.execute(
|
|
||||||
"insert into task(title, done, due_date) values(?, ?, ?)",
|
|
||||||
"This row is rolled back",
|
|
||||||
false,
|
|
||||||
Date(2026, 4, 17)
|
|
||||||
)
|
|
||||||
throw IllegalStateException("demonstrate nested rollback")
|
|
||||||
}
|
|
||||||
} catch (_: IllegalStateException) {
|
|
||||||
println("Nested transaction rolled back as expected")
|
|
||||||
}
|
|
||||||
|
|
||||||
// select(...) is for row-producing statements. ResultSet exposes metadata,
|
|
||||||
// cheap emptiness checks, iteration, and conversion to a plain list.
|
|
||||||
val tasks = tx.select("select id, title, done, due_date from task order by id")
|
|
||||||
assertEquals(false, tasks.isEmpty())
|
|
||||||
assertEquals(2, tasks.size())
|
|
||||||
|
|
||||||
println("Columns:")
|
|
||||||
for (column in tasks.columns) {
|
|
||||||
println(" " + column.name + " -> " + column.sqlType + " (native " + column.nativeType + ")")
|
|
||||||
}
|
|
||||||
|
|
||||||
val taskRows = tasks.toList()
|
|
||||||
|
|
||||||
println("Rows:")
|
|
||||||
for (row in taskRows) {
|
|
||||||
// Name lookups are case-insensitive and values are already converted.
|
|
||||||
println(" #" + row["ID"] + " " + row["title"] + " done=" + row["done"] + " due=" + row["due_date"])
|
|
||||||
}
|
|
||||||
|
|
||||||
// toList() materializes detached rows that stay usable after transaction close.
|
|
||||||
val snapshot = tx.select("select title, due_date from task order by id").toList()
|
|
||||||
assertEquals("Write a SQLite example", snapshot[0]["title"])
|
|
||||||
assertEquals(Date(2026, 4, 16), snapshot[1]["due_date"])
|
|
||||||
|
|
||||||
val count = tx.select("select count(*) as count from task").toList()[0]["count"]
|
|
||||||
assertEquals(2, count)
|
|
||||||
println("Visible rows after nested rollback: $count")
|
|
||||||
}
|
|
||||||
|
|
||||||
// The generic entry point stays useful for config-driven code.
|
|
||||||
val genericDb = openDatabase(
|
|
||||||
"sqlite::memory:",{ foreignKeys: true, busyTimeoutMillis: 1000 }
|
|
||||||
)
|
|
||||||
|
|
||||||
val answer = genericDb.transaction { tx ->
|
|
||||||
tx.select("select 42 as answer").toList()[0]["answer"]
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(42, answer)
|
|
||||||
println("Generic openDatabase(...) also works: answer=$answer")
|
|
||||||
println("OK")
|
|
||||||
@ -1,56 +0,0 @@
|
|||||||
import lyng.io.db.sqlite
|
|
||||||
|
|
||||||
println("SQLite serialization demo: write-side projection and decodeAs<T>()")
|
|
||||||
|
|
||||||
class Payload(name: String, count: Int)
|
|
||||||
|
|
||||||
class Item(
|
|
||||||
id: Int,
|
|
||||||
title: String,
|
|
||||||
@DbJson meta: Payload,
|
|
||||||
@DbLynon state: Payload
|
|
||||||
) {
|
|
||||||
var note: String = ""
|
|
||||||
@DbExcept var cache: String = ""
|
|
||||||
}
|
|
||||||
|
|
||||||
val restored = openSqlite(":memory:").transaction { tx ->
|
|
||||||
tx.execute(
|
|
||||||
"create table item(id integer not null, title text not null, meta text not null, state blob not null, note text not null)"
|
|
||||||
)
|
|
||||||
|
|
||||||
val item = Item(1, "first", Payload("json", 10), Payload("bin", 20))
|
|
||||||
item.note = "created"
|
|
||||||
item.cache = "not stored"
|
|
||||||
|
|
||||||
tx.execute("insert into item(@cols(?1)) values(@vals(?1))", item)
|
|
||||||
|
|
||||||
item.title = "second"
|
|
||||||
item.meta = Payload("json2", 11)
|
|
||||||
item.state = Payload("bin2", 21)
|
|
||||||
item.note = "updated"
|
|
||||||
|
|
||||||
tx.execute(
|
|
||||||
"update item set @set(?1 except: \"id\") where id = ?2",
|
|
||||||
item,
|
|
||||||
item.id
|
|
||||||
)
|
|
||||||
|
|
||||||
val restored = tx.select("select * from item where id = ?", 1).decodeAs<Item>().first
|
|
||||||
|
|
||||||
assertEquals("second", restored.title)
|
|
||||||
assertEquals("json2", restored.meta.name)
|
|
||||||
assertEquals(11, restored.meta.count)
|
|
||||||
assertEquals("bin2", restored.state.name)
|
|
||||||
assertEquals(21, restored.state.count)
|
|
||||||
assertEquals("updated", restored.note)
|
|
||||||
restored
|
|
||||||
}
|
|
||||||
|
|
||||||
println("Restored item:")
|
|
||||||
println(" id=" + restored.id)
|
|
||||||
println(" title=" + restored.title)
|
|
||||||
println(" meta=" + restored.meta.name + "/" + restored.meta.count)
|
|
||||||
println(" state=" + restored.state.name + "/" + restored.state.count)
|
|
||||||
println(" note=" + restored.note)
|
|
||||||
println("OK")
|
|
||||||
@ -1,68 +0,0 @@
|
|||||||
import lyng.buffer
|
|
||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val host = "127.0.0.1"
|
|
||||||
val port = 8092
|
|
||||||
val N = 5
|
|
||||||
val server = Net.tcpListen(port, host)
|
|
||||||
println("start tcp server at $host:$port")
|
|
||||||
|
|
||||||
fun serveClient(client: TcpSocket) = launch {
|
|
||||||
try {
|
|
||||||
while (true) {
|
|
||||||
val data = client.read()
|
|
||||||
if (data == null) break
|
|
||||||
var line = (data as Buffer).decodeUtf8()
|
|
||||||
line = "[" + client.remoteAddress() + "]> " + line
|
|
||||||
println(line)
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
println("ERROR [reader]: " + e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun serveRequests(server: TcpServer) = launch {
|
|
||||||
val readers = []
|
|
||||||
try {
|
|
||||||
for (i in 0..<5) {
|
|
||||||
val client = server.accept()
|
|
||||||
println("accept new connection: " + client.remoteAddress())
|
|
||||||
readers.add(serveClient(client as TcpSocket))
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
println("ERROR [listener]: " + e)
|
|
||||||
} finally {
|
|
||||||
server.close()
|
|
||||||
}
|
|
||||||
for (i in 0..<readers.size) {
|
|
||||||
val reader = readers[i]
|
|
||||||
(reader as Deferred).await()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val srv = serveRequests(server as TcpServer)
|
|
||||||
|
|
||||||
var clients = []
|
|
||||||
for (i in 0..<N) {
|
|
||||||
//delay(500)
|
|
||||||
clients.add(launch {
|
|
||||||
try{
|
|
||||||
val socket = Net.tcpConnect(host, port)
|
|
||||||
socket.writeUtf8("ping1ping2ping3ping4ping5")
|
|
||||||
socket.flush()
|
|
||||||
socket.close()
|
|
||||||
} catch (e) {
|
|
||||||
println("ERROR [client]: " + e)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i in 0..<clients.size) {
|
|
||||||
val c = clients[i]
|
|
||||||
(c as Deferred).await()
|
|
||||||
println("client done")
|
|
||||||
}
|
|
||||||
|
|
||||||
srv.await()
|
|
||||||
delay(10000)
|
|
||||||
println("FIN")
|
|
||||||
@ -1,60 +0,0 @@
|
|||||||
import lyng.io.net
|
|
||||||
|
|
||||||
val host = "127.0.0.1"
|
|
||||||
val clientCount = 1000
|
|
||||||
val clientWindow = 128
|
|
||||||
val server = Net.tcpListen(0, host, clientWindow, true)
|
|
||||||
val port = server.localAddress().port
|
|
||||||
|
|
||||||
fun payloadFor(index: Int) = "$index:${Random.nextInt()}:${Random.nextInt()}"
|
|
||||||
|
|
||||||
val serverJob = launch {
|
|
||||||
try {
|
|
||||||
while (true) {
|
|
||||||
val client = server.accept()
|
|
||||||
launch {
|
|
||||||
try {
|
|
||||||
client.readLine()?.let { source ->
|
|
||||||
client.writeUtf8("pong: $source\n")
|
|
||||||
client.flush()
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
client.close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
if (server.isOpen()) {
|
|
||||||
throw e
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (server.isOpen()) {
|
|
||||||
server.close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var completed = 0
|
|
||||||
for (batchStart in 0..<clientCount step clientWindow) {
|
|
||||||
val batchEnd = if (batchStart + clientWindow < clientCount) batchStart + clientWindow else clientCount
|
|
||||||
val replies = (batchStart..<batchEnd).map { index ->
|
|
||||||
val payload = payloadFor(index)
|
|
||||||
launch {
|
|
||||||
val socket = Net.tcpConnect(host, port) as TcpSocket
|
|
||||||
try {
|
|
||||||
socket.writeUtf8(payload + "\n")
|
|
||||||
socket.flush()
|
|
||||||
val reply = socket.readLine()
|
|
||||||
assertEquals("pong: $payload", reply)
|
|
||||||
} finally {
|
|
||||||
socket.close()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}.joinAll()
|
|
||||||
completed += replies.size
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals(clientCount, completed)
|
|
||||||
server.close()
|
|
||||||
serverJob.await()
|
|
||||||
println("OK: $clientCount concurrent tcp clients")
|
|
||||||
@ -1,822 +0,0 @@
|
|||||||
#!/usr/bin/env lyng
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Lyng Console Tetris (interactive sample)
|
|
||||||
*
|
|
||||||
* Controls:
|
|
||||||
* - Left/Right arrows or A/D: move
|
|
||||||
* - Up arrow or W: rotate
|
|
||||||
* - Down arrow or S: soft drop
|
|
||||||
* - Space: hard drop
|
|
||||||
* - P or Escape: pause
|
|
||||||
* - Q: quit
|
|
||||||
|
|
||||||
Tsted to score:
|
|
||||||
sergeych@sergeych-XPS-17-9720:~$ ~/dev/lyng/examples/tetris_console.lyng
|
|
||||||
Bye.
|
|
||||||
Score: 435480
|
|
||||||
Lines: 271
|
|
||||||
Level: 28
|
|
||||||
Ssergeych@sergeych-XPS-17-9720:~$
|
|
||||||
*/
|
|
||||||
|
|
||||||
import lyng.io.console
|
|
||||||
import lyng.io.fs
|
|
||||||
|
|
||||||
val MIN_COLS = 56
|
|
||||||
val MIN_ROWS = 24
|
|
||||||
val PANEL_WIDTH = 24
|
|
||||||
val BOARD_MARGIN_ROWS = 5
|
|
||||||
val BOARD_MIN_W = 10
|
|
||||||
val BOARD_MAX_W = 16
|
|
||||||
val BOARD_MIN_H = 16
|
|
||||||
val BOARD_MAX_H = 28
|
|
||||||
val LEVEL_LINES_STEP = 10
|
|
||||||
val DROP_FRAMES_BASE = 15
|
|
||||||
val DROP_FRAMES_MIN = 3
|
|
||||||
val FRAME_DELAY_MS = 35
|
|
||||||
val RESIZE_WAIT_MS = 250
|
|
||||||
val MAX_PENDING_INPUTS = 64
|
|
||||||
val ROTATION_KICKS = [0, -1, 1, -2, 2]
|
|
||||||
val ANSI_ESC = "\u001b["
|
|
||||||
val ANSI_RESET = ANSI_ESC + "0m"
|
|
||||||
val ERROR_LOG_PATH = "/tmp/lyng_tetris_errors.log"
|
|
||||||
val UNICODE_BLOCK = "██"
|
|
||||||
val UNICODE_TOP_LEFT = "┌"
|
|
||||||
val UNICODE_TOP_RIGHT = "┐"
|
|
||||||
val UNICODE_BOTTOM_LEFT = "└"
|
|
||||||
val UNICODE_BOTTOM_RIGHT = "┘"
|
|
||||||
val UNICODE_HORIZONTAL = "──"
|
|
||||||
val UNICODE_VERTICAL = "│"
|
|
||||||
val UNICODE_DOT = "· "
|
|
||||||
val PIECES: List<Piece> = []
|
|
||||||
|
|
||||||
type Cell = List<Int>
|
|
||||||
type Rotation = List<Cell>
|
|
||||||
type Rotations = List<Rotation>
|
|
||||||
type Row = List<Int>
|
|
||||||
type Board = List<Row>
|
|
||||||
|
|
||||||
class Piece(val name: String, val rotations: Rotations) {}
|
|
||||||
class RotateResult(val ok: Bool, val rot: Int, val px: Int) {}
|
|
||||||
class GameState(
|
|
||||||
pieceId0: Int,
|
|
||||||
nextId0: Int,
|
|
||||||
next2Id0: Int,
|
|
||||||
px0: Int,
|
|
||||||
py0: Int,
|
|
||||||
) {
|
|
||||||
var pieceId = pieceId0
|
|
||||||
var nextId = nextId0
|
|
||||||
var next2Id = next2Id0
|
|
||||||
var rot = 0
|
|
||||||
var px = px0
|
|
||||||
var py = py0
|
|
||||||
var score = 0
|
|
||||||
var totalLines = 0
|
|
||||||
var level = 1
|
|
||||||
var running = true
|
|
||||||
var gameOver = false
|
|
||||||
var paused = false
|
|
||||||
}
|
|
||||||
class LoopFrame(val resized: Bool, val originRow: Int, val originCol: Int) {}
|
|
||||||
class InputBuffer {
|
|
||||||
private val mutex: Mutex = Mutex()
|
|
||||||
private val items: List<String> = []
|
|
||||||
|
|
||||||
fun push(value: String): Void {
|
|
||||||
mutex.withLock {
|
|
||||||
if (items.size >= MAX_PENDING_INPUTS) {
|
|
||||||
items.removeAt(0)
|
|
||||||
}
|
|
||||||
items.add(value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun drain(): List<String> {
|
|
||||||
val out: List<String> = []
|
|
||||||
mutex.withLock {
|
|
||||||
while (items.size > 0) {
|
|
||||||
out.add(items[0])
|
|
||||||
items.removeAt(0)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
out
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun clearAndHome() {
|
|
||||||
Console.clear()
|
|
||||||
Console.home()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun logError(message: String, err: Object?): Void {
|
|
||||||
try {
|
|
||||||
var details = ""
|
|
||||||
if (err != null) {
|
|
||||||
try {
|
|
||||||
details = ": " + err
|
|
||||||
} catch (_: Object) {
|
|
||||||
details = ": <error-format-failed>"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Path(ERROR_LOG_PATH).appendUtf8(message + details + "\n")
|
|
||||||
} catch (_: Object) {
|
|
||||||
// Never let logging errors affect gameplay.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun repeatText(s: String, n: Int): String {
|
|
||||||
var out: String = ""
|
|
||||||
for (i in 0..<n) {
|
|
||||||
out += s
|
|
||||||
}
|
|
||||||
out
|
|
||||||
}
|
|
||||||
|
|
||||||
fun max<T>(a: T, b: T): T = if (a > b) a else b
|
|
||||||
|
|
||||||
fun emptyRow(width: Int): Row {
|
|
||||||
val r: Row = []
|
|
||||||
for (x in 0..<width) {
|
|
||||||
r.add(0)
|
|
||||||
}
|
|
||||||
r
|
|
||||||
}
|
|
||||||
|
|
||||||
fun createBoard(width: Int, height: Int): Board {
|
|
||||||
val b: Board = []
|
|
||||||
for (y in 0..<height) {
|
|
||||||
b.add(emptyRow(width))
|
|
||||||
}
|
|
||||||
b
|
|
||||||
}
|
|
||||||
|
|
||||||
fun colorize(text: String, sgr: String, useColor: Bool): String {
|
|
||||||
if (!useColor) return text
|
|
||||||
ANSI_ESC + sgr + "m" + text + ANSI_RESET
|
|
||||||
}
|
|
||||||
|
|
||||||
fun blockText(pieceId: Int, useColor: Bool): String {
|
|
||||||
if (pieceId <= 0) return " "
|
|
||||||
val sgr = when (pieceId) {
|
|
||||||
1 -> "36" // I
|
|
||||||
2 -> "33" // O
|
|
||||||
3 -> "35" // T
|
|
||||||
4 -> "32" // S
|
|
||||||
5 -> "31" // Z
|
|
||||||
6 -> "34" // J
|
|
||||||
7 -> "93" // L
|
|
||||||
else -> "37"
|
|
||||||
}
|
|
||||||
colorize(UNICODE_BLOCK, sgr, useColor)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun emptyCellText(useColor: Bool): String {
|
|
||||||
colorize(UNICODE_DOT, "90", useColor)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun canPlace(board: Board, boardW: Int, boardH: Int, pieceId: Int, rot: Int, px: Int, py: Int): Bool {
|
|
||||||
try {
|
|
||||||
if (pieceId < 1 || pieceId > 7) return false
|
|
||||||
val piece: Piece = PIECES[pieceId - 1]
|
|
||||||
if (rot < 0 || rot >= piece.rotations.size) return false
|
|
||||||
val cells = piece.rotations[rot]
|
|
||||||
|
|
||||||
for (cell in cells) {
|
|
||||||
val x = px + cell[0]
|
|
||||||
val y = py + cell[1]
|
|
||||||
|
|
||||||
if (x < 0 || x >= boardW) return false
|
|
||||||
if (y >= boardH) return false
|
|
||||||
|
|
||||||
if (y >= 0) {
|
|
||||||
if (y >= board.size) return false
|
|
||||||
val row = board[y]
|
|
||||||
if (row == null) return false
|
|
||||||
if (row[x] != 0) return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
true
|
|
||||||
} catch (_: Object) {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun lockPiece(board: Board, pieceId: Int, rot: Int, px: Int, py: Int): Void {
|
|
||||||
val piece: Piece = PIECES[pieceId - 1]
|
|
||||||
val cells = piece.rotations[rot]
|
|
||||||
|
|
||||||
for (cell in cells) {
|
|
||||||
val x = px + cell[0]
|
|
||||||
val y = py + cell[1]
|
|
||||||
|
|
||||||
if (y >= 0 && y < board.size) {
|
|
||||||
val row = board[y]
|
|
||||||
if (row != null) {
|
|
||||||
row[x] = pieceId
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun clearCompletedLines(board: Board, boardW: Int, boardH: Int): Int {
|
|
||||||
val b = board
|
|
||||||
var y = boardH - 1
|
|
||||||
var cleared = 0
|
|
||||||
|
|
||||||
while (y >= 0) {
|
|
||||||
if (y >= b.size) {
|
|
||||||
y--
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
val row = b[y]
|
|
||||||
if (row == null) {
|
|
||||||
b.removeAt(y)
|
|
||||||
b.insertAt(0, emptyRow(boardW))
|
|
||||||
cleared++
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
var full = true
|
|
||||||
for (x in 0..<boardW) {
|
|
||||||
if (row[x] == 0) {
|
|
||||||
full = false
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (full) {
|
|
||||||
b.removeAt(y)
|
|
||||||
b.insertAt(0, emptyRow(boardW))
|
|
||||||
cleared++
|
|
||||||
} else {
|
|
||||||
y--
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
cleared
|
|
||||||
}
|
|
||||||
|
|
||||||
fun activeCellId(pieceId: Int, rot: Int, px: Int, py: Int, x: Int, y: Int): Int {
|
|
||||||
val piece: Piece = PIECES[pieceId - 1]
|
|
||||||
val cells = piece.rotations[rot]
|
|
||||||
|
|
||||||
for (cell in cells) {
|
|
||||||
val ax = px + cell[0]
|
|
||||||
val ay = py + cell[1]
|
|
||||||
if (ax == x && ay == y) return pieceId
|
|
||||||
}
|
|
||||||
|
|
||||||
0
|
|
||||||
}
|
|
||||||
|
|
||||||
fun tryRotateCw(board: Board, boardW: Int, boardH: Int, pieceId: Int, rot: Int, px: Int, py: Int): RotateResult {
|
|
||||||
val piece: Piece = PIECES[pieceId - 1]
|
|
||||||
val rotations = piece.rotations.size
|
|
||||||
val nr = (rot + 1) % rotations
|
|
||||||
for (kx in ROTATION_KICKS) {
|
|
||||||
val nx = px + kx
|
|
||||||
if (canPlace(board, boardW, boardH, pieceId, nr, nx, py) == true) {
|
|
||||||
return RotateResult(true, nr, nx)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
RotateResult(false, rot, px)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun nextPreviewLines(pieceId: Int, useColor: Bool): List<String> {
|
|
||||||
val out: List<String> = []
|
|
||||||
if (pieceId <= 0) {
|
|
||||||
for (i in 0..<4) {
|
|
||||||
out.add(" ")
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
val piece: Piece = PIECES[pieceId - 1]
|
|
||||||
val cells = piece.rotations[0]
|
|
||||||
|
|
||||||
for (y in 0..<4) {
|
|
||||||
var line = ""
|
|
||||||
for (x in 0..<4) {
|
|
||||||
var filled = false
|
|
||||||
for (cell in cells) {
|
|
||||||
if (cell[0] == x && cell[1] == y) {
|
|
||||||
filled = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
line += if (filled) blockText(pieceId, useColor) else " "
|
|
||||||
}
|
|
||||||
out.add(line)
|
|
||||||
}
|
|
||||||
out
|
|
||||||
}
|
|
||||||
|
|
||||||
fun render(
|
|
||||||
state: GameState,
|
|
||||||
board: Board,
|
|
||||||
boardW: Int,
|
|
||||||
boardH: Int,
|
|
||||||
prevFrameLines: List<String>,
|
|
||||||
originRow: Int,
|
|
||||||
originCol: Int,
|
|
||||||
useColor: Bool,
|
|
||||||
): List<String> {
|
|
||||||
val bottomBorder = UNICODE_BOTTOM_LEFT + repeatText(UNICODE_HORIZONTAL, boardW) + UNICODE_BOTTOM_RIGHT
|
|
||||||
|
|
||||||
val panel: List<String> = []
|
|
||||||
val nextPiece: Piece = PIECES[state.nextId - 1]
|
|
||||||
val next2Piece: Piece = PIECES[state.next2Id - 1]
|
|
||||||
val nextName = nextPiece.name
|
|
||||||
val next2Name = next2Piece.name
|
|
||||||
val preview = nextPreviewLines(state.nextId, useColor)
|
|
||||||
|
|
||||||
panel.add("Lyng Tetris")
|
|
||||||
panel.add("")
|
|
||||||
panel.add("Score: " + state.score)
|
|
||||||
panel.add("Lines: " + state.totalLines)
|
|
||||||
panel.add("Level: " + state.level)
|
|
||||||
panel.add("")
|
|
||||||
panel.add("Next: " + nextName)
|
|
||||||
panel.add("")
|
|
||||||
for (pl in preview) panel.add(pl)
|
|
||||||
panel.add("After: " + next2Name)
|
|
||||||
panel.add("")
|
|
||||||
panel.add("Keys:")
|
|
||||||
panel.add("A/D or arrows")
|
|
||||||
panel.add("W/Up: rotate")
|
|
||||||
panel.add("S/Down: drop")
|
|
||||||
panel.add("Space: hard drop")
|
|
||||||
panel.add("P/Esc: pause")
|
|
||||||
panel.add("Q: quit")
|
|
||||||
|
|
||||||
val frameLines: List<String> = []
|
|
||||||
|
|
||||||
for (y in 0..<boardH) {
|
|
||||||
var line = UNICODE_VERTICAL
|
|
||||||
|
|
||||||
for (x in 0..<boardW) {
|
|
||||||
val a = activeCellId(state.pieceId, state.rot, state.px, state.py, x, y)
|
|
||||||
val row = if (y < board.size) board[y] else null
|
|
||||||
val b = if (row == null) 0 else row[x]
|
|
||||||
val id = if (a > 0) a else b
|
|
||||||
line += if (id > 0) blockText(id, useColor) else emptyCellText(useColor)
|
|
||||||
}
|
|
||||||
|
|
||||||
line += UNICODE_VERTICAL
|
|
||||||
|
|
||||||
if (y < panel.size) {
|
|
||||||
line += " " + panel[y]
|
|
||||||
}
|
|
||||||
|
|
||||||
frameLines.add(line)
|
|
||||||
}
|
|
||||||
|
|
||||||
frameLines.add(bottomBorder)
|
|
||||||
|
|
||||||
for (i in 0..<frameLines.size) {
|
|
||||||
val line = frameLines[i]
|
|
||||||
val old = if (i < prevFrameLines.size) prevFrameLines[i] else null
|
|
||||||
if (old != line) {
|
|
||||||
Console.moveTo(originRow + i, originCol)
|
|
||||||
Console.clearLine()
|
|
||||||
Console.write(line)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
frameLines
|
|
||||||
}
|
|
||||||
|
|
||||||
fun fitLine(line: String, width: Int): String {
|
|
||||||
val maxLen = if (width > 0) width else 0
|
|
||||||
if (maxLen <= 0) return ""
|
|
||||||
if (line.size >= maxLen) return line[..<maxLen]
|
|
||||||
line + repeatText(" ", maxLen - line.size)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun renderPauseOverlay(
|
|
||||||
originRow: Int,
|
|
||||||
originCol: Int,
|
|
||||||
boardW: Int,
|
|
||||||
boardH: Int,
|
|
||||||
): Void {
|
|
||||||
val contentWidth = boardW * 2 + 2 + 3 + PANEL_WIDTH
|
|
||||||
val contentHeight = boardH + 1
|
|
||||||
|
|
||||||
val lines: List<String> = []
|
|
||||||
lines.add("PAUSED")
|
|
||||||
lines.add("")
|
|
||||||
lines.add("Any key: continue game")
|
|
||||||
lines.add("Esc: exit game")
|
|
||||||
lines.add("")
|
|
||||||
lines.add("Move: A/D or arrows")
|
|
||||||
lines.add("Rotate: W or Up")
|
|
||||||
lines.add("Drop: S/Down, Space hard drop")
|
|
||||||
|
|
||||||
var innerWidth = 0
|
|
||||||
for (line in lines) {
|
|
||||||
if (line.size > innerWidth) innerWidth = line.size
|
|
||||||
}
|
|
||||||
innerWidth += 4
|
|
||||||
val maxInner = max(12, contentWidth - 2)
|
|
||||||
if (innerWidth > maxInner) innerWidth = maxInner
|
|
||||||
if (innerWidth % 2 != 0) innerWidth--
|
|
||||||
|
|
||||||
val boxWidth = innerWidth + 2
|
|
||||||
val boxHeight = lines.size + 2
|
|
||||||
|
|
||||||
val left = originCol + max(0, (contentWidth - boxWidth) / 2)
|
|
||||||
val top = originRow + max(0, (contentHeight - boxHeight) / 2)
|
|
||||||
|
|
||||||
val topBorder = UNICODE_TOP_LEFT + repeatText(UNICODE_HORIZONTAL, innerWidth / 2) + UNICODE_TOP_RIGHT
|
|
||||||
val bottomBorder = UNICODE_BOTTOM_LEFT + repeatText(UNICODE_HORIZONTAL, innerWidth / 2) + UNICODE_BOTTOM_RIGHT
|
|
||||||
|
|
||||||
Console.moveTo(top, left)
|
|
||||||
Console.write(topBorder)
|
|
||||||
|
|
||||||
for (i in 0..<lines.size) {
|
|
||||||
Console.moveTo(top + 1 + i, left)
|
|
||||||
Console.write(UNICODE_VERTICAL + fitLine(" " + lines[i], innerWidth) + UNICODE_VERTICAL)
|
|
||||||
}
|
|
||||||
|
|
||||||
Console.moveTo(top + boxHeight - 1, left)
|
|
||||||
Console.write(bottomBorder)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun waitForMinimumSize(minCols: Int, minRows: Int): Object {
|
|
||||||
while (true) {
|
|
||||||
val g = Console.geometry()
|
|
||||||
val cols = g?.columns ?: 0
|
|
||||||
val rows = g?.rows ?: 0
|
|
||||||
|
|
||||||
if (cols >= minCols && rows >= minRows) return g
|
|
||||||
|
|
||||||
clearAndHome()
|
|
||||||
val lines: List<String> = []
|
|
||||||
lines.add("Lyng Tetris needs at least %sx%s terminal size."(minCols, minRows))
|
|
||||||
lines.add("Current: %sx%s"(cols, rows))
|
|
||||||
lines.add("Resize the console window to continue.")
|
|
||||||
|
|
||||||
val visibleLines = if (rows < lines.size) rows else lines.size
|
|
||||||
if (visibleLines > 0) {
|
|
||||||
val startRow = max(1, ((rows - visibleLines) / 2) + 1)
|
|
||||||
for (i in 0..<visibleLines) {
|
|
||||||
val line = lines[i]
|
|
||||||
val startCol = max(1, ((cols - line.size) / 2) + 1)
|
|
||||||
Console.moveTo(startRow + i, startCol)
|
|
||||||
Console.clearLine()
|
|
||||||
Console.write(line)
|
|
||||||
}
|
|
||||||
Console.flush()
|
|
||||||
}
|
|
||||||
delay(RESIZE_WAIT_MS)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun scoreForLines(cleared: Int, level: Int): Int {
|
|
||||||
when (cleared) {
|
|
||||||
1 -> 100 * level
|
|
||||||
2 -> 300 * level
|
|
||||||
3 -> 500 * level
|
|
||||||
4 -> 800 * level
|
|
||||||
else -> 0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Classic 7 tetrominoes, minimal rotations per piece.
|
|
||||||
fun cell(x: Int, y: Int): Cell { [x, y] }
|
|
||||||
fun rot(a: Cell, b: Cell, c: Cell, d: Cell): Rotation {
|
|
||||||
val r: Rotation = []
|
|
||||||
r.add(a)
|
|
||||||
r.add(b)
|
|
||||||
r.add(c)
|
|
||||||
r.add(d)
|
|
||||||
r
|
|
||||||
}
|
|
||||||
|
|
||||||
val iRots: Rotations = []
|
|
||||||
iRots.add(rot(cell(0,1), cell(1,1), cell(2,1), cell(3,1)))
|
|
||||||
iRots.add(rot(cell(2,0), cell(2,1), cell(2,2), cell(2,3)))
|
|
||||||
PIECES.add(Piece("I", iRots))
|
|
||||||
|
|
||||||
val oRots: Rotations = []
|
|
||||||
oRots.add(rot(cell(1,0), cell(2,0), cell(1,1), cell(2,1)))
|
|
||||||
PIECES.add(Piece("O", oRots))
|
|
||||||
|
|
||||||
val tRots: Rotations = []
|
|
||||||
tRots.add(rot(cell(1,0), cell(0,1), cell(1,1), cell(2,1)))
|
|
||||||
tRots.add(rot(cell(1,0), cell(1,1), cell(2,1), cell(1,2)))
|
|
||||||
tRots.add(rot(cell(0,1), cell(1,1), cell(2,1), cell(1,2)))
|
|
||||||
tRots.add(rot(cell(1,0), cell(0,1), cell(1,1), cell(1,2)))
|
|
||||||
PIECES.add(Piece("T", tRots))
|
|
||||||
|
|
||||||
val sRots: Rotations = []
|
|
||||||
sRots.add(rot(cell(1,0), cell(2,0), cell(0,1), cell(1,1)))
|
|
||||||
sRots.add(rot(cell(1,0), cell(1,1), cell(2,1), cell(2,2)))
|
|
||||||
PIECES.add(Piece("S", sRots))
|
|
||||||
|
|
||||||
val zRots: Rotations = []
|
|
||||||
zRots.add(rot(cell(0,0), cell(1,0), cell(1,1), cell(2,1)))
|
|
||||||
zRots.add(rot(cell(2,0), cell(1,1), cell(2,1), cell(1,2)))
|
|
||||||
PIECES.add(Piece("Z", zRots))
|
|
||||||
|
|
||||||
val jRots: Rotations = []
|
|
||||||
jRots.add(rot(cell(0,0), cell(0,1), cell(1,1), cell(2,1)))
|
|
||||||
jRots.add(rot(cell(1,0), cell(2,0), cell(1,1), cell(1,2)))
|
|
||||||
jRots.add(rot(cell(0,1), cell(1,1), cell(2,1), cell(2,2)))
|
|
||||||
jRots.add(rot(cell(1,0), cell(1,1), cell(0,2), cell(1,2)))
|
|
||||||
PIECES.add(Piece("J", jRots))
|
|
||||||
|
|
||||||
val lRots: Rotations = []
|
|
||||||
lRots.add(rot(cell(2,0), cell(0,1), cell(1,1), cell(2,1)))
|
|
||||||
lRots.add(rot(cell(1,0), cell(1,1), cell(1,2), cell(2,2)))
|
|
||||||
lRots.add(rot(cell(0,1), cell(1,1), cell(2,1), cell(0,2)))
|
|
||||||
lRots.add(rot(cell(0,0), cell(1,0), cell(1,1), cell(1,2)))
|
|
||||||
PIECES.add(Piece("L", lRots))
|
|
||||||
|
|
||||||
if (!Console.isSupported()) {
|
|
||||||
println("Console API is not supported in this runtime.")
|
|
||||||
void
|
|
||||||
} else if (!Console.isTty()) {
|
|
||||||
println("This sample needs an interactive terminal (TTY).")
|
|
||||||
void
|
|
||||||
} else {
|
|
||||||
waitForMinimumSize(MIN_COLS, MIN_ROWS)
|
|
||||||
|
|
||||||
val g0 = Console.geometry()
|
|
||||||
val cols = g0?.columns ?: MIN_COLS
|
|
||||||
val rows = g0?.rows ?: MIN_ROWS
|
|
||||||
|
|
||||||
val boardW = clamp((cols - PANEL_WIDTH) / 2, BOARD_MIN_W..BOARD_MAX_W)
|
|
||||||
val boardH = clamp(rows - BOARD_MARGIN_ROWS, BOARD_MIN_H..BOARD_MAX_H)
|
|
||||||
|
|
||||||
val board: Board = createBoard(boardW, boardH)
|
|
||||||
|
|
||||||
fun nextPieceId() {
|
|
||||||
Random.next(1..7)
|
|
||||||
}
|
|
||||||
|
|
||||||
val state: GameState = GameState(
|
|
||||||
nextPieceId(),
|
|
||||||
nextPieceId(),
|
|
||||||
nextPieceId(),
|
|
||||||
(boardW / 2) - 2,
|
|
||||||
-1,
|
|
||||||
)
|
|
||||||
var prevFrameLines: List<String> = []
|
|
||||||
|
|
||||||
var forceRedraw = false
|
|
||||||
val inputBuffer: InputBuffer = InputBuffer()
|
|
||||||
|
|
||||||
val rawModeEnabled = Console.setRawMode(true)
|
|
||||||
if (!rawModeEnabled) {
|
|
||||||
println("Raw keyboard mode is not available in this terminal/runtime.")
|
|
||||||
println("Use jlyng in an interactive terminal with raw input support.")
|
|
||||||
void
|
|
||||||
} else {
|
|
||||||
val useColor = Console.ansiLevel() != ConsoleAnsiLevel.NONE
|
|
||||||
Console.enterAltScreen()
|
|
||||||
Console.setCursorVisible(false)
|
|
||||||
clearAndHome()
|
|
||||||
|
|
||||||
fun resetActivePiece(s: GameState): Void {
|
|
||||||
s.pieceId = nextPieceId()
|
|
||||||
s.rot = 0
|
|
||||||
s.px = (boardW / 2) - 2
|
|
||||||
s.py = -1
|
|
||||||
}
|
|
||||||
|
|
||||||
fun applyKeyInput(s: GameState, key: String): Void {
|
|
||||||
try {
|
|
||||||
if (key == "__CTRL_C__") {
|
|
||||||
s.running = false
|
|
||||||
}
|
|
||||||
else if (s.paused) {
|
|
||||||
if (key == "Escape") {
|
|
||||||
s.running = false
|
|
||||||
} else {
|
|
||||||
s.paused = false
|
|
||||||
forceRedraw = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (key == "p" || key == "P" || key == "Escape") {
|
|
||||||
s.paused = true
|
|
||||||
forceRedraw = true
|
|
||||||
}
|
|
||||||
else if (key == "q" || key == "Q") {
|
|
||||||
s.running = false
|
|
||||||
}
|
|
||||||
else if (key == "ArrowLeft" || key == "a" || key == "A") {
|
|
||||||
if (canPlace(board, boardW, boardH, s.pieceId, s.rot, s.px - 1, s.py) == true) s.px--
|
|
||||||
}
|
|
||||||
else if (key == "ArrowRight" || key == "d" || key == "D") {
|
|
||||||
if (canPlace(board, boardW, boardH, s.pieceId, s.rot, s.px + 1, s.py) == true) s.px++
|
|
||||||
}
|
|
||||||
else if (key == "ArrowUp" || key == "w" || key == "W") {
|
|
||||||
val rr: RotateResult = tryRotateCw(board, boardW, boardH, s.pieceId, s.rot, s.px, s.py)
|
|
||||||
if (rr.ok) {
|
|
||||||
s.rot = rr.rot
|
|
||||||
s.px = rr.px
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (key == "ArrowDown" || key == "s" || key == "S") {
|
|
||||||
if (canPlace(board, boardW, boardH, s.pieceId, s.rot, s.px, s.py + 1) == true) {
|
|
||||||
s.py++
|
|
||||||
s.score++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (key == " ") {
|
|
||||||
while (canPlace(board, boardW, boardH, s.pieceId, s.rot, s.px, s.py + 1) == true) {
|
|
||||||
s.py++
|
|
||||||
s.score += 2
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (inputErr: Object) {
|
|
||||||
logError("applyKeyInput recovered after error", inputErr)
|
|
||||||
resetActivePiece(s)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var inputRunning = true
|
|
||||||
launch {
|
|
||||||
while (inputRunning) {
|
|
||||||
try {
|
|
||||||
for (ev in Console.events()) {
|
|
||||||
if (!inputRunning) break
|
|
||||||
|
|
||||||
// Isolate per-event failures so one bad event does not unwind the stream.
|
|
||||||
try {
|
|
||||||
if (ev is ConsoleKeyEvent) {
|
|
||||||
val ke = ev as ConsoleKeyEvent
|
|
||||||
if (ke.type == ConsoleEventType.KEY_DOWN) {
|
|
||||||
var key = ""
|
|
||||||
var ctrl = false
|
|
||||||
try {
|
|
||||||
key = ke.key
|
|
||||||
} catch (keyErr: Object) {
|
|
||||||
logError("Input key read error", keyErr)
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
ctrl = ke.ctrl == true
|
|
||||||
} catch (_: Object) {
|
|
||||||
ctrl = false
|
|
||||||
}
|
|
||||||
if (key == "") {
|
|
||||||
logError("Dropped key event with empty/null key", null)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
val mapped = if (ctrl && (key == "c" || key == "C")) "__CTRL_C__" else key
|
|
||||||
inputBuffer.push(mapped)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (eventErr: Object) {
|
|
||||||
// Keep the input stream alive; report for diagnostics.
|
|
||||||
logError("Input event error", eventErr)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (err: Object) {
|
|
||||||
// Recover stream-level failures by recreating event stream in next loop turn.
|
|
||||||
if (!inputRunning) break
|
|
||||||
logError("Input stream recovered after error", err)
|
|
||||||
Console.setRawMode(true)
|
|
||||||
delay(50)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fun pollLoopFrame(): LoopFrame? {
|
|
||||||
val g = Console.geometry()
|
|
||||||
val c = g?.columns ?: 0
|
|
||||||
val r = g?.rows ?: 0
|
|
||||||
if (c < MIN_COLS || r < MIN_ROWS) {
|
|
||||||
waitForMinimumSize(MIN_COLS, MIN_ROWS)
|
|
||||||
clearAndHome()
|
|
||||||
prevFrameLines = []
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
val contentCols = boardW * 2 + 2 + 3 + PANEL_WIDTH
|
|
||||||
val contentRows = boardH + 1
|
|
||||||
val requiredCols = max(MIN_COLS, contentCols)
|
|
||||||
val requiredRows = max(MIN_ROWS, contentRows)
|
|
||||||
if (c < requiredCols || r < requiredRows) {
|
|
||||||
waitForMinimumSize(requiredCols, requiredRows)
|
|
||||||
clearAndHome()
|
|
||||||
prevFrameLines = []
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
val originCol = max(1, ((c - contentCols) / 2) + 1)
|
|
||||||
val originRow = max(1, ((r - contentRows) / 2) + 1)
|
|
||||||
LoopFrame(false, originRow, originCol)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun advanceGravity(s: GameState, frame: Int): Int {
|
|
||||||
s.level = 1 + (s.totalLines / LEVEL_LINES_STEP)
|
|
||||||
val dropEvery = max(DROP_FRAMES_MIN, DROP_FRAMES_BASE - s.level)
|
|
||||||
|
|
||||||
var nextFrame = frame + 1
|
|
||||||
if (nextFrame < dropEvery) return nextFrame
|
|
||||||
nextFrame = 0
|
|
||||||
|
|
||||||
if (canPlace(board, boardW, boardH, s.pieceId, s.rot, s.px, s.py + 1) == true) {
|
|
||||||
s.py++
|
|
||||||
return nextFrame
|
|
||||||
}
|
|
||||||
|
|
||||||
lockPiece(board, s.pieceId, s.rot, s.px, s.py)
|
|
||||||
|
|
||||||
val cleared = clearCompletedLines(board, boardW, boardH)
|
|
||||||
if (cleared > 0) {
|
|
||||||
s.totalLines += cleared
|
|
||||||
s.score += scoreForLines(cleared, s.level)
|
|
||||||
}
|
|
||||||
|
|
||||||
s.pieceId = s.nextId
|
|
||||||
s.nextId = s.next2Id
|
|
||||||
s.next2Id = nextPieceId()
|
|
||||||
s.rot = 0
|
|
||||||
s.px = (boardW / 2) - 2
|
|
||||||
s.py = -1
|
|
||||||
|
|
||||||
if (!canPlace(board, boardW, boardH, s.pieceId, s.rot, s.px, s.py)) {
|
|
||||||
s.gameOver = true
|
|
||||||
}
|
|
||||||
nextFrame
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (!canPlace(board, boardW, boardH, state.pieceId, state.rot, state.px, state.py)) {
|
|
||||||
state.gameOver = true
|
|
||||||
}
|
|
||||||
|
|
||||||
var frame = 0
|
|
||||||
var shouldStop = false
|
|
||||||
var prevOriginRow = -1
|
|
||||||
var prevOriginCol = -1
|
|
||||||
var prevPaused = false
|
|
||||||
while (!shouldStop) {
|
|
||||||
val frameData = pollLoopFrame()
|
|
||||||
if (frameData == null) {
|
|
||||||
frame = 0
|
|
||||||
prevPaused = false
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
val toApply = inputBuffer.drain()
|
|
||||||
if (toApply.size > 0) {
|
|
||||||
for (k in toApply) {
|
|
||||||
applyKeyInput(state, k)
|
|
||||||
if (!state.running || state.gameOver) break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!state.running || state.gameOver) {
|
|
||||||
shouldStop = true
|
|
||||||
} else {
|
|
||||||
val localForceRedraw = forceRedraw
|
|
||||||
forceRedraw = false
|
|
||||||
val movedOrigin = frameData.originRow != prevOriginRow || frameData.originCol != prevOriginCol
|
|
||||||
if (frameData.resized || movedOrigin || localForceRedraw) {
|
|
||||||
clearAndHome()
|
|
||||||
prevFrameLines = []
|
|
||||||
}
|
|
||||||
prevOriginRow = frameData.originRow
|
|
||||||
prevOriginCol = frameData.originCol
|
|
||||||
if (!state.paused) {
|
|
||||||
frame = advanceGravity(state, frame)
|
|
||||||
}
|
|
||||||
prevFrameLines = render(
|
|
||||||
state,
|
|
||||||
board,
|
|
||||||
boardW,
|
|
||||||
boardH,
|
|
||||||
prevFrameLines,
|
|
||||||
frameData.originRow,
|
|
||||||
frameData.originCol,
|
|
||||||
useColor
|
|
||||||
)
|
|
||||||
if (state.paused && (!prevPaused || frameData.resized || movedOrigin)) {
|
|
||||||
renderPauseOverlay(frameData.originRow, frameData.originCol, boardW, boardH)
|
|
||||||
}
|
|
||||||
prevPaused = state.paused
|
|
||||||
}
|
|
||||||
Console.flush()
|
|
||||||
delay(FRAME_DELAY_MS)
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
inputRunning = false
|
|
||||||
Console.setRawMode(false)
|
|
||||||
Console.setCursorVisible(true)
|
|
||||||
Console.leaveAltScreen()
|
|
||||||
Console.flush()
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.gameOver) {
|
|
||||||
println("Game over.")
|
|
||||||
} else {
|
|
||||||
println("Bye.")
|
|
||||||
}
|
|
||||||
println("Score: %s"(state.score))
|
|
||||||
println("Lines: %s"(state.totalLines))
|
|
||||||
println("Level: %s"(state.level))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -27,6 +27,9 @@ kotlin.mpp.enableCInteropCommonization=true
|
|||||||
android.useAndroidX=true
|
android.useAndroidX=true
|
||||||
android.nonTransitiveRClass=true
|
android.nonTransitiveRClass=true
|
||||||
|
|
||||||
|
# other
|
||||||
|
kotlin.native.cacheKind.linuxX64=none
|
||||||
|
|
||||||
# Workaround: Ensure Gradle uses a JDK with `jlink` available for AGP's JDK image transform.
|
# Workaround: Ensure Gradle uses a JDK with `jlink` available for AGP's JDK image transform.
|
||||||
# On this environment, the system JDK 21 installation lacks `jlink`, causing
|
# On this environment, the system JDK 21 installation lacks `jlink`, causing
|
||||||
# :lynglib:androidJdkImage to fail. Point Gradle to a JDK that includes `jlink`.
|
# :lynglib:androidJdkImage to fail. Point Gradle to a JDK that includes `jlink`.
|
||||||
@ -34,6 +37,4 @@ android.nonTransitiveRClass=true
|
|||||||
#org.gradle.java.home=/home/sergeych/.jdks/corretto-21.0.9
|
#org.gradle.java.home=/home/sergeych/.jdks/corretto-21.0.9
|
||||||
android.experimental.lint.migrateToK2=false
|
android.experimental.lint.migrateToK2=false
|
||||||
android.lint.useK2Uast=false
|
android.lint.useK2Uast=false
|
||||||
kotlin.mpp.applyDefaultHierarchyTemplate=false
|
kotlin.mpp.applyDefaultHierarchyTemplate=true
|
||||||
|
|
||||||
org.gradle.parallel=true
|
|
||||||
@ -1,61 +1,31 @@
|
|||||||
[versions]
|
[versions]
|
||||||
agp = "8.5.2"
|
agp = "8.5.2"
|
||||||
clikt = "5.0.3"
|
clikt = "5.0.3"
|
||||||
mordant = "3.0.2"
|
kotlin = "2.3.0"
|
||||||
kotlin = "2.3.20"
|
|
||||||
android-minSdk = "24"
|
android-minSdk = "24"
|
||||||
android-compileSdk = "34"
|
android-compileSdk = "34"
|
||||||
kotlinx-coroutines = "1.10.2"
|
kotlinx-coroutines = "1.10.2"
|
||||||
kotlinx-datetime = "0.6.1"
|
kotlinx-datetime = "0.6.1"
|
||||||
mp_bintools = "0.3.2"
|
mp_bintools = "0.3.2"
|
||||||
ionspin-bignum = "0.3.10"
|
|
||||||
multik = "0.3.0"
|
|
||||||
firebaseCrashlyticsBuildtools = "3.0.3"
|
firebaseCrashlyticsBuildtools = "3.0.3"
|
||||||
okioVersion = "3.10.2"
|
okioVersion = "3.10.2"
|
||||||
compiler = "3.2.0-alpha11"
|
compiler = "3.2.0-alpha11"
|
||||||
ktor = "3.3.1"
|
|
||||||
slf4j = "2.0.17"
|
|
||||||
sqlite-jdbc = "3.50.3.0"
|
|
||||||
h2 = "2.4.240"
|
|
||||||
postgresql = "42.7.8"
|
|
||||||
testcontainers = "1.20.6"
|
|
||||||
hikaricp = "6.2.1"
|
|
||||||
|
|
||||||
[libraries]
|
[libraries]
|
||||||
clikt = { module = "com.github.ajalt.clikt:clikt", version.ref = "clikt" }
|
clikt = { module = "com.github.ajalt.clikt:clikt", version.ref = "clikt" }
|
||||||
clikt-core = { module = "com.github.ajalt.clikt:clikt-core", version.ref = "clikt" }
|
|
||||||
clikt-markdown = { module = "com.github.ajalt.clikt:clikt-markdown", version.ref = "clikt" }
|
clikt-markdown = { module = "com.github.ajalt.clikt:clikt-markdown", version.ref = "clikt" }
|
||||||
mordant-core = { module = "com.github.ajalt.mordant:mordant-core", version.ref = "mordant" }
|
|
||||||
mordant-jvm-jna = { module = "com.github.ajalt.mordant:mordant-jvm-jna", version.ref = "mordant" }
|
|
||||||
kotlin-test = { module = "org.jetbrains.kotlin:kotlin-test", version.ref = "kotlin" }
|
kotlin-test = { module = "org.jetbrains.kotlin:kotlin-test", version.ref = "kotlin" }
|
||||||
kotlinx-coroutines-test = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-test", version.ref = "kotlinx-coroutines" }
|
kotlinx-coroutines-test = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-test", version.ref = "kotlinx-coroutines" }
|
||||||
kotlinx-coroutines-core = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-core", version.ref = "kotlinx-coroutines" }
|
kotlinx-coroutines-core = { module = "org.jetbrains.kotlinx:kotlinx-coroutines-core", version.ref = "kotlinx-coroutines" }
|
||||||
kotlinx-datetime = { module = "org.jetbrains.kotlinx:kotlinx-datetime", version.ref = "kotlinx-datetime" }
|
kotlinx-datetime = { module = "org.jetbrains.kotlinx:kotlinx-datetime", version.ref = "kotlinx-datetime" }
|
||||||
mp_bintools = { module = "net.sergeych:mp_bintools", version.ref = "mp_bintools" }
|
mp_bintools = { module = "net.sergeych:mp_bintools", version.ref = "mp_bintools" }
|
||||||
ionspin-bignum = { module = "com.ionspin.kotlin:bignum", version.ref = "ionspin-bignum" }
|
|
||||||
multik-default = { module = "org.jetbrains.kotlinx:multik-default", version.ref = "multik" }
|
|
||||||
firebase-crashlytics-buildtools = { group = "com.google.firebase", name = "firebase-crashlytics-buildtools", version.ref = "firebaseCrashlyticsBuildtools" }
|
firebase-crashlytics-buildtools = { group = "com.google.firebase", name = "firebase-crashlytics-buildtools", version.ref = "firebaseCrashlyticsBuildtools" }
|
||||||
okio = { module = "com.squareup.okio:okio", version.ref = "okioVersion" }
|
okio = { module = "com.squareup.okio:okio", version.ref = "okioVersion" }
|
||||||
okio-fakefilesystem = { module = "com.squareup.okio:okio-fakefilesystem", version.ref = "okioVersion" }
|
okio-fakefilesystem = { module = "com.squareup.okio:okio-fakefilesystem", version.ref = "okioVersion" }
|
||||||
okio-nodefilesystem = { module = "com.squareup.okio:okio-nodefilesystem", version.ref = "okioVersion" }
|
okio-nodefilesystem = { module = "com.squareup.okio:okio-nodefilesystem", version.ref = "okioVersion" }
|
||||||
compiler = { group = "androidx.databinding", name = "compiler", version.ref = "compiler" }
|
compiler = { group = "androidx.databinding", name = "compiler", version.ref = "compiler" }
|
||||||
ktor-client-core = { module = "io.ktor:ktor-client-core", version.ref = "ktor" }
|
|
||||||
ktor-client-cio = { module = "io.ktor:ktor-client-cio", version.ref = "ktor" }
|
|
||||||
ktor-client-curl = { module = "io.ktor:ktor-client-curl", version.ref = "ktor" }
|
|
||||||
ktor-client-darwin = { module = "io.ktor:ktor-client-darwin", version.ref = "ktor" }
|
|
||||||
ktor-client-js = { module = "io.ktor:ktor-client-js", version.ref = "ktor" }
|
|
||||||
ktor-client-winhttp = { module = "io.ktor:ktor-client-winhttp", version.ref = "ktor" }
|
|
||||||
ktor-client-websockets = { module = "io.ktor:ktor-client-websockets", version.ref = "ktor" }
|
|
||||||
ktor-network = { module = "io.ktor:ktor-network", version.ref = "ktor" }
|
|
||||||
slf4j-nop = { module = "org.slf4j:slf4j-nop", version.ref = "slf4j" }
|
|
||||||
sqlite-jdbc = { module = "org.xerial:sqlite-jdbc", version.ref = "sqlite-jdbc" }
|
|
||||||
h2 = { module = "com.h2database:h2", version.ref = "h2" }
|
|
||||||
postgresql = { module = "org.postgresql:postgresql", version.ref = "postgresql" }
|
|
||||||
testcontainers = { module = "org.testcontainers:testcontainers", version.ref = "testcontainers" }
|
|
||||||
testcontainers-postgresql = { module = "org.testcontainers:postgresql", version.ref = "testcontainers" }
|
|
||||||
hikaricp = { module = "com.zaxxer:HikariCP", version.ref = "hikaricp" }
|
|
||||||
|
|
||||||
[plugins]
|
[plugins]
|
||||||
androidLibrary = { id = "com.android.library", version.ref = "agp" }
|
androidLibrary = { id = "com.android.library", version.ref = "agp" }
|
||||||
kotlinMultiplatform = { id = "org.jetbrains.kotlin.multiplatform", version.ref = "kotlin" }
|
kotlinMultiplatform = { id = "org.jetbrains.kotlin.multiplatform", version.ref = "kotlin" }
|
||||||
vanniktech-mavenPublish = { id = "com.vanniktech.maven.publish", version = "0.29.0" }
|
vanniktech-mavenPublish = { id = "com.vanniktech.maven.publish", version = "0.29.0" }
|
||||||
@ -1,2 +0,0 @@
|
|||||||
[vps]
|
|
||||||
94.130.36.94 ansible_user=sergeych
|
|
||||||
@ -1,100 +0,0 @@
|
|||||||
---
|
|
||||||
- name: Setup lynglang.com static site on VPS
|
|
||||||
hosts: vps
|
|
||||||
become: yes
|
|
||||||
vars:
|
|
||||||
domain: lynglang.com
|
|
||||||
web_root: /var/www/lynglang
|
|
||||||
deploy_user: sergeych
|
|
||||||
certbot_email: real.sergeych@gmail.com
|
|
||||||
|
|
||||||
tasks:
|
|
||||||
# Debian 10 buster is EOL; security/backports repos moved to archive.debian.org
|
|
||||||
- name: Fix sources.list for Debian buster EOL
|
|
||||||
copy:
|
|
||||||
dest: /etc/apt/sources.list
|
|
||||||
content: |
|
|
||||||
deb http://archive.debian.org/debian/ buster main contrib non-free
|
|
||||||
deb http://archive.debian.org/debian-security/ buster/updates main contrib non-free
|
|
||||||
deb http://archive.debian.org/debian/ buster-backports main contrib non-free
|
|
||||||
|
|
||||||
- name: Remove stale third-party sources (broken for buster EOL)
|
|
||||||
file:
|
|
||||||
path: "/etc/apt/sources.list.d/{{ item }}"
|
|
||||||
state: absent
|
|
||||||
loop:
|
|
||||||
- cassandra.list
|
|
||||||
- icinga.list
|
|
||||||
- postgres.list
|
|
||||||
- salt-stack.list
|
|
||||||
- yarn.list
|
|
||||||
|
|
||||||
- name: Install nginx, certbot, and python3-certbot-nginx
|
|
||||||
apt:
|
|
||||||
name:
|
|
||||||
- nginx
|
|
||||||
- certbot
|
|
||||||
- python3-certbot-nginx
|
|
||||||
state: present
|
|
||||||
update_cache: yes
|
|
||||||
|
|
||||||
- name: Create web root directory
|
|
||||||
file:
|
|
||||||
path: "{{ web_root }}/release/dist"
|
|
||||||
state: directory
|
|
||||||
owner: "{{ deploy_user }}"
|
|
||||||
group: www-data
|
|
||||||
mode: "0755"
|
|
||||||
recurse: yes
|
|
||||||
|
|
||||||
- name: Create distributables directory
|
|
||||||
file:
|
|
||||||
path: "{{ web_root }}/release/dist/distributables"
|
|
||||||
state: directory
|
|
||||||
owner: "{{ deploy_user }}"
|
|
||||||
group: www-data
|
|
||||||
mode: "0755"
|
|
||||||
|
|
||||||
- name: Deploy nginx site config (HTTP, pre-certbot)
|
|
||||||
template:
|
|
||||||
src: templates/nginx_lynglang.conf.j2
|
|
||||||
dest: /etc/nginx/sites-available/{{ domain }}
|
|
||||||
notify: reload nginx
|
|
||||||
|
|
||||||
- name: Enable nginx site
|
|
||||||
file:
|
|
||||||
src: /etc/nginx/sites-available/{{ domain }}
|
|
||||||
dest: /etc/nginx/sites-enabled/{{ domain }}
|
|
||||||
state: link
|
|
||||||
notify: reload nginx
|
|
||||||
|
|
||||||
- name: Disable default nginx site
|
|
||||||
file:
|
|
||||||
path: /etc/nginx/sites-enabled/default
|
|
||||||
state: absent
|
|
||||||
notify: reload nginx
|
|
||||||
|
|
||||||
- name: Ensure nginx is started
|
|
||||||
service:
|
|
||||||
name: nginx
|
|
||||||
state: started
|
|
||||||
enabled: yes
|
|
||||||
|
|
||||||
- name: Reload nginx before certbot
|
|
||||||
meta: flush_handlers
|
|
||||||
|
|
||||||
- name: Obtain SSL certificate via certbot (--nginx plugin)
|
|
||||||
command: >
|
|
||||||
certbot --nginx
|
|
||||||
-d {{ domain }} -d www.{{ domain }}
|
|
||||||
--non-interactive --agree-tos
|
|
||||||
--email {{ certbot_email }}
|
|
||||||
--redirect
|
|
||||||
args:
|
|
||||||
creates: /etc/letsencrypt/live/{{ domain }}/fullchain.pem
|
|
||||||
|
|
||||||
handlers:
|
|
||||||
- name: reload nginx
|
|
||||||
service:
|
|
||||||
name: nginx
|
|
||||||
state: reloaded
|
|
||||||
@ -1,24 +0,0 @@
|
|||||||
server {
|
|
||||||
listen 80;
|
|
||||||
server_name {{ domain }} www.{{ domain }};
|
|
||||||
|
|
||||||
root {{ web_root }}/release/dist;
|
|
||||||
index index.html;
|
|
||||||
|
|
||||||
# SPA fallback
|
|
||||||
location / {
|
|
||||||
try_files $uri $uri/ /index.html;
|
|
||||||
}
|
|
||||||
|
|
||||||
# Distributables served directly
|
|
||||||
location /distributables/ {
|
|
||||||
try_files $uri =404;
|
|
||||||
autoindex on;
|
|
||||||
}
|
|
||||||
|
|
||||||
# Long-lived cache for hashed assets
|
|
||||||
location ~* \.(js|css|woff2?|ttf|eot|svg|png|jpg|ico)$ {
|
|
||||||
expires 1y;
|
|
||||||
add_header Cache-Control "public, immutable";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -23,8 +23,6 @@ import com.intellij.execution.ui.ConsoleViewContentType
|
|||||||
import com.intellij.openapi.actionSystem.AnAction
|
import com.intellij.openapi.actionSystem.AnAction
|
||||||
import com.intellij.openapi.actionSystem.AnActionEvent
|
import com.intellij.openapi.actionSystem.AnActionEvent
|
||||||
import com.intellij.openapi.actionSystem.CommonDataKeys
|
import com.intellij.openapi.actionSystem.CommonDataKeys
|
||||||
import com.intellij.openapi.application.ApplicationManager
|
|
||||||
import com.intellij.openapi.fileEditor.FileDocumentManager
|
|
||||||
import com.intellij.openapi.project.Project
|
import com.intellij.openapi.project.Project
|
||||||
import com.intellij.openapi.wm.ToolWindow
|
import com.intellij.openapi.wm.ToolWindow
|
||||||
import com.intellij.openapi.wm.ToolWindowAnchor
|
import com.intellij.openapi.wm.ToolWindowAnchor
|
||||||
@ -37,11 +35,16 @@ import kotlinx.coroutines.CoroutineScope
|
|||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.SupervisorJob
|
import kotlinx.coroutines.SupervisorJob
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
|
import net.sergeych.lyng.ExecutionError
|
||||||
|
import net.sergeych.lyng.Script
|
||||||
|
import net.sergeych.lyng.Source
|
||||||
|
import net.sergeych.lyng.requireScope
|
||||||
import net.sergeych.lyng.idea.LyngIcons
|
import net.sergeych.lyng.idea.LyngIcons
|
||||||
import java.io.File
|
import net.sergeych.lyng.obj.ObjVoid
|
||||||
|
import net.sergeych.lyng.obj.getLyngExceptionMessageWithStackTrace
|
||||||
|
|
||||||
class RunLyngScriptAction : AnAction(LyngIcons.FILE) {
|
class RunLyngScriptAction : AnAction(LyngIcons.FILE) {
|
||||||
private val scope = CoroutineScope(Dispatchers.IO + SupervisorJob())
|
private val scope = CoroutineScope(Dispatchers.Default + SupervisorJob())
|
||||||
|
|
||||||
private fun getPsiFile(e: AnActionEvent): PsiFile? {
|
private fun getPsiFile(e: AnActionEvent): PsiFile? {
|
||||||
val project = e.project ?: return null
|
val project = e.project ?: return null
|
||||||
@ -51,99 +54,66 @@ class RunLyngScriptAction : AnAction(LyngIcons.FILE) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun getRunnableFile(e: AnActionEvent): PsiFile? {
|
|
||||||
val psiFile = getPsiFile(e) ?: return null
|
|
||||||
val virtualFile = psiFile.virtualFile ?: return null
|
|
||||||
if (!virtualFile.isInLocalFileSystem) return null
|
|
||||||
if (!psiFile.name.endsWith(".lyng")) return null
|
|
||||||
return psiFile
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun update(e: AnActionEvent) {
|
override fun update(e: AnActionEvent) {
|
||||||
val psiFile = getRunnableFile(e)
|
val psiFile = getPsiFile(e)
|
||||||
val isRunnable = psiFile != null
|
val isLyng = psiFile?.name?.endsWith(".lyng") == true
|
||||||
e.presentation.isEnabledAndVisible = isRunnable
|
e.presentation.isEnabledAndVisible = isLyng
|
||||||
if (isRunnable) {
|
if (isLyng) {
|
||||||
e.presentation.text = "Run '${psiFile.name}'"
|
e.presentation.text = "Run '${psiFile.name}'"
|
||||||
e.presentation.description = "Run the current Lyng script using the Lyng CLI"
|
|
||||||
} else {
|
} else {
|
||||||
e.presentation.text = "Run Lyng Script"
|
e.presentation.text = "Run Lyng Script"
|
||||||
e.presentation.description = "Run the current Lyng script"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun actionPerformed(e: AnActionEvent) {
|
override fun actionPerformed(e: AnActionEvent) {
|
||||||
val project = e.project ?: return
|
val project = e.project ?: return
|
||||||
val psiFile = getRunnableFile(e) ?: return
|
val psiFile = getPsiFile(e) ?: return
|
||||||
val virtualFile = psiFile.virtualFile ?: return
|
val text = psiFile.text
|
||||||
FileDocumentManager.getInstance().getDocument(virtualFile)?.let { document ->
|
val fileName = psiFile.name
|
||||||
FileDocumentManager.getInstance().saveDocument(document)
|
|
||||||
}
|
|
||||||
val filePath = virtualFile.path
|
|
||||||
val workingDir = virtualFile.parent?.path ?: project.basePath ?: File(filePath).parent
|
|
||||||
|
|
||||||
val (console, toolWindow) = getConsoleAndToolWindow(project)
|
val (console, toolWindow) = getConsoleAndToolWindow(project)
|
||||||
console.clear()
|
console.clear()
|
||||||
|
|
||||||
toolWindow.show {
|
toolWindow.show {
|
||||||
scope.launch {
|
scope.launch {
|
||||||
val command = startLyngProcess(filePath, workingDir)
|
try {
|
||||||
if (command == null) {
|
val lyngScope = Script.newScope()
|
||||||
printToConsole(console, "Unable to start Lyng CLI.\n", ConsoleViewContentType.ERROR_OUTPUT)
|
lyngScope.addFn("print") {
|
||||||
printToConsole(console, "Tried commands: lyng, jlyng.\n", ConsoleViewContentType.ERROR_OUTPUT)
|
val sb = StringBuilder()
|
||||||
printToConsole(console, "Install `lyng` or `jlyng` and make sure it is available on PATH.\n", ConsoleViewContentType.NORMAL_OUTPUT)
|
for ((i, arg) in args.list.withIndex()) {
|
||||||
return@launch
|
if (i > 0) sb.append(" ")
|
||||||
|
sb.append(arg.toString(requireScope()).value)
|
||||||
|
}
|
||||||
|
console.print(sb.toString(), ConsoleViewContentType.NORMAL_OUTPUT)
|
||||||
|
ObjVoid
|
||||||
|
}
|
||||||
|
lyngScope.addFn("println") {
|
||||||
|
val sb = StringBuilder()
|
||||||
|
for ((i, arg) in args.list.withIndex()) {
|
||||||
|
if (i > 0) sb.append(" ")
|
||||||
|
sb.append(arg.toString(requireScope()).value)
|
||||||
|
}
|
||||||
|
console.print(sb.toString() + "\n", ConsoleViewContentType.NORMAL_OUTPUT)
|
||||||
|
ObjVoid
|
||||||
|
}
|
||||||
|
|
||||||
|
console.print("--- Running $fileName ---\n", ConsoleViewContentType.SYSTEM_OUTPUT)
|
||||||
|
val result = lyngScope.eval(Source(fileName, text))
|
||||||
|
console.print("\n--- Finished with result: ${result.inspect(lyngScope)} ---\n", ConsoleViewContentType.SYSTEM_OUTPUT)
|
||||||
|
} catch (t: Throwable) {
|
||||||
|
console.print("\n--- Error ---\n", ConsoleViewContentType.ERROR_OUTPUT)
|
||||||
|
if( t is ExecutionError ) {
|
||||||
|
val m = t.errorObject.getLyngExceptionMessageWithStackTrace()
|
||||||
|
console.print(m, ConsoleViewContentType.ERROR_OUTPUT)
|
||||||
|
}
|
||||||
|
else
|
||||||
|
console.print(t.message ?: t.toString(), ConsoleViewContentType.ERROR_OUTPUT)
|
||||||
|
console.print("\n", ConsoleViewContentType.ERROR_OUTPUT)
|
||||||
}
|
}
|
||||||
|
|
||||||
printToConsole(
|
|
||||||
console,
|
|
||||||
"Running ${command.commandLine} in ${command.workingDir}\n",
|
|
||||||
ConsoleViewContentType.SYSTEM_OUTPUT
|
|
||||||
)
|
|
||||||
streamProcess(command.process, console)
|
|
||||||
val exitCode = command.process.waitFor()
|
|
||||||
val outputType = if (exitCode == 0) ConsoleViewContentType.SYSTEM_OUTPUT else ConsoleViewContentType.ERROR_OUTPUT
|
|
||||||
printToConsole(console, "\nProcess finished with exit code $exitCode\n", outputType)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private suspend fun streamProcess(process: Process, console: ConsoleView) {
|
|
||||||
val stdout = scope.launch {
|
|
||||||
process.inputStream.bufferedReader().useLines { lines ->
|
|
||||||
lines.forEach { printToConsole(console, "$it\n", ConsoleViewContentType.NORMAL_OUTPUT) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val stderr = scope.launch {
|
|
||||||
process.errorStream.bufferedReader().useLines { lines ->
|
|
||||||
lines.forEach { printToConsole(console, "$it\n", ConsoleViewContentType.ERROR_OUTPUT) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
stdout.join()
|
|
||||||
stderr.join()
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun printToConsole(console: ConsoleView, text: String, type: ConsoleViewContentType) {
|
|
||||||
ApplicationManager.getApplication().invokeLater {
|
|
||||||
console.print(text, type)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun startLyngProcess(filePath: String, workingDir: String?): StartedProcess? {
|
|
||||||
val candidates = listOf("lyng", "jlyng")
|
|
||||||
for (candidate in candidates) {
|
|
||||||
try {
|
|
||||||
val process = ProcessBuilder(candidate, filePath)
|
|
||||||
.directory(workingDir?.let(::File))
|
|
||||||
.start()
|
|
||||||
return StartedProcess(process, "$candidate $filePath", workingDir ?: File(filePath).parent.orEmpty())
|
|
||||||
} catch (_: java.io.IOException) {
|
|
||||||
// Try the next candidate when the command is not available.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getConsoleAndToolWindow(project: Project): Pair<ConsoleView, ToolWindow> {
|
private fun getConsoleAndToolWindow(project: Project): Pair<ConsoleView, ToolWindow> {
|
||||||
val toolWindowManager = ToolWindowManager.getInstance(project)
|
val toolWindowManager = ToolWindowManager.getInstance(project)
|
||||||
var toolWindow = toolWindowManager.getToolWindow(ToolWindowId.RUN)
|
var toolWindow = toolWindowManager.getToolWindow(ToolWindowId.RUN)
|
||||||
@ -172,10 +142,4 @@ class RunLyngScriptAction : AnAction(LyngIcons.FILE) {
|
|||||||
contentManager.setSelectedContent(content)
|
contentManager.setSelectedContent(content)
|
||||||
return console to actualToolWindow
|
return console to actualToolWindow
|
||||||
}
|
}
|
||||||
|
|
||||||
private data class StartedProcess(
|
|
||||||
val process: Process,
|
|
||||||
val commandLine: String,
|
|
||||||
val workingDir: String
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -88,11 +88,9 @@ class LyngExternalAnnotator : ExternalAnnotator<LyngExternalAnnotator.Input, Lyn
|
|||||||
// Imports: each segment as namespace/path
|
// Imports: each segment as namespace/path
|
||||||
mini?.imports?.forEach { imp ->
|
mini?.imports?.forEach { imp ->
|
||||||
imp.segments.forEach { seg ->
|
imp.segments.forEach { seg ->
|
||||||
if (seg.range.start.source === analysis.source && seg.range.end.source === analysis.source) {
|
val start = analysis.source.offsetOf(seg.range.start)
|
||||||
val start = analysis.source.offsetOf(seg.range.start)
|
val end = analysis.source.offsetOf(seg.range.end)
|
||||||
val end = analysis.source.offsetOf(seg.range.end)
|
putRange(start, end, LyngHighlighterColors.NAMESPACE)
|
||||||
putRange(start, end, LyngHighlighterColors.NAMESPACE)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -24,7 +24,6 @@ import com.intellij.openapi.editor.Editor
|
|||||||
import com.intellij.openapi.util.TextRange
|
import com.intellij.openapi.util.TextRange
|
||||||
import com.intellij.psi.PsiElement
|
import com.intellij.psi.PsiElement
|
||||||
import com.intellij.psi.PsiFile
|
import com.intellij.psi.PsiFile
|
||||||
import kotlinx.coroutines.runBlocking
|
|
||||||
import net.sergeych.lyng.highlight.offsetOf
|
import net.sergeych.lyng.highlight.offsetOf
|
||||||
import net.sergeych.lyng.idea.LyngLanguage
|
import net.sergeych.lyng.idea.LyngLanguage
|
||||||
import net.sergeych.lyng.idea.util.LyngAstManager
|
import net.sergeych.lyng.idea.util.LyngAstManager
|
||||||
@ -76,51 +75,7 @@ class LyngDocumentationProvider : AbstractDocumentationProvider() {
|
|||||||
|
|
||||||
// Single-source quick doc lookup
|
// Single-source quick doc lookup
|
||||||
LyngLanguageTools.docAt(analysis, offset)?.let { info ->
|
LyngLanguageTools.docAt(analysis, offset)?.let { info ->
|
||||||
val enriched = if (info.doc == null) {
|
renderDocFromInfo(info)?.let { return it }
|
||||||
findDocInDeclarationFiles(file, info.target.containerName, info.target.name)
|
|
||||||
?.let { info.copy(doc = it) } ?: info
|
|
||||||
} else {
|
|
||||||
info
|
|
||||||
}
|
|
||||||
renderDocFromInfo(enriched)?.let { return it }
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fallback: resolve references against merged MiniAst (including .lyng.d) when binder cannot
|
|
||||||
run {
|
|
||||||
val dotPos = DocLookupUtils.findDotLeft(text, idRange.startOffset)
|
|
||||||
if (dotPos != null) {
|
|
||||||
val receiverClass = DocLookupUtils.guessReceiverClassViaMini(mini, text, dotPos, imported, analysis.binding)
|
|
||||||
?: DocLookupUtils.guessReceiverClass(text, dotPos, imported, mini)
|
|
||||||
if (receiverClass != null) {
|
|
||||||
val resolved = DocLookupUtils.resolveMemberWithInheritance(imported, receiverClass, ident, mini)
|
|
||||||
if (resolved != null) {
|
|
||||||
val owner = resolved.first
|
|
||||||
val member = resolved.second
|
|
||||||
val withDoc = if (member.doc == null) {
|
|
||||||
findDocInDeclarationFiles(file, owner, member.name)?.let { doc ->
|
|
||||||
when (member) {
|
|
||||||
is MiniMemberFunDecl -> member.copy(doc = doc)
|
|
||||||
is MiniMemberValDecl -> member.copy(doc = doc)
|
|
||||||
is MiniMemberTypeAliasDecl -> member.copy(doc = doc)
|
|
||||||
else -> member
|
|
||||||
}
|
|
||||||
} ?: member
|
|
||||||
} else {
|
|
||||||
member
|
|
||||||
}
|
|
||||||
return when (withDoc) {
|
|
||||||
is MiniMemberFunDecl -> renderMemberFunDoc(owner, withDoc)
|
|
||||||
is MiniMemberValDecl -> renderMemberValDoc(owner, withDoc)
|
|
||||||
is MiniMemberTypeAliasDecl -> renderMemberTypeAliasDoc(owner, withDoc)
|
|
||||||
else -> null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
mini.declarations.firstOrNull { it.name == ident }?.let { decl ->
|
|
||||||
return renderDeclDoc(decl, text, mini, imported)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try resolve to: function param at position, function/class/val declaration at position
|
// Try resolve to: function param at position, function/class/val declaration at position
|
||||||
@ -310,7 +265,7 @@ class LyngDocumentationProvider : AbstractDocumentationProvider() {
|
|||||||
// Try literal and call-based receiver inference around the dot
|
// Try literal and call-based receiver inference around the dot
|
||||||
val i = TextCtx.prevNonWs(text, dotPos - 1)
|
val i = TextCtx.prevNonWs(text, dotPos - 1)
|
||||||
val className: String? = when {
|
val className: String? = when {
|
||||||
i >= 0 && (text[i] == '"' || text[i] == '`') -> "String"
|
i >= 0 && text[i] == '"' -> "String"
|
||||||
i >= 0 && text[i] == ']' -> "List"
|
i >= 0 && text[i] == ']' -> "List"
|
||||||
i >= 0 && text[i] == '}' -> "Dict"
|
i >= 0 && text[i] == '}' -> "Dict"
|
||||||
i >= 0 && text[i] == ')' -> {
|
i >= 0 && text[i] == ')' -> {
|
||||||
@ -615,55 +570,6 @@ class LyngDocumentationProvider : AbstractDocumentationProvider() {
|
|||||||
return sb.toString()
|
return sb.toString()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun findDocInDeclarationFiles(file: PsiFile, container: String?, name: String): MiniDoc? {
|
|
||||||
val declFiles = LyngAstManager.getDeclarationFiles(file)
|
|
||||||
if (declFiles.isEmpty()) return null
|
|
||||||
|
|
||||||
fun findInMini(mini: MiniScript): MiniDoc? {
|
|
||||||
if (container == null) {
|
|
||||||
mini.declarations.firstOrNull { it.name == name }?.let { return it.doc }
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
val cls = mini.declarations.filterIsInstance<MiniClassDecl>().firstOrNull { it.name == container } ?: return null
|
|
||||||
cls.members.firstOrNull { it.name == name }?.let { return it.doc }
|
|
||||||
cls.ctorFields.firstOrNull { it.name == name }?.let { return null }
|
|
||||||
cls.classFields.firstOrNull { it.name == name }?.let { return null }
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
for (df in declFiles) {
|
|
||||||
val mini = LyngAstManager.getMiniAst(df)
|
|
||||||
?: run {
|
|
||||||
try {
|
|
||||||
val res = runBlocking {
|
|
||||||
LyngLanguageTools.analyze(df.text, df.name)
|
|
||||||
}
|
|
||||||
res.mini
|
|
||||||
} catch (_: Throwable) {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (mini != null) {
|
|
||||||
val doc = findInMini(mini)
|
|
||||||
if (doc != null) return doc
|
|
||||||
}
|
|
||||||
// Text fallback: parse preceding doc comment for the symbol
|
|
||||||
val parsed = parseDocFromText(df.text, name)
|
|
||||||
if (parsed != null) return parsed
|
|
||||||
}
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun parseDocFromText(text: String, name: String): MiniDoc? {
|
|
||||||
if (text.isBlank()) return null
|
|
||||||
val pattern = Regex("/\\*\\*([\\s\\S]*?)\\*/\\s*(?:public|private|protected|static|abstract|extern|open|closed|override\\s+)*\\s*(?:fun|val|var|class|interface|enum|type)\\s+$name\\b")
|
|
||||||
val m = pattern.find(text) ?: return null
|
|
||||||
val raw = m.groupValues.getOrNull(1)?.trim() ?: return null
|
|
||||||
if (raw.isBlank()) return null
|
|
||||||
val src = net.sergeych.lyng.Source("<doc>", raw)
|
|
||||||
return MiniDoc.parse(MiniRange(src.startPos, src.startPos), raw.lines())
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun renderParamDoc(fn: MiniFunDecl, p: MiniParam): String {
|
private fun renderParamDoc(fn: MiniFunDecl, p: MiniParam): String {
|
||||||
val title = "parameter ${p.name}${typeOf(p.type)} in ${fn.name}${signatureOf(fn)}"
|
val title = "parameter ${p.name}${typeOf(p.type)} in ${fn.name}${signatureOf(fn)}"
|
||||||
val sb = StringBuilder()
|
val sb = StringBuilder()
|
||||||
|
|||||||
@ -24,7 +24,6 @@ import com.intellij.psi.codeStyle.CodeStyleManager
|
|||||||
import com.intellij.psi.impl.source.codeStyle.PreFormatProcessor
|
import com.intellij.psi.impl.source.codeStyle.PreFormatProcessor
|
||||||
import net.sergeych.lyng.format.LyngFormatConfig
|
import net.sergeych.lyng.format.LyngFormatConfig
|
||||||
import net.sergeych.lyng.format.LyngFormatter
|
import net.sergeych.lyng.format.LyngFormatter
|
||||||
import net.sergeych.lyng.format.LyngStringDelimiterPolicy
|
|
||||||
import net.sergeych.lyng.idea.LyngLanguage
|
import net.sergeych.lyng.idea.LyngLanguage
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -171,7 +170,6 @@ class LyngPreFormatProcessor : PreFormatProcessor {
|
|||||||
continuationIndentSize = options.CONTINUATION_INDENT_SIZE.coerceAtLeast(options.INDENT_SIZE.coerceAtLeast(1)),
|
continuationIndentSize = options.CONTINUATION_INDENT_SIZE.coerceAtLeast(options.INDENT_SIZE.coerceAtLeast(1)),
|
||||||
applySpacing = true,
|
applySpacing = true,
|
||||||
applyWrapping = false,
|
applyWrapping = false,
|
||||||
stringDelimiterPolicy = LyngStringDelimiterPolicy.PreferFewerEscapes,
|
|
||||||
)
|
)
|
||||||
val r = if (runFullFileIndent) currentLocalRange() else workingRangeLocal.intersection(currentLocalRange()) ?: currentLocalRange()
|
val r = if (runFullFileIndent) currentLocalRange() else workingRangeLocal.intersection(currentLocalRange()) ?: currentLocalRange()
|
||||||
val text = doc.getText(r)
|
val text = doc.getText(r)
|
||||||
@ -191,7 +189,6 @@ class LyngPreFormatProcessor : PreFormatProcessor {
|
|||||||
continuationIndentSize = options.CONTINUATION_INDENT_SIZE.coerceAtLeast(options.INDENT_SIZE.coerceAtLeast(1)),
|
continuationIndentSize = options.CONTINUATION_INDENT_SIZE.coerceAtLeast(options.INDENT_SIZE.coerceAtLeast(1)),
|
||||||
applySpacing = settings.enableSpacing,
|
applySpacing = settings.enableSpacing,
|
||||||
applyWrapping = true,
|
applyWrapping = true,
|
||||||
stringDelimiterPolicy = LyngStringDelimiterPolicy.PreferFewerEscapes,
|
|
||||||
)
|
)
|
||||||
val r = if (runFullFileIndent) currentLocalRange() else workingRangeLocal.intersection(currentLocalRange()) ?: currentLocalRange()
|
val r = if (runFullFileIndent) currentLocalRange() else workingRangeLocal.intersection(currentLocalRange()) ?: currentLocalRange()
|
||||||
val text = doc.getText(r)
|
val text = doc.getText(r)
|
||||||
|
|||||||
@ -35,7 +35,7 @@ class LyngLexer : LexerBase() {
|
|||||||
"fun", "val", "var", "class", "interface", "type", "import", "as",
|
"fun", "val", "var", "class", "interface", "type", "import", "as",
|
||||||
"abstract", "closed", "override", "static", "extern", "open", "private", "protected",
|
"abstract", "closed", "override", "static", "extern", "open", "private", "protected",
|
||||||
"if", "else", "for", "while", "return", "true", "false", "null",
|
"if", "else", "for", "while", "return", "true", "false", "null",
|
||||||
"when", "in", "is", "break", "continue", "try", "catch", "finally", "void",
|
"when", "in", "is", "break", "continue", "try", "catch", "finally",
|
||||||
"get", "set", "object", "enum", "init", "by", "step", "property", "constructor"
|
"get", "set", "object", "enum", "init", "by", "step", "property", "constructor"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -101,8 +101,8 @@ class LyngLexer : LexerBase() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// String "...", `...`, or '...' with simple escape handling
|
// String "..." or '...' with simple escape handling
|
||||||
if (ch == '"' || ch == '\'' || ch == '`') {
|
if (ch == '"' || ch == '\'') {
|
||||||
val quote = ch
|
val quote = ch
|
||||||
i++
|
i++
|
||||||
while (i < endOffset) {
|
while (i < endOffset) {
|
||||||
|
|||||||
@ -20,18 +20,12 @@ package net.sergeych.lyng.idea.navigation
|
|||||||
import com.intellij.openapi.project.Project
|
import com.intellij.openapi.project.Project
|
||||||
import com.intellij.openapi.util.TextRange
|
import com.intellij.openapi.util.TextRange
|
||||||
import com.intellij.psi.*
|
import com.intellij.psi.*
|
||||||
import com.intellij.psi.search.FileTypeIndex
|
|
||||||
import com.intellij.psi.search.FilenameIndex
|
import com.intellij.psi.search.FilenameIndex
|
||||||
import com.intellij.psi.search.GlobalSearchScope
|
import com.intellij.psi.search.GlobalSearchScope
|
||||||
import kotlinx.coroutines.runBlocking
|
|
||||||
import net.sergeych.lyng.highlight.offsetOf
|
import net.sergeych.lyng.highlight.offsetOf
|
||||||
import net.sergeych.lyng.idea.LyngFileType
|
|
||||||
import net.sergeych.lyng.idea.util.LyngAstManager
|
import net.sergeych.lyng.idea.util.LyngAstManager
|
||||||
import net.sergeych.lyng.idea.util.LyngIdeaImportProvider
|
|
||||||
import net.sergeych.lyng.idea.util.TextCtx
|
import net.sergeych.lyng.idea.util.TextCtx
|
||||||
import net.sergeych.lyng.miniast.*
|
import net.sergeych.lyng.miniast.*
|
||||||
import net.sergeych.lyng.tools.LyngAnalysisRequest
|
|
||||||
import net.sergeych.lyng.tools.LyngLanguageTools
|
|
||||||
|
|
||||||
class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiElement>(element, TextRange(0, element.textLength)) {
|
class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiElement>(element, TextRange(0, element.textLength)) {
|
||||||
|
|
||||||
@ -64,7 +58,7 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
|
|
||||||
// We need to find the actual PSI element for this member
|
// We need to find the actual PSI element for this member
|
||||||
val targetFile = findFileForClass(file.project, owner) ?: file
|
val targetFile = findFileForClass(file.project, owner) ?: file
|
||||||
val targetMini = loadMini(targetFile)
|
val targetMini = LyngAstManager.getMiniAst(targetFile)
|
||||||
if (targetMini != null) {
|
if (targetMini != null) {
|
||||||
val targetSrc = targetMini.range.start.source
|
val targetSrc = targetMini.range.start.source
|
||||||
val off = targetSrc.offsetOf(member.nameStart)
|
val off = targetSrc.offsetOf(member.nameStart)
|
||||||
@ -129,37 +123,24 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun findFileForClass(project: Project, className: String): PsiFile? {
|
private fun findFileForClass(project: Project, className: String): PsiFile? {
|
||||||
// 1. Try file with matching name first (optimization)
|
|
||||||
val scope = GlobalSearchScope.projectScope(project)
|
|
||||||
val psiManager = PsiManager.getInstance(project)
|
val psiManager = PsiManager.getInstance(project)
|
||||||
val matchingFiles = FileTypeIndex.getFiles(LyngFileType, scope)
|
|
||||||
.asSequence()
|
// 1. Try file with matching name first (optimization)
|
||||||
.filter { it.name == "$className.lyng" }
|
val matchingFiles = FilenameIndex.getFilesByName(project, "$className.lyng", GlobalSearchScope.projectScope(project))
|
||||||
.mapNotNull { psiManager.findFile(it) }
|
|
||||||
.toList()
|
|
||||||
val matchingDeclFiles = FileTypeIndex.getFiles(LyngFileType, scope)
|
|
||||||
.asSequence()
|
|
||||||
.filter { it.name == "$className.lyng.d" }
|
|
||||||
.mapNotNull { psiManager.findFile(it) }
|
|
||||||
.toList()
|
|
||||||
for (file in matchingFiles) {
|
for (file in matchingFiles) {
|
||||||
val mini = loadMini(file) ?: continue
|
val mini = LyngAstManager.getMiniAst(file) ?: continue
|
||||||
if (mini.declarations.any { isLocalDecl(mini, it) && ((it is MiniClassDecl && it.name == className) || (it is MiniEnumDecl && it.name == className)) }) {
|
if (mini.declarations.any { (it is MiniClassDecl && it.name == className) || (it is MiniEnumDecl && it.name == className) }) {
|
||||||
return file
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (file in matchingDeclFiles) {
|
|
||||||
val mini = loadMini(file) ?: continue
|
|
||||||
if (mini.declarations.any { isLocalDecl(mini, it) && ((it is MiniClassDecl && it.name == className) || (it is MiniEnumDecl && it.name == className)) }) {
|
|
||||||
return file
|
return file
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Fallback to full project scan
|
// 2. Fallback to full project scan
|
||||||
for (file in collectLyngFiles(project)) {
|
val allFiles = FilenameIndex.getAllFilesByExt(project, "lyng", GlobalSearchScope.projectScope(project))
|
||||||
if (matchingFiles.contains(file) || matchingDeclFiles.contains(file)) continue // already checked
|
for (vFile in allFiles) {
|
||||||
val mini = loadMini(file) ?: continue
|
val file = psiManager.findFile(vFile) ?: continue
|
||||||
if (mini.declarations.any { isLocalDecl(mini, it) && ((it is MiniClassDecl && it.name == className) || (it is MiniEnumDecl && it.name == className)) }) {
|
if (matchingFiles.contains(file)) continue // already checked
|
||||||
|
val mini = LyngAstManager.getMiniAst(file) ?: continue
|
||||||
|
if (mini.declarations.any { (it is MiniClassDecl && it.name == className) || (it is MiniEnumDecl && it.name == className) }) {
|
||||||
return file
|
return file
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -167,7 +148,7 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun getPackageName(file: PsiFile): String? {
|
private fun getPackageName(file: PsiFile): String? {
|
||||||
val mini = loadMini(file) ?: return null
|
val mini = LyngAstManager.getMiniAst(file) ?: return null
|
||||||
return try {
|
return try {
|
||||||
val pkg = mini.range.start.source.extractPackageName()
|
val pkg = mini.range.start.source.extractPackageName()
|
||||||
if (pkg.startsWith("lyng.")) pkg else "lyng.$pkg"
|
if (pkg.startsWith("lyng.")) pkg else "lyng.$pkg"
|
||||||
@ -191,19 +172,19 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
|
|
||||||
private fun resolveGlobally(project: Project, name: String, membersOnly: Boolean = false, allowedPackages: Set<String>? = null): List<ResolveResult> {
|
private fun resolveGlobally(project: Project, name: String, membersOnly: Boolean = false, allowedPackages: Set<String>? = null): List<ResolveResult> {
|
||||||
val results = mutableListOf<ResolveResult>()
|
val results = mutableListOf<ResolveResult>()
|
||||||
|
val files = FilenameIndex.getAllFilesByExt(project, "lyng", GlobalSearchScope.projectScope(project))
|
||||||
val psiManager = PsiManager.getInstance(project)
|
val psiManager = PsiManager.getInstance(project)
|
||||||
|
|
||||||
for (file in collectLyngFiles(project)) {
|
for (vFile in files) {
|
||||||
|
val file = psiManager.findFile(vFile) ?: continue
|
||||||
|
|
||||||
// Filter by package if requested
|
// Filter by package if requested
|
||||||
if (allowedPackages != null) {
|
if (allowedPackages != null) {
|
||||||
val pkg = getPackageName(file)
|
val pkg = getPackageName(file)
|
||||||
if (pkg == null) {
|
if (pkg == null || pkg !in allowedPackages) continue
|
||||||
if (!file.name.endsWith(".lyng.d")) continue
|
|
||||||
} else if (pkg !in allowedPackages) continue
|
|
||||||
}
|
}
|
||||||
|
|
||||||
val mini = loadMini(file) ?: continue
|
val mini = LyngAstManager.getMiniAst(file) ?: continue
|
||||||
val src = mini.range.start.source
|
val src = mini.range.start.source
|
||||||
|
|
||||||
fun addIfMatch(dName: String, nameStart: net.sergeych.lyng.Pos, dKind: String) {
|
fun addIfMatch(dName: String, nameStart: net.sergeych.lyng.Pos, dKind: String) {
|
||||||
@ -216,7 +197,6 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (d in mini.declarations) {
|
for (d in mini.declarations) {
|
||||||
if (!isLocalDecl(mini, d)) continue
|
|
||||||
if (!membersOnly) {
|
if (!membersOnly) {
|
||||||
val dKind = when(d) {
|
val dKind = when(d) {
|
||||||
is net.sergeych.lyng.miniast.MiniFunDecl -> "Function"
|
is net.sergeych.lyng.miniast.MiniFunDecl -> "Function"
|
||||||
@ -236,7 +216,6 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (m in members) {
|
for (m in members) {
|
||||||
if (m.range.start.source != src) continue
|
|
||||||
val mKind = when(m) {
|
val mKind = when(m) {
|
||||||
is net.sergeych.lyng.miniast.MiniMemberFunDecl -> "Function"
|
is net.sergeych.lyng.miniast.MiniMemberFunDecl -> "Function"
|
||||||
is net.sergeych.lyng.miniast.MiniMemberValDecl -> if (m.mutable) "Variable" else "Value"
|
is net.sergeych.lyng.miniast.MiniMemberValDecl -> if (m.mutable) "Variable" else "Value"
|
||||||
@ -250,42 +229,5 @@ class LyngPsiReference(element: PsiElement) : PsiPolyVariantReferenceBase<PsiEle
|
|||||||
return results
|
return results
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun collectLyngFiles(project: Project): List<PsiFile> {
|
|
||||||
val scope = GlobalSearchScope.projectScope(project)
|
|
||||||
val psiManager = PsiManager.getInstance(project)
|
|
||||||
val out = LinkedHashSet<PsiFile>()
|
|
||||||
|
|
||||||
val lyngFiles = FilenameIndex.getAllFilesByExt(project, "lyng", scope)
|
|
||||||
for (vFile in lyngFiles) {
|
|
||||||
psiManager.findFile(vFile)?.let { out.add(it) }
|
|
||||||
}
|
|
||||||
|
|
||||||
// Include declaration files (*.lyng.d) which are indexed as extension "d".
|
|
||||||
val dFiles = FilenameIndex.getAllFilesByExt(project, "d", scope)
|
|
||||||
for (vFile in dFiles) {
|
|
||||||
if (!vFile.name.endsWith(".lyng.d")) continue
|
|
||||||
psiManager.findFile(vFile)?.let { out.add(it) }
|
|
||||||
}
|
|
||||||
|
|
||||||
return out.toList()
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun loadMini(file: PsiFile): MiniScript? {
|
|
||||||
LyngAstManager.getMiniAst(file)?.let { return it }
|
|
||||||
return try {
|
|
||||||
val provider = LyngIdeaImportProvider.create()
|
|
||||||
runBlocking {
|
|
||||||
LyngLanguageTools.analyze(
|
|
||||||
LyngAnalysisRequest(text = file.text, fileName = file.name, importProvider = provider)
|
|
||||||
)
|
|
||||||
}.mini
|
|
||||||
} catch (_: Throwable) {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun isLocalDecl(mini: MiniScript, decl: MiniDecl): Boolean =
|
|
||||||
decl.range.start.source == mini.range.start.source
|
|
||||||
|
|
||||||
override fun getVariants(): Array<Any> = emptyArray()
|
override fun getVariants(): Array<Any> = emptyArray()
|
||||||
}
|
}
|
||||||
|
|||||||
@ -21,16 +21,13 @@ import com.intellij.openapi.application.runReadAction
|
|||||||
import com.intellij.openapi.util.Key
|
import com.intellij.openapi.util.Key
|
||||||
import com.intellij.psi.PsiFile
|
import com.intellij.psi.PsiFile
|
||||||
import com.intellij.psi.PsiManager
|
import com.intellij.psi.PsiManager
|
||||||
import com.intellij.psi.search.FileTypeIndex
|
|
||||||
import com.intellij.psi.search.FilenameIndex
|
|
||||||
import com.intellij.psi.search.GlobalSearchScope
|
|
||||||
import kotlinx.coroutines.runBlocking
|
import kotlinx.coroutines.runBlocking
|
||||||
import net.sergeych.lyng.binding.BindingSnapshot
|
import net.sergeych.lyng.binding.BindingSnapshot
|
||||||
import net.sergeych.lyng.idea.LyngFileType
|
import net.sergeych.lyng.miniast.DocLookupUtils
|
||||||
import net.sergeych.lyng.miniast.*
|
import net.sergeych.lyng.miniast.MiniScript
|
||||||
|
import net.sergeych.lyng.tools.IdeLenientImportProvider
|
||||||
import net.sergeych.lyng.tools.LyngAnalysisRequest
|
import net.sergeych.lyng.tools.LyngAnalysisRequest
|
||||||
import net.sergeych.lyng.tools.LyngAnalysisResult
|
import net.sergeych.lyng.tools.LyngAnalysisResult
|
||||||
import net.sergeych.lyng.tools.LyngDiagnostic
|
|
||||||
import net.sergeych.lyng.tools.LyngLanguageTools
|
import net.sergeych.lyng.tools.LyngLanguageTools
|
||||||
|
|
||||||
object LyngAstManager {
|
object LyngAstManager {
|
||||||
@ -38,8 +35,6 @@ object LyngAstManager {
|
|||||||
private val BINDING_KEY = Key.create<BindingSnapshot>("lyng.binding.cache")
|
private val BINDING_KEY = Key.create<BindingSnapshot>("lyng.binding.cache")
|
||||||
private val STAMP_KEY = Key.create<Long>("lyng.mini.cache.stamp")
|
private val STAMP_KEY = Key.create<Long>("lyng.mini.cache.stamp")
|
||||||
private val ANALYSIS_KEY = Key.create<LyngAnalysisResult>("lyng.analysis.cache")
|
private val ANALYSIS_KEY = Key.create<LyngAnalysisResult>("lyng.analysis.cache")
|
||||||
private val implicitBuiltinNames = setOf("void")
|
|
||||||
private val includeSymbolsDirective = Regex("""(?im)^\s*//\s*include\s+symbols\s*:\s*(.+?)\s*$""")
|
|
||||||
|
|
||||||
fun getMiniAst(file: PsiFile): MiniScript? = runReadAction {
|
fun getMiniAst(file: PsiFile): MiniScript? = runReadAction {
|
||||||
getAnalysis(file)?.mini
|
getAnalysis(file)?.mini
|
||||||
@ -48,8 +43,8 @@ object LyngAstManager {
|
|||||||
fun getCombinedStamp(file: PsiFile): Long = runReadAction {
|
fun getCombinedStamp(file: PsiFile): Long = runReadAction {
|
||||||
var combinedStamp = file.viewProvider.modificationStamp
|
var combinedStamp = file.viewProvider.modificationStamp
|
||||||
if (!file.name.endsWith(".lyng.d")) {
|
if (!file.name.endsWith(".lyng.d")) {
|
||||||
collectDeclarationFiles(file).forEach { symbolsFile ->
|
collectDeclarationFiles(file).forEach { df ->
|
||||||
combinedStamp += symbolsFile.viewProvider.modificationStamp
|
combinedStamp += df.viewProvider.modificationStamp
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
combinedStamp
|
combinedStamp
|
||||||
@ -57,81 +52,22 @@ object LyngAstManager {
|
|||||||
|
|
||||||
private fun collectDeclarationFiles(file: PsiFile): List<PsiFile> = runReadAction {
|
private fun collectDeclarationFiles(file: PsiFile): List<PsiFile> = runReadAction {
|
||||||
val psiManager = PsiManager.getInstance(file.project)
|
val psiManager = PsiManager.getInstance(file.project)
|
||||||
|
var current = file.virtualFile?.parent
|
||||||
val seen = mutableSetOf<String>()
|
val seen = mutableSetOf<String>()
|
||||||
val result = mutableListOf<PsiFile>()
|
val result = mutableListOf<PsiFile>()
|
||||||
|
|
||||||
var currentDir = file.containingDirectory
|
while (current != null) {
|
||||||
while (currentDir != null) {
|
for (child in current.children) {
|
||||||
for (child in currentDir.files) {
|
if (child.name.endsWith(".lyng.d") && child != file.virtualFile && seen.add(child.path)) {
|
||||||
if (child.name.endsWith(".lyng.d") && child != file && seen.add(child.virtualFile.path)) {
|
val psiD = psiManager.findFile(child) ?: continue
|
||||||
result.add(child)
|
result.add(psiD)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
currentDir = currentDir.parentDirectory
|
current = current.parent
|
||||||
}
|
|
||||||
|
|
||||||
val includeSpecs = includeSymbolsDirective.findAll(file.viewProvider.contents)
|
|
||||||
.flatMap { it.groupValues[1].split(',').asSequence() }
|
|
||||||
.map { it.trim() }
|
|
||||||
.filter { it.isNotEmpty() }
|
|
||||||
.toList()
|
|
||||||
val baseDir = file.virtualFile?.parent
|
|
||||||
if (baseDir != null) {
|
|
||||||
for (spec in includeSpecs) {
|
|
||||||
val included = baseDir.findFileByRelativePath(spec) ?: continue
|
|
||||||
if (included.path == file.virtualFile?.path) continue
|
|
||||||
if (seen.add(included.path)) {
|
|
||||||
psiManager.findFile(included)?.let { result.add(it) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result.isNotEmpty()) return@runReadAction result
|
|
||||||
|
|
||||||
// Fallback for virtual/light files without a stable parent chain (e.g., tests)
|
|
||||||
val basePath = file.virtualFile?.path ?: return@runReadAction result
|
|
||||||
val scope = GlobalSearchScope.projectScope(file.project)
|
|
||||||
val dFiles = FilenameIndex.getAllFilesByExt(file.project, "d", scope)
|
|
||||||
for (vFile in dFiles) {
|
|
||||||
if (!vFile.name.endsWith(".lyng.d")) continue
|
|
||||||
if (vFile.path == basePath) continue
|
|
||||||
val parentPath = vFile.parent?.path ?: continue
|
|
||||||
if (basePath == parentPath || basePath.startsWith(parentPath.trimEnd('/') + "/")) {
|
|
||||||
if (seen.add(vFile.path)) {
|
|
||||||
psiManager.findFile(vFile)?.let { result.add(it) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result.isNotEmpty()) return@runReadAction result
|
|
||||||
|
|
||||||
// Fallback: scan all Lyng files in project index and filter by .lyng.d
|
|
||||||
val lyngFiles = FileTypeIndex.getFiles(LyngFileType, scope)
|
|
||||||
for (vFile in lyngFiles) {
|
|
||||||
if (!vFile.name.endsWith(".lyng.d")) continue
|
|
||||||
if (vFile.path == basePath) continue
|
|
||||||
if (seen.add(vFile.path)) {
|
|
||||||
psiManager.findFile(vFile)?.let { result.add(it) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result.isNotEmpty()) return@runReadAction result
|
|
||||||
|
|
||||||
// Final fallback: include all .lyng.d files in project scope
|
|
||||||
for (vFile in dFiles) {
|
|
||||||
if (!vFile.name.endsWith(".lyng.d")) continue
|
|
||||||
if (vFile.path == basePath) continue
|
|
||||||
if (seen.add(vFile.path)) {
|
|
||||||
psiManager.findFile(vFile)?.let { result.add(it) }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
fun getDeclarationFiles(file: PsiFile): List<PsiFile> = runReadAction {
|
|
||||||
collectDeclarationFiles(file)
|
|
||||||
}
|
|
||||||
|
|
||||||
fun getBinding(file: PsiFile): BindingSnapshot? = runReadAction {
|
fun getBinding(file: PsiFile): BindingSnapshot? = runReadAction {
|
||||||
getAnalysis(file)?.binding
|
getAnalysis(file)?.binding
|
||||||
}
|
}
|
||||||
@ -145,8 +81,7 @@ object LyngAstManager {
|
|||||||
|
|
||||||
val text = file.viewProvider.contents.toString()
|
val text = file.viewProvider.contents.toString()
|
||||||
val built = try {
|
val built = try {
|
||||||
DocsBootstrap.ensure()
|
val provider = IdeLenientImportProvider.create()
|
||||||
val provider = LyngIdeaImportProvider.create()
|
|
||||||
runBlocking {
|
runBlocking {
|
||||||
LyngLanguageTools.analyze(
|
LyngLanguageTools.analyze(
|
||||||
LyngAnalysisRequest(text = text, fileName = file.name, importProvider = provider)
|
LyngAnalysisRequest(text = text, fileName = file.name, importProvider = provider)
|
||||||
@ -157,38 +92,20 @@ object LyngAstManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (built != null) {
|
if (built != null) {
|
||||||
val isDecl = file.name.endsWith(".lyng.d")
|
val merged = built.mini
|
||||||
val merged = if (!isDecl && built.mini == null) {
|
if (merged != null && !file.name.endsWith(".lyng.d")) {
|
||||||
MiniScript(MiniRange(built.source.startPos, built.source.startPos))
|
|
||||||
} else {
|
|
||||||
built.mini
|
|
||||||
}
|
|
||||||
if (merged != null && !isDecl) {
|
|
||||||
val dFiles = collectDeclarationFiles(file)
|
val dFiles = collectDeclarationFiles(file)
|
||||||
for (df in dFiles) {
|
for (df in dFiles) {
|
||||||
val dMini = getAnalysis(df)?.mini ?: run {
|
val dAnalysis = getAnalysis(df)
|
||||||
val dText = df.viewProvider.contents.toString()
|
val dMini = dAnalysis?.mini ?: continue
|
||||||
try {
|
|
||||||
val provider = LyngIdeaImportProvider.create()
|
|
||||||
runBlocking {
|
|
||||||
LyngLanguageTools.analyze(
|
|
||||||
LyngAnalysisRequest(text = dText, fileName = df.name, importProvider = provider)
|
|
||||||
)
|
|
||||||
}.mini
|
|
||||||
} catch (_: Throwable) {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
} ?: continue
|
|
||||||
merged.declarations.addAll(dMini.declarations)
|
merged.declarations.addAll(dMini.declarations)
|
||||||
merged.imports.addAll(dMini.imports)
|
merged.imports.addAll(dMini.imports)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
val finalAnalysis = if (merged != null) {
|
val finalAnalysis = if (merged != null) {
|
||||||
val mergedImports = DocLookupUtils.canonicalImportedModules(merged, text)
|
|
||||||
built.copy(
|
built.copy(
|
||||||
mini = merged,
|
mini = merged,
|
||||||
importedModules = mergedImports,
|
importedModules = DocLookupUtils.canonicalImportedModules(merged, text)
|
||||||
diagnostics = filterDiagnostics(built.diagnostics, merged, text, mergedImports)
|
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
built
|
built
|
||||||
@ -201,45 +118,4 @@ object LyngAstManager {
|
|||||||
}
|
}
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun filterDiagnostics(
|
|
||||||
diagnostics: List<LyngDiagnostic>,
|
|
||||||
merged: MiniScript,
|
|
||||||
text: String,
|
|
||||||
importedModules: List<String>
|
|
||||||
): List<LyngDiagnostic> {
|
|
||||||
if (diagnostics.isEmpty()) return diagnostics
|
|
||||||
val declaredTopLevel = merged.declarations.map { it.name }.toSet()
|
|
||||||
|
|
||||||
val declaredMembers = linkedSetOf<String>()
|
|
||||||
val aggregatedClasses = DocLookupUtils.aggregateClasses(importedModules, merged)
|
|
||||||
for (cls in aggregatedClasses.values) {
|
|
||||||
cls.members.forEach { declaredMembers.add(it.name) }
|
|
||||||
cls.ctorFields.forEach { declaredMembers.add(it.name) }
|
|
||||||
cls.classFields.forEach { declaredMembers.add(it.name) }
|
|
||||||
}
|
|
||||||
merged.declarations.filterIsInstance<MiniEnumDecl>().forEach { en ->
|
|
||||||
DocLookupUtils.enumToSyntheticClass(en).members.forEach { declaredMembers.add(it.name) }
|
|
||||||
}
|
|
||||||
|
|
||||||
val builtinTopLevel = linkedSetOf<String>()
|
|
||||||
for (mod in importedModules) {
|
|
||||||
BuiltinDocRegistry.docsForModule(mod).forEach { builtinTopLevel.add(it.name) }
|
|
||||||
}
|
|
||||||
|
|
||||||
return diagnostics.filterNot { diag ->
|
|
||||||
val msg = diag.message
|
|
||||||
if (msg.startsWith("unresolved name: ")) {
|
|
||||||
val name = msg.removePrefix("unresolved name: ").trim()
|
|
||||||
name in declaredTopLevel || name in builtinTopLevel || name in implicitBuiltinNames
|
|
||||||
} else if (msg.startsWith("unresolved member: ")) {
|
|
||||||
val name = msg.removePrefix("unresolved member: ").trim()
|
|
||||||
val range = diag.range
|
|
||||||
val dotLeft = if (range != null) DocLookupUtils.findDotLeft(text, range.start) else null
|
|
||||||
dotLeft != null && name in declaredMembers
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user