@fairyhunter13/opentui-core 0.1.113 → 0.1.114
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dev/keypress-debug-renderer.ts +148 -0
- package/dev/keypress-debug.ts +43 -0
- package/dev/print-env-vars.ts +32 -0
- package/dev/test-tmux-graphics-334.sh +68 -0
- package/dev/thai-debug-test.ts +68 -0
- package/docs/development.md +144 -0
- package/package.json +62 -53
- package/scripts/build.ts +400 -0
- package/scripts/publish.ts +60 -0
- package/src/3d/SpriteResourceManager.ts +286 -0
- package/src/3d/SpriteUtils.ts +70 -0
- package/src/3d/TextureUtils.ts +196 -0
- package/src/3d/ThreeRenderable.ts +197 -0
- package/src/3d/WGPURenderer.ts +294 -0
- package/src/3d/animation/ExplodingSpriteEffect.ts +513 -0
- package/src/3d/animation/PhysicsExplodingSpriteEffect.ts +429 -0
- package/src/3d/animation/SpriteAnimator.ts +633 -0
- package/src/3d/animation/SpriteParticleGenerator.ts +435 -0
- package/src/3d/canvas.ts +464 -0
- package/src/3d/index.ts +12 -0
- package/src/3d/physics/PlanckPhysicsAdapter.ts +72 -0
- package/src/3d/physics/RapierPhysicsAdapter.ts +66 -0
- package/src/3d/physics/physics-interface.ts +31 -0
- package/src/3d/shaders/supersampling.wgsl +201 -0
- package/src/3d.ts +3 -0
- package/src/NativeSpanFeed.ts +300 -0
- package/src/Renderable.ts +1704 -0
- package/src/__snapshots__/buffer.test.ts.snap +28 -0
- package/src/animation/Timeline.test.ts +2709 -0
- package/src/animation/Timeline.ts +598 -0
- package/src/ansi.ts +18 -0
- package/src/benchmark/attenuation-benchmark.ts +81 -0
- package/src/benchmark/colormatrix-benchmark.ts +128 -0
- package/src/benchmark/gain-benchmark.ts +80 -0
- package/src/benchmark/latest-all-bench-run.json +707 -0
- package/src/benchmark/latest-async-bench-run.json +336 -0
- package/src/benchmark/latest-default-bench-run.json +657 -0
- package/src/benchmark/latest-large-bench-run.json +707 -0
- package/src/benchmark/latest-quick-bench-run.json +207 -0
- package/src/benchmark/markdown-benchmark.ts +1796 -0
- package/src/benchmark/native-span-feed-async-benchmark.ts +355 -0
- package/src/benchmark/native-span-feed-benchmark.md +56 -0
- package/src/benchmark/native-span-feed-benchmark.ts +596 -0
- package/src/benchmark/native-span-feed-compare.ts +280 -0
- package/src/benchmark/renderer-benchmark.ts +754 -0
- package/src/benchmark/text-table-benchmark.ts +948 -0
- package/src/buffer.test.ts +291 -0
- package/src/buffer.ts +554 -0
- package/src/console.test.ts +612 -0
- package/src/console.ts +1254 -0
- package/src/edit-buffer.test.ts +1769 -0
- package/src/edit-buffer.ts +411 -0
- package/src/editor-view.test.ts +1032 -0
- package/src/editor-view.ts +284 -0
- package/src/examples/ascii-font-selection-demo.ts +245 -0
- package/src/examples/assets/Water_2_M_Normal.jpg +0 -0
- package/src/examples/assets/concrete.png +0 -0
- package/src/examples/assets/crate.png +0 -0
- package/src/examples/assets/crate_emissive.png +0 -0
- package/src/examples/assets/forrest_background.png +0 -0
- package/src/examples/assets/hast-example.json +1018 -0
- package/src/examples/assets/heart.png +0 -0
- package/src/examples/assets/main_char_heavy_attack.png +0 -0
- package/src/examples/assets/main_char_idle.png +0 -0
- package/src/examples/assets/main_char_jump_end.png +0 -0
- package/src/examples/assets/main_char_jump_landing.png +0 -0
- package/src/examples/assets/main_char_jump_start.png +0 -0
- package/src/examples/assets/main_char_run_loop.png +0 -0
- package/src/examples/assets/roughness_map.jpg +0 -0
- package/src/examples/build.ts +115 -0
- package/src/examples/code-demo.ts +924 -0
- package/src/examples/console-demo.ts +358 -0
- package/src/examples/core-plugin-slots-demo.ts +759 -0
- package/src/examples/diff-demo.ts +701 -0
- package/src/examples/draggable-three-demo.ts +259 -0
- package/src/examples/editor-demo.ts +322 -0
- package/src/examples/extmarks-demo.ts +196 -0
- package/src/examples/focus-restore-demo.ts +310 -0
- package/src/examples/fonts.ts +245 -0
- package/src/examples/fractal-shader-demo.ts +268 -0
- package/src/examples/framebuffer-demo.ts +674 -0
- package/src/examples/full-unicode-demo.ts +241 -0
- package/src/examples/golden-star-demo.ts +933 -0
- package/src/examples/grayscale-buffer-demo.ts +249 -0
- package/src/examples/hast-syntax-highlighting-demo.ts +129 -0
- package/src/examples/index.ts +926 -0
- package/src/examples/input-demo.ts +377 -0
- package/src/examples/input-select-layout-demo.ts +425 -0
- package/src/examples/install.sh +143 -0
- package/src/examples/keypress-debug-demo.ts +452 -0
- package/src/examples/lib/HexList.ts +122 -0
- package/src/examples/lib/PaletteGrid.ts +125 -0
- package/src/examples/lib/standalone-keys.ts +25 -0
- package/src/examples/lib/tab-controller.ts +243 -0
- package/src/examples/lights-phong-demo.ts +290 -0
- package/src/examples/link-demo.ts +220 -0
- package/src/examples/live-state-demo.ts +480 -0
- package/src/examples/markdown-demo.ts +725 -0
- package/src/examples/mouse-interaction-demo.ts +428 -0
- package/src/examples/nested-zindex-demo.ts +357 -0
- package/src/examples/opacity-example.ts +235 -0
- package/src/examples/opentui-demo.ts +1057 -0
- package/src/examples/physx-planck-2d-demo.ts +623 -0
- package/src/examples/physx-rapier-2d-demo.ts +655 -0
- package/src/examples/relative-positioning-demo.ts +323 -0
- package/src/examples/scroll-example.ts +214 -0
- package/src/examples/scrollbox-mouse-test.ts +112 -0
- package/src/examples/scrollbox-overlay-hit-test.ts +206 -0
- package/src/examples/select-demo.ts +237 -0
- package/src/examples/shader-cube-demo.ts +1015 -0
- package/src/examples/simple-layout-example.ts +591 -0
- package/src/examples/slider-demo.ts +617 -0
- package/src/examples/split-mode-demo.ts +453 -0
- package/src/examples/sprite-animation-demo.ts +443 -0
- package/src/examples/sprite-particle-generator-demo.ts +486 -0
- package/src/examples/static-sprite-demo.ts +193 -0
- package/src/examples/sticky-scroll-example.ts +308 -0
- package/src/examples/styled-text-demo.ts +282 -0
- package/src/examples/tab-select-demo.ts +219 -0
- package/src/examples/terminal-title.ts +29 -0
- package/src/examples/terminal.ts +305 -0
- package/src/examples/text-node-demo.ts +416 -0
- package/src/examples/text-selection-demo.ts +377 -0
- package/src/examples/text-table-demo.ts +503 -0
- package/src/examples/text-truncation-demo.ts +481 -0
- package/src/examples/text-wrap.ts +757 -0
- package/src/examples/texture-loading-demo.ts +259 -0
- package/src/examples/timeline-example.ts +670 -0
- package/src/examples/transparency-demo.ts +400 -0
- package/src/examples/vnode-composition-demo.ts +404 -0
- package/src/examples/wide-grapheme-overlay-demo.ts +280 -0
- package/src/index.ts +24 -0
- package/src/lib/KeyHandler.integration.test.ts +292 -0
- package/src/lib/KeyHandler.stopPropagation.test.ts +289 -0
- package/src/lib/KeyHandler.test.ts +662 -0
- package/src/lib/KeyHandler.ts +222 -0
- package/src/lib/RGBA.test.ts +984 -0
- package/src/lib/RGBA.ts +204 -0
- package/src/lib/ascii.font.ts +330 -0
- package/src/lib/border.test.ts +83 -0
- package/src/lib/border.ts +170 -0
- package/src/lib/bunfs.test.ts +27 -0
- package/src/lib/bunfs.ts +18 -0
- package/src/lib/clipboard.test.ts +41 -0
- package/src/lib/clipboard.ts +47 -0
- package/src/lib/clock.ts +35 -0
- package/src/lib/data-paths.test.ts +133 -0
- package/src/lib/data-paths.ts +109 -0
- package/src/lib/debounce.ts +106 -0
- package/src/lib/detect-links.test.ts +98 -0
- package/src/lib/detect-links.ts +56 -0
- package/src/lib/env.test.ts +228 -0
- package/src/lib/env.ts +209 -0
- package/src/lib/extmarks-history.ts +51 -0
- package/src/lib/extmarks-multiwidth.test.ts +322 -0
- package/src/lib/extmarks.test.ts +3457 -0
- package/src/lib/extmarks.ts +843 -0
- package/src/lib/fonts/block.json +405 -0
- package/src/lib/fonts/grid.json +265 -0
- package/src/lib/fonts/huge.json +741 -0
- package/src/lib/fonts/pallet.json +314 -0
- package/src/lib/fonts/shade.json +591 -0
- package/src/lib/fonts/slick.json +321 -0
- package/src/lib/fonts/tiny.json +69 -0
- package/src/lib/hast-styled-text.ts +59 -0
- package/src/lib/index.ts +21 -0
- package/src/lib/keymapping.test.ts +317 -0
- package/src/lib/keymapping.ts +115 -0
- package/src/lib/objects-in-viewport.test.ts +787 -0
- package/src/lib/objects-in-viewport.ts +153 -0
- package/src/lib/output.capture.ts +58 -0
- package/src/lib/parse.keypress-kitty.protocol.test.ts +340 -0
- package/src/lib/parse.keypress-kitty.test.ts +663 -0
- package/src/lib/parse.keypress-kitty.ts +439 -0
- package/src/lib/parse.keypress.test.ts +1849 -0
- package/src/lib/parse.keypress.ts +397 -0
- package/src/lib/parse.mouse.test.ts +552 -0
- package/src/lib/parse.mouse.ts +232 -0
- package/src/lib/paste.ts +16 -0
- package/src/lib/queue.ts +65 -0
- package/src/lib/renderable.validations.test.ts +87 -0
- package/src/lib/renderable.validations.ts +83 -0
- package/src/lib/scroll-acceleration.ts +98 -0
- package/src/lib/selection.ts +240 -0
- package/src/lib/singleton.ts +28 -0
- package/src/lib/stdin-parser.test.ts +2290 -0
- package/src/lib/stdin-parser.ts +1810 -0
- package/src/lib/styled-text.ts +178 -0
- package/src/lib/terminal-capability-detection.test.ts +202 -0
- package/src/lib/terminal-capability-detection.ts +79 -0
- package/src/lib/terminal-palette.test.ts +878 -0
- package/src/lib/terminal-palette.ts +383 -0
- package/src/lib/tree-sitter/assets/README.md +118 -0
- package/src/lib/tree-sitter/assets/update.ts +334 -0
- package/src/lib/tree-sitter/assets.d.ts +9 -0
- package/src/lib/tree-sitter/cache.test.ts +273 -0
- package/src/lib/tree-sitter/client.test.ts +1165 -0
- package/src/lib/tree-sitter/client.ts +607 -0
- package/src/lib/tree-sitter/default-parsers.ts +86 -0
- package/src/lib/tree-sitter/download-utils.ts +148 -0
- package/src/lib/tree-sitter/index.ts +28 -0
- package/src/lib/tree-sitter/parser.worker.ts +1042 -0
- package/src/lib/tree-sitter/parsers-config.ts +81 -0
- package/src/lib/tree-sitter/resolve-ft.test.ts +55 -0
- package/src/lib/tree-sitter/resolve-ft.ts +189 -0
- package/src/lib/tree-sitter/types.ts +82 -0
- package/src/lib/tree-sitter-styled-text.test.ts +1253 -0
- package/src/lib/tree-sitter-styled-text.ts +306 -0
- package/src/lib/validate-dir-name.ts +55 -0
- package/src/lib/yoga.options.test.ts +628 -0
- package/src/lib/yoga.options.ts +346 -0
- package/src/plugins/core-slot.ts +579 -0
- package/src/plugins/registry.ts +402 -0
- package/src/plugins/types.ts +46 -0
- package/src/post/effects.ts +930 -0
- package/src/post/filters.ts +489 -0
- package/src/post/matrices.ts +288 -0
- package/src/renderables/ASCIIFont.ts +219 -0
- package/src/renderables/Box.test.ts +205 -0
- package/src/renderables/Box.ts +326 -0
- package/src/renderables/Code.test.ts +2062 -0
- package/src/renderables/Code.ts +357 -0
- package/src/renderables/Diff.regression.test.ts +226 -0
- package/src/renderables/Diff.test.ts +3101 -0
- package/src/renderables/Diff.ts +1211 -0
- package/src/renderables/EditBufferRenderable.test.ts +288 -0
- package/src/renderables/EditBufferRenderable.ts +1166 -0
- package/src/renderables/FrameBuffer.ts +47 -0
- package/src/renderables/Input.test.ts +1228 -0
- package/src/renderables/Input.ts +247 -0
- package/src/renderables/LineNumberRenderable.ts +724 -0
- package/src/renderables/Markdown.ts +1393 -0
- package/src/renderables/ScrollBar.ts +422 -0
- package/src/renderables/ScrollBox.ts +883 -0
- package/src/renderables/Select.test.ts +1033 -0
- package/src/renderables/Select.ts +524 -0
- package/src/renderables/Slider.test.ts +456 -0
- package/src/renderables/Slider.ts +342 -0
- package/src/renderables/TabSelect.test.ts +197 -0
- package/src/renderables/TabSelect.ts +455 -0
- package/src/renderables/Text.selection-buffer.test.ts +123 -0
- package/src/renderables/Text.test.ts +2660 -0
- package/src/renderables/Text.ts +147 -0
- package/src/renderables/TextBufferRenderable.ts +518 -0
- package/src/renderables/TextNode.test.ts +1058 -0
- package/src/renderables/TextNode.ts +325 -0
- package/src/renderables/TextTable.test.ts +1421 -0
- package/src/renderables/TextTable.ts +1344 -0
- package/src/renderables/Textarea.ts +430 -0
- package/src/renderables/TimeToFirstDraw.ts +89 -0
- package/src/renderables/__snapshots__/Code.test.ts.snap +13 -0
- package/src/renderables/__snapshots__/Diff.test.ts.snap +785 -0
- package/src/renderables/__snapshots__/Text.test.ts.snap +421 -0
- package/src/renderables/__snapshots__/TextTable.test.ts.snap +215 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox-simple.test.ts +144 -0
- package/src/renderables/__tests__/LineNumberRenderable.scrollbox.test.ts +816 -0
- package/src/renderables/__tests__/LineNumberRenderable.test.ts +1865 -0
- package/src/renderables/__tests__/LineNumberRenderable.wrapping.test.ts +85 -0
- package/src/renderables/__tests__/Markdown.code-colors.test.ts +242 -0
- package/src/renderables/__tests__/Markdown.test.ts +2518 -0
- package/src/renderables/__tests__/MultiRenderable.selection.test.ts +87 -0
- package/src/renderables/__tests__/Textarea.buffer.test.ts +682 -0
- package/src/renderables/__tests__/Textarea.destroyed-events.test.ts +675 -0
- package/src/renderables/__tests__/Textarea.editing.test.ts +2041 -0
- package/src/renderables/__tests__/Textarea.error-handling.test.ts +35 -0
- package/src/renderables/__tests__/Textarea.events.test.ts +738 -0
- package/src/renderables/__tests__/Textarea.highlights.test.ts +590 -0
- package/src/renderables/__tests__/Textarea.keybinding.test.ts +3149 -0
- package/src/renderables/__tests__/Textarea.paste.test.ts +357 -0
- package/src/renderables/__tests__/Textarea.rendering.test.ts +1866 -0
- package/src/renderables/__tests__/Textarea.scroll.test.ts +733 -0
- package/src/renderables/__tests__/Textarea.selection.test.ts +1590 -0
- package/src/renderables/__tests__/Textarea.stress.test.ts +670 -0
- package/src/renderables/__tests__/Textarea.undo-redo.test.ts +383 -0
- package/src/renderables/__tests__/Textarea.visual-lines.test.ts +310 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.code.test.ts.snap +221 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox-simple.test.ts.snap +89 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.scrollbox.test.ts.snap +457 -0
- package/src/renderables/__tests__/__snapshots__/LineNumberRenderable.test.ts.snap +158 -0
- package/src/renderables/__tests__/__snapshots__/Textarea.rendering.test.ts.snap +387 -0
- package/src/renderables/__tests__/markdown-parser.test.ts +217 -0
- package/src/renderables/__tests__/renderable-test-utils.ts +60 -0
- package/src/renderables/composition/README.md +8 -0
- package/src/renderables/composition/VRenderable.ts +32 -0
- package/src/renderables/composition/constructs.ts +127 -0
- package/src/renderables/composition/vnode.ts +289 -0
- package/src/renderables/index.ts +23 -0
- package/src/renderables/markdown-parser.ts +66 -0
- package/src/renderer.ts +2681 -0
- package/src/runtime-plugin-support.ts +39 -0
- package/src/runtime-plugin.ts +615 -0
- package/src/syntax-style.test.ts +841 -0
- package/src/syntax-style.ts +257 -0
- package/src/testing/README.md +210 -0
- package/src/testing/capture-spans.test.ts +194 -0
- package/src/testing/integration.test.ts +276 -0
- package/src/testing/manual-clock.ts +117 -0
- package/src/testing/mock-keys.test.ts +1378 -0
- package/src/testing/mock-keys.ts +457 -0
- package/src/testing/mock-mouse.test.ts +218 -0
- package/src/testing/mock-mouse.ts +247 -0
- package/src/testing/mock-tree-sitter-client.ts +73 -0
- package/src/testing/spy.ts +13 -0
- package/src/testing/test-recorder.test.ts +415 -0
- package/src/testing/test-recorder.ts +145 -0
- package/src/testing/test-renderer.ts +132 -0
- package/src/testing.ts +7 -0
- package/src/tests/__snapshots__/absolute-positioning.snapshot.test.ts.snap +481 -0
- package/src/tests/__snapshots__/renderable.snapshot.test.ts.snap +19 -0
- package/src/tests/__snapshots__/scrollbox.test.ts.snap +29 -0
- package/src/tests/absolute-positioning.snapshot.test.ts +638 -0
- package/src/tests/allocator-stats.test.ts +38 -0
- package/src/tests/destroy-during-render.test.ts +200 -0
- package/src/tests/destroy-on-exit.fixture.ts +36 -0
- package/src/tests/destroy-on-exit.test.ts +41 -0
- package/src/tests/hover-cursor.test.ts +98 -0
- package/src/tests/native-span-feed-async.test.ts +173 -0
- package/src/tests/native-span-feed-close.test.ts +120 -0
- package/src/tests/native-span-feed-coverage.test.ts +227 -0
- package/src/tests/native-span-feed-edge-cases.test.ts +352 -0
- package/src/tests/native-span-feed-use-after-free.test.ts +45 -0
- package/src/tests/opacity.test.ts +123 -0
- package/src/tests/renderable.snapshot.test.ts +524 -0
- package/src/tests/renderable.test.ts +1281 -0
- package/src/tests/renderer.clock.test.ts +158 -0
- package/src/tests/renderer.console-startup.test.ts +185 -0
- package/src/tests/renderer.control.test.ts +425 -0
- package/src/tests/renderer.core-slot-binding.test.ts +952 -0
- package/src/tests/renderer.cursor.test.ts +26 -0
- package/src/tests/renderer.destroy-during-render.test.ts +147 -0
- package/src/tests/renderer.focus-restore.test.ts +257 -0
- package/src/tests/renderer.focus.test.ts +294 -0
- package/src/tests/renderer.idle.test.ts +219 -0
- package/src/tests/renderer.input.test.ts +2237 -0
- package/src/tests/renderer.kitty-flags.test.ts +195 -0
- package/src/tests/renderer.mouse.test.ts +1274 -0
- package/src/tests/renderer.palette.test.ts +629 -0
- package/src/tests/renderer.selection.test.ts +49 -0
- package/src/tests/renderer.slot-registry.test.ts +684 -0
- package/src/tests/renderer.useMouse.test.ts +47 -0
- package/src/tests/runtime-plugin-node-modules-cycle.fixture.ts +76 -0
- package/src/tests/runtime-plugin-node-modules-mjs.fixture.ts +43 -0
- package/src/tests/runtime-plugin-node-modules-no-bare-rewrite.fixture.ts +67 -0
- package/src/tests/runtime-plugin-node-modules-package-type-cache.fixture.ts +72 -0
- package/src/tests/runtime-plugin-node-modules-runtime-specifier.fixture.ts +44 -0
- package/src/tests/runtime-plugin-node-modules-scoped-package-bare-rewrite.fixture.ts +85 -0
- package/src/tests/runtime-plugin-path-alias.fixture.ts +43 -0
- package/src/tests/runtime-plugin-resolve-roots.fixture.ts +65 -0
- package/src/tests/runtime-plugin-support.fixture.ts +11 -0
- package/src/tests/runtime-plugin-support.test.ts +19 -0
- package/src/tests/runtime-plugin-windows-file-url.fixture.ts +30 -0
- package/src/tests/runtime-plugin.fixture.ts +40 -0
- package/src/tests/runtime-plugin.test.ts +354 -0
- package/src/tests/scrollbox-culling-bug.test.ts +114 -0
- package/src/tests/scrollbox-hitgrid-resize.test.ts +136 -0
- package/src/tests/scrollbox-hitgrid.test.ts +909 -0
- package/src/tests/scrollbox.test.ts +1530 -0
- package/src/tests/wrap-resize-perf.test.ts +276 -0
- package/src/tests/yoga-setters.test.ts +921 -0
- package/src/text-buffer-view.test.ts +705 -0
- package/src/text-buffer-view.ts +189 -0
- package/src/text-buffer.test.ts +347 -0
- package/src/text-buffer.ts +250 -0
- package/src/types.ts +161 -0
- package/src/utils.ts +88 -0
- package/src/zig/ansi.zig +268 -0
- package/src/zig/bench/README.md +50 -0
- package/src/zig/bench/buffer-draw-text-buffer_bench.zig +887 -0
- package/src/zig/bench/edit-buffer_bench.zig +476 -0
- package/src/zig/bench/native-span-feed_bench.zig +100 -0
- package/src/zig/bench/rope-markers_bench.zig +713 -0
- package/src/zig/bench/rope_bench.zig +514 -0
- package/src/zig/bench/styled-text_bench.zig +470 -0
- package/src/zig/bench/text-buffer-coords_bench.zig +362 -0
- package/src/zig/bench/text-buffer-view_bench.zig +459 -0
- package/src/zig/bench/text-chunk-graphemes_bench.zig +273 -0
- package/src/zig/bench/utf8_bench.zig +799 -0
- package/src/zig/bench-utils.zig +431 -0
- package/src/zig/bench.zig +217 -0
- package/src/zig/buffer-methods.zig +211 -0
- package/src/zig/buffer.zig +2281 -0
- package/src/zig/build.zig +289 -0
- package/src/zig/build.zig.zon +16 -0
- package/src/zig/edit-buffer.zig +825 -0
- package/src/zig/editor-view.zig +802 -0
- package/src/zig/event-bus.zig +13 -0
- package/src/zig/event-emitter.zig +65 -0
- package/src/zig/file-logger.zig +92 -0
- package/src/zig/grapheme.zig +599 -0
- package/src/zig/lib.zig +1854 -0
- package/src/zig/link.zig +333 -0
- package/src/zig/logger.zig +43 -0
- package/src/zig/mem-registry.zig +125 -0
- package/src/zig/native-span-feed-bench-lib.zig +7 -0
- package/src/zig/native-span-feed.zig +708 -0
- package/src/zig/renderer.zig +1393 -0
- package/src/zig/rope.zig +1220 -0
- package/src/zig/syntax-style.zig +161 -0
- package/src/zig/terminal.zig +987 -0
- package/src/zig/test.zig +72 -0
- package/src/zig/tests/README.md +18 -0
- package/src/zig/tests/buffer-methods_test.zig +1109 -0
- package/src/zig/tests/buffer_test.zig +2557 -0
- package/src/zig/tests/edit-buffer-history_test.zig +271 -0
- package/src/zig/tests/edit-buffer_test.zig +1689 -0
- package/src/zig/tests/editor-view_test.zig +3299 -0
- package/src/zig/tests/event-emitter_test.zig +249 -0
- package/src/zig/tests/grapheme_test.zig +1304 -0
- package/src/zig/tests/link_test.zig +190 -0
- package/src/zig/tests/mem-registry_test.zig +473 -0
- package/src/zig/tests/memory_leak_regression_test.zig +159 -0
- package/src/zig/tests/native-span-feed_test.zig +1264 -0
- package/src/zig/tests/renderer_test.zig +1017 -0
- package/src/zig/tests/rope-nested_test.zig +712 -0
- package/src/zig/tests/rope_fuzz_test.zig +238 -0
- package/src/zig/tests/rope_test.zig +2362 -0
- package/src/zig/tests/segment-merge.test.zig +148 -0
- package/src/zig/tests/syntax-style_test.zig +557 -0
- package/src/zig/tests/terminal_test.zig +754 -0
- package/src/zig/tests/text-buffer-drawing_test.zig +3237 -0
- package/src/zig/tests/text-buffer-highlights_test.zig +666 -0
- package/src/zig/tests/text-buffer-iterators_test.zig +776 -0
- package/src/zig/tests/text-buffer-segment_test.zig +320 -0
- package/src/zig/tests/text-buffer-selection_test.zig +1035 -0
- package/src/zig/tests/text-buffer-selection_viewport_test.zig +358 -0
- package/src/zig/tests/text-buffer-view_test.zig +3649 -0
- package/src/zig/tests/text-buffer_test.zig +2191 -0
- package/src/zig/tests/unicode-width-map.zon +3909 -0
- package/src/zig/tests/utf8_no_zwj_test.zig +260 -0
- package/src/zig/tests/utf8_test.zig +4057 -0
- package/src/zig/tests/utf8_wcwidth_cursor_test.zig +267 -0
- package/src/zig/tests/utf8_wcwidth_test.zig +357 -0
- package/src/zig/tests/word-wrap-editing_test.zig +498 -0
- package/src/zig/tests/wrap-cache-perf_test.zig +113 -0
- package/src/zig/text-buffer-iterators.zig +499 -0
- package/src/zig/text-buffer-segment.zig +404 -0
- package/src/zig/text-buffer-view.zig +1371 -0
- package/src/zig/text-buffer.zig +1180 -0
- package/src/zig/utf8.zig +1948 -0
- package/src/zig/utils.zig +9 -0
- package/src/zig-structs.ts +261 -0
- package/src/zig.ts +3884 -0
- package/tsconfig.build.json +24 -0
- package/tsconfig.json +27 -0
- package/3d/SpriteResourceManager.d.ts +0 -74
- package/3d/SpriteUtils.d.ts +0 -13
- package/3d/TextureUtils.d.ts +0 -24
- package/3d/ThreeRenderable.d.ts +0 -40
- package/3d/WGPURenderer.d.ts +0 -61
- package/3d/animation/ExplodingSpriteEffect.d.ts +0 -71
- package/3d/animation/PhysicsExplodingSpriteEffect.d.ts +0 -76
- package/3d/animation/SpriteAnimator.d.ts +0 -124
- package/3d/animation/SpriteParticleGenerator.d.ts +0 -62
- package/3d/canvas.d.ts +0 -44
- package/3d/index.d.ts +0 -12
- package/3d/physics/PlanckPhysicsAdapter.d.ts +0 -19
- package/3d/physics/RapierPhysicsAdapter.d.ts +0 -19
- package/3d/physics/physics-interface.d.ts +0 -27
- package/3d.d.ts +0 -2
- package/3d.js +0 -34041
- package/3d.js.map +0 -155
- package/LICENSE +0 -21
- package/NativeSpanFeed.d.ts +0 -41
- package/Renderable.d.ts +0 -334
- package/animation/Timeline.d.ts +0 -126
- package/ansi.d.ts +0 -13
- package/buffer.d.ts +0 -111
- package/console.d.ts +0 -144
- package/edit-buffer.d.ts +0 -98
- package/editor-view.d.ts +0 -73
- package/index-9vwc3fg6.js +0 -12260
- package/index-9vwc3fg6.js.map +0 -42
- package/index-dcj62y8t.js +0 -20614
- package/index-dcj62y8t.js.map +0 -67
- package/index-f7n39gpy.js +0 -411
- package/index-f7n39gpy.js.map +0 -10
- package/index.d.ts +0 -23
- package/index.js +0 -478
- package/index.js.map +0 -9
- package/lib/KeyHandler.d.ts +0 -61
- package/lib/RGBA.d.ts +0 -25
- package/lib/ascii.font.d.ts +0 -508
- package/lib/border.d.ts +0 -51
- package/lib/bunfs.d.ts +0 -7
- package/lib/clipboard.d.ts +0 -17
- package/lib/clock.d.ts +0 -15
- package/lib/data-paths.d.ts +0 -26
- package/lib/debounce.d.ts +0 -42
- package/lib/detect-links.d.ts +0 -6
- package/lib/env.d.ts +0 -42
- package/lib/extmarks-history.d.ts +0 -17
- package/lib/extmarks.d.ts +0 -89
- package/lib/hast-styled-text.d.ts +0 -17
- package/lib/index.d.ts +0 -21
- package/lib/keymapping.d.ts +0 -25
- package/lib/objects-in-viewport.d.ts +0 -24
- package/lib/output.capture.d.ts +0 -24
- package/lib/parse.keypress-kitty.d.ts +0 -2
- package/lib/parse.keypress.d.ts +0 -26
- package/lib/parse.mouse.d.ts +0 -30
- package/lib/paste.d.ts +0 -7
- package/lib/queue.d.ts +0 -15
- package/lib/renderable.validations.d.ts +0 -12
- package/lib/scroll-acceleration.d.ts +0 -43
- package/lib/selection.d.ts +0 -63
- package/lib/singleton.d.ts +0 -7
- package/lib/stdin-parser.d.ts +0 -87
- package/lib/styled-text.d.ts +0 -63
- package/lib/terminal-capability-detection.d.ts +0 -30
- package/lib/terminal-palette.d.ts +0 -50
- package/lib/tree-sitter/assets/update.d.ts +0 -11
- package/lib/tree-sitter/client.d.ts +0 -47
- package/lib/tree-sitter/default-parsers.d.ts +0 -2
- package/lib/tree-sitter/download-utils.d.ts +0 -21
- package/lib/tree-sitter/index.d.ts +0 -8
- package/lib/tree-sitter/parser.worker.d.ts +0 -1
- package/lib/tree-sitter/parsers-config.d.ts +0 -53
- package/lib/tree-sitter/resolve-ft.d.ts +0 -5
- package/lib/tree-sitter/types.d.ts +0 -82
- package/lib/tree-sitter-styled-text.d.ts +0 -14
- package/lib/validate-dir-name.d.ts +0 -1
- package/lib/yoga.options.d.ts +0 -32
- package/parser.worker.js +0 -899
- package/parser.worker.js.map +0 -12
- package/plugins/core-slot.d.ts +0 -72
- package/plugins/registry.d.ts +0 -42
- package/plugins/types.d.ts +0 -34
- package/post/effects.d.ts +0 -147
- package/post/filters.d.ts +0 -65
- package/post/matrices.d.ts +0 -20
- package/renderables/ASCIIFont.d.ts +0 -52
- package/renderables/Box.d.ts +0 -81
- package/renderables/Code.d.ts +0 -78
- package/renderables/Diff.d.ts +0 -142
- package/renderables/EditBufferRenderable.d.ts +0 -237
- package/renderables/FrameBuffer.d.ts +0 -16
- package/renderables/Input.d.ts +0 -67
- package/renderables/LineNumberRenderable.d.ts +0 -78
- package/renderables/Markdown.d.ts +0 -185
- package/renderables/ScrollBar.d.ts +0 -77
- package/renderables/ScrollBox.d.ts +0 -124
- package/renderables/Select.d.ts +0 -115
- package/renderables/Slider.d.ts +0 -47
- package/renderables/TabSelect.d.ts +0 -96
- package/renderables/Text.d.ts +0 -36
- package/renderables/TextBufferRenderable.d.ts +0 -105
- package/renderables/TextNode.d.ts +0 -91
- package/renderables/TextTable.d.ts +0 -140
- package/renderables/Textarea.d.ts +0 -63
- package/renderables/TimeToFirstDraw.d.ts +0 -24
- package/renderables/__tests__/renderable-test-utils.d.ts +0 -12
- package/renderables/composition/VRenderable.d.ts +0 -16
- package/renderables/composition/constructs.d.ts +0 -35
- package/renderables/composition/vnode.d.ts +0 -46
- package/renderables/index.d.ts +0 -23
- package/renderables/markdown-parser.d.ts +0 -10
- package/renderer.d.ts +0 -419
- package/runtime-plugin-support.d.ts +0 -3
- package/runtime-plugin-support.js +0 -29
- package/runtime-plugin-support.js.map +0 -10
- package/runtime-plugin.d.ts +0 -16
- package/runtime-plugin.js +0 -16
- package/runtime-plugin.js.map +0 -9
- package/syntax-style.d.ts +0 -54
- package/testing/manual-clock.d.ts +0 -17
- package/testing/mock-keys.d.ts +0 -81
- package/testing/mock-mouse.d.ts +0 -38
- package/testing/mock-tree-sitter-client.d.ts +0 -23
- package/testing/spy.d.ts +0 -7
- package/testing/test-recorder.d.ts +0 -61
- package/testing/test-renderer.d.ts +0 -23
- package/testing.d.ts +0 -6
- package/testing.js +0 -697
- package/testing.js.map +0 -15
- package/text-buffer-view.d.ts +0 -42
- package/text-buffer.d.ts +0 -67
- package/types.d.ts +0 -139
- package/utils.d.ts +0 -14
- package/zig-structs.d.ts +0 -155
- package/zig.d.ts +0 -353
- /package/{assets → src/lib/tree-sitter/assets}/javascript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/javascript/tree-sitter-javascript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/injections.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown/tree-sitter-markdown.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/markdown_inline/tree-sitter-markdown_inline.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/typescript/tree-sitter-typescript.wasm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/highlights.scm +0 -0
- /package/{assets → src/lib/tree-sitter/assets}/zig/tree-sitter-zig.wasm +0 -0
|
@@ -0,0 +1,431 @@
|
|
|
1
|
+
const std = @import("std");
|
|
2
|
+
|
|
3
|
+
pub fn matchesBenchFilter(bench_name: []const u8, filter: ?[]const u8) bool {
|
|
4
|
+
if (filter == null) return true;
|
|
5
|
+
const filter_str = filter.?;
|
|
6
|
+
if (filter_str.len == 0) return true;
|
|
7
|
+
|
|
8
|
+
var i: usize = 0;
|
|
9
|
+
while (i + filter_str.len <= bench_name.len) : (i += 1) {
|
|
10
|
+
var matches = true;
|
|
11
|
+
for (filter_str, 0..) |filter_char, j| {
|
|
12
|
+
const bench_char = bench_name[i + j];
|
|
13
|
+
const filter_lower = if (filter_char >= 'A' and filter_char <= 'Z') filter_char + 32 else filter_char;
|
|
14
|
+
const bench_lower = if (bench_char >= 'A' and bench_char <= 'Z') bench_char + 32 else bench_char;
|
|
15
|
+
if (filter_lower != bench_lower) {
|
|
16
|
+
matches = false;
|
|
17
|
+
break;
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
if (matches) return true;
|
|
21
|
+
}
|
|
22
|
+
return false;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
pub const MemStat = struct {
|
|
26
|
+
name: []const u8,
|
|
27
|
+
bytes: usize,
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
pub const BenchResult = struct {
|
|
31
|
+
name: []const u8,
|
|
32
|
+
min_ns: u64,
|
|
33
|
+
avg_ns: u64,
|
|
34
|
+
max_ns: u64,
|
|
35
|
+
total_ns: u64,
|
|
36
|
+
iterations: usize,
|
|
37
|
+
mem_stats: ?[]const MemStat,
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
/// Timing statistics collected during benchmark iterations
|
|
41
|
+
pub const BenchStats = struct {
|
|
42
|
+
min_ns: u64 = std.math.maxInt(u64),
|
|
43
|
+
max_ns: u64 = 0,
|
|
44
|
+
total_ns: u64 = 0,
|
|
45
|
+
count: usize = 0,
|
|
46
|
+
|
|
47
|
+
pub fn record(self: *BenchStats, elapsed_ns: u64) void {
|
|
48
|
+
self.min_ns = @min(self.min_ns, elapsed_ns);
|
|
49
|
+
self.max_ns = @max(self.max_ns, elapsed_ns);
|
|
50
|
+
self.total_ns += elapsed_ns;
|
|
51
|
+
self.count += 1;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
pub fn avg(self: *const BenchStats) u64 {
|
|
55
|
+
if (self.count == 0) return 0;
|
|
56
|
+
return self.total_ns / self.count;
|
|
57
|
+
}
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
/// Helper for running benchmark iterations with timing
|
|
61
|
+
pub const BenchRunner = struct {
|
|
62
|
+
allocator: std.mem.Allocator,
|
|
63
|
+
results: std.ArrayListUnmanaged(BenchResult),
|
|
64
|
+
|
|
65
|
+
pub fn init(allocator: std.mem.Allocator) BenchRunner {
|
|
66
|
+
return .{
|
|
67
|
+
.allocator = allocator,
|
|
68
|
+
.results = .{},
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
/// Add a benchmark result from collected stats
|
|
73
|
+
pub fn addResult(
|
|
74
|
+
self: *BenchRunner,
|
|
75
|
+
name: []const u8,
|
|
76
|
+
stats: BenchStats,
|
|
77
|
+
mem_stats: ?[]const MemStat,
|
|
78
|
+
) !void {
|
|
79
|
+
try self.results.append(self.allocator, BenchResult{
|
|
80
|
+
.name = name,
|
|
81
|
+
.min_ns = stats.min_ns,
|
|
82
|
+
.avg_ns = stats.avg(),
|
|
83
|
+
.max_ns = stats.max_ns,
|
|
84
|
+
.total_ns = stats.total_ns,
|
|
85
|
+
.iterations = stats.count,
|
|
86
|
+
.mem_stats = mem_stats,
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
/// Convenience: run a simple benchmark with the given function
|
|
91
|
+
pub fn bench(
|
|
92
|
+
self: *BenchRunner,
|
|
93
|
+
name: []const u8,
|
|
94
|
+
iterations: usize,
|
|
95
|
+
comptime benchFn: anytype,
|
|
96
|
+
args: anytype,
|
|
97
|
+
) !void {
|
|
98
|
+
var stats = BenchStats{};
|
|
99
|
+
var iter: usize = 0;
|
|
100
|
+
while (iter < iterations) : (iter += 1) {
|
|
101
|
+
var timer = try std.time.Timer.start();
|
|
102
|
+
@call(.auto, benchFn, args);
|
|
103
|
+
stats.record(timer.read());
|
|
104
|
+
}
|
|
105
|
+
try self.addResult(name, stats, null);
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/// Get the results slice (caller owns memory via arena)
|
|
109
|
+
pub fn finish(self: *BenchRunner) ![]BenchResult {
|
|
110
|
+
return try self.results.toOwnedSlice(self.allocator);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/// Append results from another runner or slice
|
|
114
|
+
pub fn appendSlice(self: *BenchRunner, other_results: []const BenchResult) !void {
|
|
115
|
+
try self.results.appendSlice(self.allocator, other_results);
|
|
116
|
+
}
|
|
117
|
+
};
|
|
118
|
+
|
|
119
|
+
/// Create a stdout writer with buffer for benchmark output
|
|
120
|
+
pub const StdoutWriter = struct {
|
|
121
|
+
buffer: [4096]u8 = undefined,
|
|
122
|
+
writer: std.fs.File.Writer = undefined,
|
|
123
|
+
|
|
124
|
+
pub fn init() StdoutWriter {
|
|
125
|
+
var self = StdoutWriter{};
|
|
126
|
+
self.writer = std.fs.File.stdout().writer(&self.buffer);
|
|
127
|
+
return self;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
pub fn interface(self: *StdoutWriter) *std.Io.Writer {
|
|
131
|
+
return &self.writer.interface;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
pub fn print(self: *StdoutWriter, comptime fmt: []const u8, args: anytype) !void {
|
|
135
|
+
try self.writer.interface.print(fmt, args);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
pub fn flush(self: *StdoutWriter) !void {
|
|
139
|
+
try self.writer.interface.flush();
|
|
140
|
+
}
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
pub fn formatDuration(ns: u64) struct { value: f64, unit: []const u8, color: []const u8 } {
|
|
144
|
+
if (ns < 1_000) {
|
|
145
|
+
// Bright green for nanoseconds
|
|
146
|
+
return .{ .value = @as(f64, @floatFromInt(ns)), .unit = "ns", .color = "\x1b[92m" };
|
|
147
|
+
} else if (ns < 1_000_000) {
|
|
148
|
+
// Normal green for microseconds
|
|
149
|
+
return .{ .value = @as(f64, @floatFromInt(ns)) / 1_000.0, .unit = "us", .color = "\x1b[32m" };
|
|
150
|
+
} else if (ns < 1_000_000_000) {
|
|
151
|
+
const ms = @as(f64, @floatFromInt(ns)) / 1_000_000.0;
|
|
152
|
+
if (ms < 1.0) {
|
|
153
|
+
// Normal green for < 1ms
|
|
154
|
+
return .{ .value = ms, .unit = "ms", .color = "\x1b[32m" };
|
|
155
|
+
} else if (ms < 3.0) {
|
|
156
|
+
// Yellow to red gradient from 1ms to 3ms
|
|
157
|
+
if (ms < 1.5) {
|
|
158
|
+
return .{ .value = ms, .unit = "ms", .color = "\x1b[33m" }; // Yellow
|
|
159
|
+
} else if (ms < 2.0) {
|
|
160
|
+
return .{ .value = ms, .unit = "ms", .color = "\x1b[38;5;208m" }; // Orange
|
|
161
|
+
} else if (ms < 2.5) {
|
|
162
|
+
return .{ .value = ms, .unit = "ms", .color = "\x1b[38;5;202m" }; // Dark orange
|
|
163
|
+
} else {
|
|
164
|
+
return .{ .value = ms, .unit = "ms", .color = "\x1b[31m" }; // Red
|
|
165
|
+
}
|
|
166
|
+
} else {
|
|
167
|
+
// Full red for >= 3ms
|
|
168
|
+
return .{ .value = ms, .unit = "ms", .color = "\x1b[31m" };
|
|
169
|
+
}
|
|
170
|
+
} else {
|
|
171
|
+
// Red for seconds
|
|
172
|
+
return .{ .value = @as(f64, @floatFromInt(ns)) / 1_000_000_000.0, .unit = "s", .color = "\x1b[31m" };
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
pub fn formatBytes(bytes: usize) struct { value: f64, unit: []const u8 } {
|
|
177
|
+
if (bytes < 1024) {
|
|
178
|
+
return .{ .value = @as(f64, @floatFromInt(bytes)), .unit = "B" };
|
|
179
|
+
} else if (bytes < 1024 * 1024) {
|
|
180
|
+
return .{ .value = @as(f64, @floatFromInt(bytes)) / 1024.0, .unit = "KiB" };
|
|
181
|
+
} else {
|
|
182
|
+
return .{ .value = @as(f64, @floatFromInt(bytes)) / (1024.0 * 1024.0), .unit = "MiB" };
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
pub fn printResults(writer: anytype, results: []const BenchResult) !void {
|
|
187
|
+
if (results.len == 0) return;
|
|
188
|
+
|
|
189
|
+
var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator);
|
|
190
|
+
defer arena.deinit();
|
|
191
|
+
const allocator = arena.allocator();
|
|
192
|
+
|
|
193
|
+
// Collect all unique memory stat names
|
|
194
|
+
var mem_stat_names: std.ArrayListUnmanaged([]const u8) = .{};
|
|
195
|
+
for (results) |result| {
|
|
196
|
+
if (result.mem_stats) |stats| {
|
|
197
|
+
for (stats) |stat| {
|
|
198
|
+
// Check if we already have this name
|
|
199
|
+
var found = false;
|
|
200
|
+
for (mem_stat_names.items) |existing_name| {
|
|
201
|
+
if (std.mem.eql(u8, existing_name, stat.name)) {
|
|
202
|
+
found = true;
|
|
203
|
+
break;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
if (!found) {
|
|
207
|
+
try mem_stat_names.append(allocator, stat.name);
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
// Calculate column widths
|
|
214
|
+
var max_name_len: usize = 20; // minimum
|
|
215
|
+
var min_col_width: usize = 3; // minimum for "Min"
|
|
216
|
+
var avg_col_width: usize = 3; // minimum for "Avg"
|
|
217
|
+
var max_col_width: usize = 3; // minimum for "Max"
|
|
218
|
+
|
|
219
|
+
// Create a map to store column widths for each memory stat
|
|
220
|
+
var mem_col_widths: std.ArrayListUnmanaged(usize) = .{};
|
|
221
|
+
for (mem_stat_names.items) |name| {
|
|
222
|
+
try mem_col_widths.append(allocator, name.len); // minimum is the name length
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
// First pass: calculate maximum widths
|
|
226
|
+
for (results) |result| {
|
|
227
|
+
if (result.name.len > max_name_len) {
|
|
228
|
+
max_name_len = result.name.len;
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
const min = formatDuration(result.min_ns);
|
|
232
|
+
const avg = formatDuration(result.avg_ns);
|
|
233
|
+
const max = formatDuration(result.max_ns);
|
|
234
|
+
|
|
235
|
+
var min_buf: [32]u8 = undefined;
|
|
236
|
+
const min_str = std.fmt.bufPrint(&min_buf, "{d:.2}{s}", .{ min.value, min.unit }) catch unreachable;
|
|
237
|
+
if (min_str.len > min_col_width) min_col_width = min_str.len;
|
|
238
|
+
|
|
239
|
+
var avg_buf: [32]u8 = undefined;
|
|
240
|
+
const avg_str = std.fmt.bufPrint(&avg_buf, "{d:.2}{s}", .{ avg.value, avg.unit }) catch unreachable;
|
|
241
|
+
if (avg_str.len > avg_col_width) avg_col_width = avg_str.len;
|
|
242
|
+
|
|
243
|
+
var max_buf: [32]u8 = undefined;
|
|
244
|
+
const max_str = std.fmt.bufPrint(&max_buf, "{d:.2}{s}", .{ max.value, max.unit }) catch unreachable;
|
|
245
|
+
if (max_str.len > max_col_width) max_col_width = max_str.len;
|
|
246
|
+
|
|
247
|
+
if (result.mem_stats) |stats| {
|
|
248
|
+
for (stats) |stat| {
|
|
249
|
+
const mem = formatBytes(stat.bytes);
|
|
250
|
+
var mem_buf: [32]u8 = undefined;
|
|
251
|
+
const mem_str = std.fmt.bufPrint(&mem_buf, "{d:.2} {s}", .{ mem.value, mem.unit }) catch unreachable;
|
|
252
|
+
|
|
253
|
+
// Find the index of this stat name
|
|
254
|
+
for (mem_stat_names.items, 0..) |name, i| {
|
|
255
|
+
if (std.mem.eql(u8, name, stat.name)) {
|
|
256
|
+
if (mem_str.len > mem_col_widths.items[i]) {
|
|
257
|
+
mem_col_widths.items[i] = mem_str.len;
|
|
258
|
+
}
|
|
259
|
+
break;
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
// Print header
|
|
267
|
+
var total_width = max_name_len + 3 + min_col_width + 3 + avg_col_width + 3 + max_col_width;
|
|
268
|
+
for (mem_col_widths.items) |width| {
|
|
269
|
+
total_width += 3 + width;
|
|
270
|
+
}
|
|
271
|
+
try writer.writeAll("\x1b[2m");
|
|
272
|
+
try writer.splatByteAll('-', total_width);
|
|
273
|
+
try writer.writeAll("\x1b[0m\n");
|
|
274
|
+
|
|
275
|
+
// Column headers
|
|
276
|
+
try writer.writeAll("\x1b[36m");
|
|
277
|
+
try writer.writeAll("Benchmark");
|
|
278
|
+
try writer.splatByteAll(' ', max_name_len - 9);
|
|
279
|
+
try writer.writeAll("\x1b[0m\x1b[2m | \x1b[0m");
|
|
280
|
+
|
|
281
|
+
try writer.writeAll("\x1b[36m");
|
|
282
|
+
try writer.writeAll("Min");
|
|
283
|
+
try writer.splatByteAll(' ', min_col_width - 3);
|
|
284
|
+
try writer.writeAll("\x1b[0m\x1b[2m | \x1b[0m");
|
|
285
|
+
|
|
286
|
+
try writer.writeAll("\x1b[36m");
|
|
287
|
+
try writer.writeAll("Avg");
|
|
288
|
+
try writer.splatByteAll(' ', avg_col_width - 3);
|
|
289
|
+
try writer.writeAll("\x1b[0m\x1b[2m | \x1b[0m");
|
|
290
|
+
|
|
291
|
+
try writer.writeAll("\x1b[36m");
|
|
292
|
+
try writer.writeAll("Max");
|
|
293
|
+
try writer.splatByteAll(' ', max_col_width - 3);
|
|
294
|
+
try writer.writeAll("\x1b[0m");
|
|
295
|
+
|
|
296
|
+
// Dynamic memory stat headers
|
|
297
|
+
for (mem_stat_names.items, 0..) |name, i| {
|
|
298
|
+
try writer.writeAll("\x1b[2m | \x1b[0m");
|
|
299
|
+
try writer.writeAll("\x1b[36m");
|
|
300
|
+
try writer.writeAll(name);
|
|
301
|
+
if (name.len < mem_col_widths.items[i]) {
|
|
302
|
+
try writer.splatByteAll(' ', mem_col_widths.items[i] - name.len);
|
|
303
|
+
}
|
|
304
|
+
try writer.writeAll("\x1b[0m");
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
try writer.writeByte('\n');
|
|
308
|
+
|
|
309
|
+
try writer.writeAll("\x1b[2m");
|
|
310
|
+
try writer.splatByteAll('-', total_width);
|
|
311
|
+
try writer.writeAll("\x1b[0m\n");
|
|
312
|
+
|
|
313
|
+
// Print each result
|
|
314
|
+
for (results, 0..) |result, row_idx| {
|
|
315
|
+
const min = formatDuration(result.min_ns);
|
|
316
|
+
const avg = formatDuration(result.avg_ns);
|
|
317
|
+
const max = formatDuration(result.max_ns);
|
|
318
|
+
|
|
319
|
+
// Format duration strings
|
|
320
|
+
var min_buf: [32]u8 = undefined;
|
|
321
|
+
const min_str = try std.fmt.bufPrint(&min_buf, "{d:.2}{s}", .{ min.value, min.unit });
|
|
322
|
+
|
|
323
|
+
var avg_buf: [32]u8 = undefined;
|
|
324
|
+
const avg_str = try std.fmt.bufPrint(&avg_buf, "{d:.2}{s}", .{ avg.value, avg.unit });
|
|
325
|
+
|
|
326
|
+
var max_buf: [32]u8 = undefined;
|
|
327
|
+
const max_str = try std.fmt.bufPrint(&max_buf, "{d:.2}{s}", .{ max.value, max.unit });
|
|
328
|
+
|
|
329
|
+
if (row_idx % 2 == 1) {
|
|
330
|
+
try writer.writeAll("\x1b[48;5;234m");
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
// Benchmark name
|
|
334
|
+
try writer.writeAll(result.name);
|
|
335
|
+
try writer.splatByteAll(' ', max_name_len - result.name.len);
|
|
336
|
+
try writer.writeAll("\x1b[2m | \x1b[0m");
|
|
337
|
+
if (row_idx % 2 == 1) {
|
|
338
|
+
try writer.writeAll("\x1b[48;5;234m");
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
// Min (right-aligned with color)
|
|
342
|
+
if (min_str.len < min_col_width) {
|
|
343
|
+
try writer.splatByteAll(' ', min_col_width - min_str.len);
|
|
344
|
+
}
|
|
345
|
+
try writer.writeAll(min.color);
|
|
346
|
+
try writer.writeAll(min_str);
|
|
347
|
+
try writer.writeAll("\x1b[0m");
|
|
348
|
+
try writer.writeAll("\x1b[2m | \x1b[0m");
|
|
349
|
+
if (row_idx % 2 == 1) {
|
|
350
|
+
try writer.writeAll("\x1b[48;5;234m");
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
// Avg (right-aligned with color)
|
|
354
|
+
if (avg_str.len < avg_col_width) {
|
|
355
|
+
try writer.splatByteAll(' ', avg_col_width - avg_str.len);
|
|
356
|
+
}
|
|
357
|
+
try writer.writeAll(avg.color);
|
|
358
|
+
try writer.writeAll(avg_str);
|
|
359
|
+
try writer.writeAll("\x1b[0m");
|
|
360
|
+
try writer.writeAll("\x1b[2m | \x1b[0m");
|
|
361
|
+
if (row_idx % 2 == 1) {
|
|
362
|
+
try writer.writeAll("\x1b[48;5;234m");
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
// Max (right-aligned with color)
|
|
366
|
+
if (max_str.len < max_col_width) {
|
|
367
|
+
try writer.splatByteAll(' ', max_col_width - max_str.len);
|
|
368
|
+
}
|
|
369
|
+
try writer.writeAll(max.color);
|
|
370
|
+
try writer.writeAll(max_str);
|
|
371
|
+
try writer.writeAll("\x1b[0m");
|
|
372
|
+
|
|
373
|
+
// Dynamic memory stats columns
|
|
374
|
+
for (mem_stat_names.items, 0..) |stat_name, i| {
|
|
375
|
+
try writer.writeAll("\x1b[2m | \x1b[0m");
|
|
376
|
+
if (row_idx % 2 == 1) {
|
|
377
|
+
try writer.writeAll("\x1b[48;5;234m");
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
// Look for this stat in the result's memory stats
|
|
381
|
+
var found_stat: ?usize = null;
|
|
382
|
+
if (result.mem_stats) |stats| {
|
|
383
|
+
for (stats) |stat| {
|
|
384
|
+
if (std.mem.eql(u8, stat.name, stat_name)) {
|
|
385
|
+
found_stat = stat.bytes;
|
|
386
|
+
break;
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
if (found_stat) |bytes| {
|
|
392
|
+
const mem = formatBytes(bytes);
|
|
393
|
+
var mem_buf: [32]u8 = undefined;
|
|
394
|
+
const mem_str = std.fmt.bufPrint(&mem_buf, "{d:.2} {s}", .{ mem.value, mem.unit }) catch unreachable;
|
|
395
|
+
|
|
396
|
+
// Right-aligned
|
|
397
|
+
if (mem_str.len < mem_col_widths.items[i]) {
|
|
398
|
+
try writer.splatByteAll(' ', mem_col_widths.items[i] - mem_str.len);
|
|
399
|
+
}
|
|
400
|
+
try writer.writeAll(mem_str);
|
|
401
|
+
} else {
|
|
402
|
+
// Empty column
|
|
403
|
+
try writer.splatByteAll(' ', mem_col_widths.items[i]);
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
if (row_idx % 2 == 1) {
|
|
408
|
+
try writer.writeAll("\x1b[0m");
|
|
409
|
+
}
|
|
410
|
+
try writer.writeByte('\n');
|
|
411
|
+
}
|
|
412
|
+
|
|
413
|
+
try writer.writeAll("\x1b[2m");
|
|
414
|
+
try writer.splatByteAll('-', total_width);
|
|
415
|
+
try writer.writeAll("\x1b[0m\n");
|
|
416
|
+
try writer.flush();
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
const BenchResultsJson = struct {
|
|
420
|
+
benchmark: []const u8,
|
|
421
|
+
results: []const BenchResult,
|
|
422
|
+
};
|
|
423
|
+
|
|
424
|
+
pub fn printResultsJson(writer: anytype, results: []const BenchResult, bench_name: []const u8) !void {
|
|
425
|
+
const output = BenchResultsJson{
|
|
426
|
+
.benchmark = bench_name,
|
|
427
|
+
.results = results,
|
|
428
|
+
};
|
|
429
|
+
try std.json.Stringify.value(output, .{ .emit_null_optional_fields = false }, writer);
|
|
430
|
+
try writer.writeByte('\n');
|
|
431
|
+
}
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
// Benchmark Runner CLI
|
|
2
|
+
//
|
|
3
|
+
// This is the main entry point for running performance benchmarks for opentui core components.
|
|
4
|
+
//
|
|
5
|
+
// Usage:
|
|
6
|
+
// zig build bench - Run all benchmarks
|
|
7
|
+
// zig build bench -- --help - Show help message with available options
|
|
8
|
+
//
|
|
9
|
+
// Options:
|
|
10
|
+
// --mem Show memory statistics after each benchmark
|
|
11
|
+
// --filter, -f NAME Run only benchmark categories matching NAME (case-insensitive substring)
|
|
12
|
+
// --bench, -b NAME Run only specific benchmarks matching NAME
|
|
13
|
+
// --json Output results in JSON format (machine-readable)
|
|
14
|
+
// --help, -h Display help message and list available benchmarks
|
|
15
|
+
//
|
|
16
|
+
// Examples:
|
|
17
|
+
// zig build bench -- --mem
|
|
18
|
+
// Run all benchmarks with memory statistics
|
|
19
|
+
//
|
|
20
|
+
// zig build bench -- --filter rope
|
|
21
|
+
// Run only benchmarks with "rope" in their name (Rope Data Structure, Rope Marker Tracking)
|
|
22
|
+
//
|
|
23
|
+
// zig build bench -- -f textbuffer --mem
|
|
24
|
+
// Run TextBuffer benchmarks with memory statistics
|
|
25
|
+
//
|
|
26
|
+
// zig build bench -- --filter "edit"
|
|
27
|
+
// Run EditBuffer Operations benchmarks
|
|
28
|
+
//
|
|
29
|
+
// zig build bench -- --bench "ASCII"
|
|
30
|
+
// Run only benchmarks with "ASCII" in their name
|
|
31
|
+
//
|
|
32
|
+
// zig build bench -- --json
|
|
33
|
+
// Output results in JSON format for CI integration
|
|
34
|
+
//
|
|
35
|
+
// Adding New Benchmarks:
|
|
36
|
+
// 1. Create a new file in bench/ directory (e.g., bench/my_bench.zig)
|
|
37
|
+
// 2. Export `pub const benchName = "My Benchmark";`
|
|
38
|
+
// 3. Export `pub fn run(allocator: std.mem.Allocator, show_mem: bool, bench_filter: ?[]const u8) ![]BenchResult`
|
|
39
|
+
// 4. Import the module at the top of this file
|
|
40
|
+
// 5. Add an entry to the `benchmarks` array in main() with your module
|
|
41
|
+
|
|
42
|
+
const std = @import("std");
|
|
43
|
+
const bench_utils = @import("bench-utils.zig");
|
|
44
|
+
const gp = @import("grapheme.zig");
|
|
45
|
+
|
|
46
|
+
// Import all benchmark modules
|
|
47
|
+
const text_buffer_view_bench = @import("bench/text-buffer-view_bench.zig");
|
|
48
|
+
const edit_buffer_bench = @import("bench/edit-buffer_bench.zig");
|
|
49
|
+
const rope_bench = @import("bench/rope_bench.zig");
|
|
50
|
+
const rope_markers_bench = @import("bench/rope-markers_bench.zig");
|
|
51
|
+
const text_buffer_coords_bench = @import("bench/text-buffer-coords_bench.zig");
|
|
52
|
+
const styled_text_bench = @import("bench/styled-text_bench.zig");
|
|
53
|
+
const buffer_draw_text_buffer_bench = @import("bench/buffer-draw-text-buffer_bench.zig");
|
|
54
|
+
const utf8_bench = @import("bench/utf8_bench.zig");
|
|
55
|
+
const text_chunk_graphemes_bench = @import("bench/text-chunk-graphemes_bench.zig");
|
|
56
|
+
|
|
57
|
+
const BenchModule = struct {
|
|
58
|
+
name: []const u8,
|
|
59
|
+
run: *const fn (std.mem.Allocator, bool, ?[]const u8) anyerror![]bench_utils.BenchResult,
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
fn matchesFilter(bench_name: []const u8, filter: ?[]const u8) bool {
|
|
63
|
+
if (filter == null) return true;
|
|
64
|
+
const filter_str = filter.?;
|
|
65
|
+
if (filter_str.len == 0) return true;
|
|
66
|
+
|
|
67
|
+
var i: usize = 0;
|
|
68
|
+
while (i + filter_str.len <= bench_name.len) : (i += 1) {
|
|
69
|
+
var matches = true;
|
|
70
|
+
for (filter_str, 0..) |filter_char, j| {
|
|
71
|
+
const bench_char = bench_name[i + j];
|
|
72
|
+
const filter_lower = if (filter_char >= 'A' and filter_char <= 'Z') filter_char + 32 else filter_char;
|
|
73
|
+
const bench_lower = if (bench_char >= 'A' and bench_char <= 'Z') bench_char + 32 else bench_char;
|
|
74
|
+
if (filter_lower != bench_lower) {
|
|
75
|
+
matches = false;
|
|
76
|
+
break;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
if (matches) return true;
|
|
80
|
+
}
|
|
81
|
+
return false;
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
pub fn main() !void {
|
|
85
|
+
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
|
86
|
+
defer _ = gpa.deinit();
|
|
87
|
+
const allocator = gpa.allocator();
|
|
88
|
+
|
|
89
|
+
// Initialize global pool and unicode data ONCE with base GPA allocator
|
|
90
|
+
// This ensures they persist across all benchmarks (even with arena allocators)
|
|
91
|
+
_ = gp.initGlobalPool(allocator);
|
|
92
|
+
defer gp.deinitGlobalPool();
|
|
93
|
+
|
|
94
|
+
const benchmarks = [_]BenchModule{
|
|
95
|
+
.{ .name = text_buffer_view_bench.benchName, .run = text_buffer_view_bench.run },
|
|
96
|
+
.{ .name = edit_buffer_bench.benchName, .run = edit_buffer_bench.run },
|
|
97
|
+
.{ .name = rope_bench.benchName, .run = rope_bench.run },
|
|
98
|
+
.{ .name = rope_markers_bench.benchName, .run = rope_markers_bench.run },
|
|
99
|
+
.{ .name = text_buffer_coords_bench.benchName, .run = text_buffer_coords_bench.run },
|
|
100
|
+
.{ .name = styled_text_bench.benchName, .run = styled_text_bench.run },
|
|
101
|
+
.{ .name = buffer_draw_text_buffer_bench.benchName, .run = buffer_draw_text_buffer_bench.run },
|
|
102
|
+
.{ .name = utf8_bench.benchName, .run = utf8_bench.run },
|
|
103
|
+
.{ .name = text_chunk_graphemes_bench.benchName, .run = text_chunk_graphemes_bench.run },
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
const args = try std.process.argsAlloc(allocator);
|
|
107
|
+
defer std.process.argsFree(allocator, args);
|
|
108
|
+
|
|
109
|
+
var show_mem = false;
|
|
110
|
+
var json_output = false;
|
|
111
|
+
var filter: ?[]const u8 = null;
|
|
112
|
+
var bench_filter: ?[]const u8 = null;
|
|
113
|
+
var i: usize = 1;
|
|
114
|
+
while (i < args.len) : (i += 1) {
|
|
115
|
+
const arg = args[i];
|
|
116
|
+
if (std.mem.eql(u8, arg, "--mem")) {
|
|
117
|
+
show_mem = true;
|
|
118
|
+
} else if (std.mem.eql(u8, arg, "--json")) {
|
|
119
|
+
json_output = true;
|
|
120
|
+
} else if (std.mem.eql(u8, arg, "--filter") or std.mem.eql(u8, arg, "-f")) {
|
|
121
|
+
if (i + 1 < args.len) {
|
|
122
|
+
i += 1;
|
|
123
|
+
filter = args[i];
|
|
124
|
+
}
|
|
125
|
+
} else if (std.mem.eql(u8, arg, "--bench") or std.mem.eql(u8, arg, "-b")) {
|
|
126
|
+
if (i + 1 < args.len) {
|
|
127
|
+
i += 1;
|
|
128
|
+
bench_filter = args[i];
|
|
129
|
+
}
|
|
130
|
+
} else if (std.mem.eql(u8, arg, "--help") or std.mem.eql(u8, arg, "-h")) {
|
|
131
|
+
var stdout_buffer: [4096]u8 = undefined;
|
|
132
|
+
var stdout_writer = std.fs.File.stdout().writer(&stdout_buffer);
|
|
133
|
+
const stdout = &stdout_writer.interface;
|
|
134
|
+
try stdout.print("Usage: bench [options]\n\n", .{});
|
|
135
|
+
try stdout.print("Options:\n", .{});
|
|
136
|
+
try stdout.print(" --mem Show memory statistics\n", .{});
|
|
137
|
+
try stdout.print(" --json Output in JSON format (machine-readable)\n", .{});
|
|
138
|
+
try stdout.print(" --filter, -f NAME Run only benchmark categories matching NAME\n", .{});
|
|
139
|
+
try stdout.print(" --bench, -b NAME Run only specific benchmarks matching NAME\n", .{});
|
|
140
|
+
try stdout.print(" --help, -h Show this help message\n\n", .{});
|
|
141
|
+
try stdout.print("Available benchmarks:\n", .{});
|
|
142
|
+
for (benchmarks) |bench| {
|
|
143
|
+
try stdout.print(" - {s}\n", .{bench.name});
|
|
144
|
+
}
|
|
145
|
+
try stdout.flush();
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
var stdout_buffer: [4096]u8 = undefined;
|
|
151
|
+
var stdout_writer = std.fs.File.stdout().writer(&stdout_buffer);
|
|
152
|
+
const stdout = &stdout_writer.interface;
|
|
153
|
+
|
|
154
|
+
if (!json_output and filter != null) {
|
|
155
|
+
try stdout.print("Filtering benchmarks by: \"{s}\"\n", .{filter.?});
|
|
156
|
+
}
|
|
157
|
+
if (!json_output and bench_filter != null) {
|
|
158
|
+
try stdout.print("Filtering individual benchmarks by: \"{s}\"\n", .{bench_filter.?});
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
var ran_any = false;
|
|
162
|
+
|
|
163
|
+
for (benchmarks) |bench| {
|
|
164
|
+
if (!matchesFilter(bench.name, filter)) continue;
|
|
165
|
+
|
|
166
|
+
// Use arena for results only - benchmark modules manage their own temp memory
|
|
167
|
+
var results_arena = std.heap.ArenaAllocator.init(allocator);
|
|
168
|
+
defer results_arena.deinit();
|
|
169
|
+
|
|
170
|
+
const start_time = std.time.nanoTimestamp();
|
|
171
|
+
const results = try bench.run(results_arena.allocator(), show_mem, bench_filter);
|
|
172
|
+
const end_time = std.time.nanoTimestamp();
|
|
173
|
+
const elapsed_ns = end_time - start_time;
|
|
174
|
+
|
|
175
|
+
if (results.len == 0) continue;
|
|
176
|
+
|
|
177
|
+
if (!json_output) {
|
|
178
|
+
try stdout.print("\n=== {s} Benchmarks ===\n\n", .{bench.name});
|
|
179
|
+
try stdout.flush();
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
if (json_output) {
|
|
183
|
+
try bench_utils.printResultsJson(stdout, results, bench.name);
|
|
184
|
+
} else {
|
|
185
|
+
try bench_utils.printResults(stdout, results);
|
|
186
|
+
const elapsed_ms = @as(f64, @floatFromInt(elapsed_ns)) / 1_000_000.0;
|
|
187
|
+
try stdout.print("\n Overall time: {d:.2}ms\n", .{elapsed_ms});
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
ran_any = true;
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
if (!ran_any) {
|
|
194
|
+
if (!json_output) {
|
|
195
|
+
if (filter != null and bench_filter != null) {
|
|
196
|
+
try stdout.print(
|
|
197
|
+
"\nNo benchmarks matched filters: category=\"{s}\", bench=\"{s}\"\n",
|
|
198
|
+
.{ filter.?, bench_filter.? },
|
|
199
|
+
);
|
|
200
|
+
} else if (bench_filter != null) {
|
|
201
|
+
try stdout.print("\nNo benchmarks matched bench filter: \"{s}\"\n", .{bench_filter.?});
|
|
202
|
+
} else if (filter != null) {
|
|
203
|
+
try stdout.print("\nNo benchmarks matched filter: \"{s}\"\n", .{filter.?});
|
|
204
|
+
} else {
|
|
205
|
+
try stdout.print("\nNo benchmarks ran.\n", .{});
|
|
206
|
+
}
|
|
207
|
+
try stdout.print("Use --help to see available benchmarks.\n", .{});
|
|
208
|
+
}
|
|
209
|
+
try stdout.flush();
|
|
210
|
+
return;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
if (!json_output) {
|
|
214
|
+
try stdout.print("\n✓ Benchmarks complete\n", .{});
|
|
215
|
+
}
|
|
216
|
+
try stdout.flush();
|
|
217
|
+
}
|