diff --git a/.github/workflows/gha-ci.yml b/.github/workflows/gha-ci.yml index 59c986b5ab..7f42a33295 100644 --- a/.github/workflows/gha-ci.yml +++ b/.github/workflows/gha-ci.yml @@ -3,13 +3,15 @@ name: GitHubActions CI tests on: push: branches: [master, dev, ci] - pull_request: + # pull_request: + workflow_dispatch: jobs: build-nix: strategy: matrix: - os: [ubuntu-20.04, macos-10.15] + #os: [ubuntu-20.04, macos-11] + os: [ubuntu-20.04] build_suite: [autoconf, cmake] with_coveralls: [no] # Run one build with gcov and coveralls @@ -22,7 +24,7 @@ jobs: steps: - name: checkout code - uses: actions/checkout@v2 + uses: actions/checkout@v3 - name: install deps run: | @@ -120,7 +122,7 @@ jobs: - name: upload artifact if: ${{ startsWith( matrix.os, 'macos-' ) && matrix.build_suite == 'autoconf' }} - uses: actions/upload-artifact@v2 + uses: actions/upload-artifact@v3 with: name: aegisub-macos-GHA path: aegisub-macos-GHA.dmg @@ -129,35 +131,49 @@ jobs: runs-on: windows-latest steps: - name: checkout code - uses: actions/checkout@v2 + uses: actions/checkout@v3 - name: cache vcpkg packages - uses: pat-s/always-upload-cache@v2 + uses: pat-s/always-upload-cache@v3 with: path: C:\Users\runneradmin\AppData\Local\vcpkg\archives key: vcpkg-archives-${{ github.run_id }} restore-keys: vcpkg-archives- + - name: Run vcpkg + uses: lukka/run-vcpkg@v7 + with: + setupOnly: true + doNotCache: true + vcpkgDirectory: '${{ github.workspace }}/vcpkg' + vcpkgGitCommitId: 96ec7fb25da25e0463446e552d59715a47c95e73 # boost 1.79.0, Apr 21 2022. + - name: install deps run: | Invoke-WebRequest -Uri https://github.com/FFMS/ffms2/releases/download/2.23/ffms2-2.23.1-msvc.7z -OutFile vendor/ffms2-2.23.1-msvc.7z 7z.exe x vendor/ffms2-2.23.1-msvc.7z -ovendor/ffms2 - Set-Content -Value "" -Path "C:\vcpkg\vcpkg.disable-metrics" -Force + # msys mirrors have deleted the version of libtool used by vcpkg, so we have to update this file. + # see https://github.com/microsoft/vcpkg/commit/fd766eba2b4cf59c7123d46189be373e2cee959d + (Get-Content "${{github.workspace}}\vcpkg\scripts\cmake\vcpkg_acquire_msys.cmake") -replace 'libtool-2.4.6-9-x86_64.pkg.tar.xz', 'libtool-2.4.7-3-x86_64.pkg.tar.zst' -replace 'b309799e5a9d248ef66eaf11a0bd21bf4e8b9bd5c677c627ec83fa760ce9f0b54ddf1b62cbb436e641fbbde71e3b61cb71ff541d866f8ca7717a3a0dbeb00ebf', 'a202ddaefa93d8a4b15431dc514e3a6200c47275c5a0027c09cc32b28bc079b1b9a93d5ef65adafdc9aba5f76a42f3303b1492106ddf72e67f1801ebfe6d02cc' | Set-Content "${{github.workspace}}\vcpkg\scripts\cmake\vcpkg_acquire_msys.cmake" + # zlib: update URL. + (Get-Content "${{github.workspace}}\vcpkg\ports\zlib\portfile.cmake") -replace 'https://www.zlib.net', 'https://www.zlib.net/fossils/' | Set-Content "${{github.workspace}}\vcpkg\ports\zlib\portfile.cmake" + + Set-Content -Value "" -Path "${{ github.workspace }}\vcpkg\vcpkg.disable-metrics" -Force # Fix vcpkg wxwidgets fail when building without debug - (Get-Content "C:\vcpkg\ports\wxwidgets\vcpkg-cmake-wrapper.cmake") -replace ' _find_package', ' #_find_package' | Set-Content "C:\vcpkg\ports\wxwidgets\vcpkg-cmake-wrapper.cmake" - vcpkg.exe install --triplet x64-windows-release avisynthplus boost-asio boost-container boost-crc boost-filesystem boost-flyweight boost-gil boost-interprocess 'boost-locale[icu]' 'boost-regex[icu]' boost-scope-exit boost-thread fftw3 hunspell icu libass libiconv uchardet wxwidgets xaudio2redist gtest + (Get-Content "${{github.workspace}}\vcpkg\ports\wxwidgets\vcpkg-cmake-wrapper.cmake") -replace ' _find_package', ' #_find_package' | Set-Content "${{github.workspace}}\vcpkg\ports\wxwidgets\vcpkg-cmake-wrapper.cmake" + & "${{github.workspace}}\vcpkg\vcpkg.exe" install --triplet x64-windows-release avisynthplus boost-asio boost-container boost-crc boost-filesystem boost-flyweight boost-gil boost-interprocess 'boost-locale[icu]' 'boost-regex[icu]' boost-scope-exit boost-thread fftw3 hunspell icu libass libiconv uchardet wxwidgets xaudio2redist gtest - name: configure run: | bash.exe build/version.sh . New-Item -Path build-dir -ItemType directory Set-Location build-dir - cmake.exe '-DCMAKE_TOOLCHAIN_FILE=C:/vcpkg/scripts/buildsystems/vcpkg.cmake' '-DVCPKG_TARGET_TRIPLET=x64-windows-release' '-DFFMS2_INCLUDE_DIRS=../vendor/ffms2/ffms2-2.23.1-msvc/include' '-DFFMS2_LIBRARIES=../vendor/ffms2/ffms2-2.23.1-msvc/x64/ffms2.lib' '-DXAUDIO2_REDIST=ON' '-DCMAKE_CXX_FLAGS=/DWIN32 /D_WINDOWS /GR /EHsc /DUNICODE /D_UNICODE /MP' '-DCMAKE_C_FLAGS=/DWIN32 /D_WINDOWS /DUNICODE /D_UNICODE /MP' '-DWITH_TEST=ON' .. + cmake.exe '-DCMAKE_TOOLCHAIN_FILE=${{ github.workspace }}/vcpkg/scripts/buildsystems/vcpkg.cmake' '-DVCPKG_TARGET_TRIPLET=x64-windows-release' '-DFFMS2_INCLUDE_DIRS=../vendor/ffms2/ffms2-2.23.1-msvc/include' '-DFFMS2_LIBRARIES=../vendor/ffms2/ffms2-2.23.1-msvc/x64/ffms2.lib' '-DXAUDIO2_REDIST=ON' '-DCMAKE_CXX_FLAGS=/DWIN32 /D_WINDOWS /GR /EHsc /DUNICODE /D_UNICODE /MP /DwxDEBUG_LEVEL=0' '-DCMAKE_C_FLAGS=/DWIN32 /D_WINDOWS /DUNICODE /D_UNICODE /MP /DwxDEBUG_LEVEL=0' '-DWITH_TEST=ON' .. - name: Add msbuild to PATH - uses: microsoft/setup-msbuild@v1.0.2 + uses: microsoft/setup-msbuild@v1.1 - name: build run: | @@ -169,24 +185,47 @@ jobs: Set-Location build-dir MSBuild.exe test.vcxproj -p:Configuration=Release + - name: upload build artifact + if: always() + uses: actions/upload-artifact@v3 + with: + name: aegisub-win64-builddir + path: build-dir/ + - name: package run: | - Invoke-WebRequest -Uri https://ftp.wangqr.tk/aegisub/aegisub-win64-cmake-20210713-r9212.zip -OutFile vendor/aegisub9212.zip + Invoke-WebRequest -Uri https://github.com/AmusementClub/Aegisub/releases/download/artifacts/aegisub-win64-cmake-20210713-r9212.zip -OutFile vendor/aegisub9212.zip 7z.exe x vendor/aegisub9212.zip -ovendor/aegisub9212 + Invoke-WebRequest -Uri https://github.com/AmusementClub/ffms2/releases/download/nightly-20230218/release-ffms2-x86_64-nightly-20230218-ge4270cc.zip -OutFile vendor/ffms2.zip + + 7z.exe x vendor/ffms2.zip -ovendor + + Invoke-WebRequest -Uri https://github.com/AmusementClub/assrender/releases/download/0.38.0/assrender_v0380_0.17.0.zip -OutFile vendor/assrender.zip + Invoke-WebRequest -Uri https://github.com/HomeOfVapourSynthEvolution/xy-VSFilter/releases/download/r3/xy-VSFilter-r3.7z -OutFile vendor/xy-vsf.7z + Invoke-WebRequest -Uri https://github.com/Masaiki/VSFilterMod/releases/download/r5.2.7/VSFilterMod_bin.zip -OutFile vendor/vsfmod.zip + + 7z.exe x vendor/assrender.zip -ovendor + 7z.exe x vendor/xy-vsf.7z -ovendor + 7z.exe x vendor/vsfmod.zip -ovendor + Move-Item -Path build-dir\Release\Aegisub.exe -Destination build-dir\Aegisub.exe Remove-Item -Path build-dir\Release\*.exp,build-dir\Release\*.lib,build-dir\Release\*.exe Move-Item -Path build-dir\Aegisub.exe -Destination build-dir\Release\Aegisub.exe New-Item -Path build-dir\Release\automation -ItemType directory Copy-Item -Path automation\autoload,automation\demos,automation\include -Destination build-dir\Release\automation -Recurse - Copy-Item -Path C:\vcpkg\installed\x64-windows-release\bin\AviSynth.dll -Destination build-dir\Release - Copy-Item -Path vendor\ffms2\ffms2-2.23.1-msvc\x64\ffms2.dll -Destination build-dir\Release - Copy-Item -Path vendor\aegisub9212\csri,vendor\aegisub9212\dictionaries,vendor\aegisub9212\licenses,vendor\aegisub9212\locale,vendor\aegisub9212\AssDraw3.exe,vendor\aegisub9212\AssDraw3.chm,vendor\aegisub9212\DirectShowSource.dll -Destination build-dir\Release -Recurse + New-Item -Path build-dir\Release\csri -ItemType directory + + Copy-Item -Path "${{ github.workspace }}\vcpkg\installed\x64-windows-release\bin\AviSynth.dll" -Destination build-dir\Release + Copy-Item -Path vendor\ffms2.dll -Destination build-dir\Release + Copy-Item -Path vendor\x64\*.dll -Destination build-dir\Release\csri + Copy-Item -Path vendor\plugins64\xy-VSFilter.dll -Destination build-dir\Release\csri + Copy-Item -Path vendor\aegisub9212\dictionaries,vendor\aegisub9212\licenses,vendor\aegisub9212\locale,vendor\aegisub9212\AssDraw3.exe,vendor\aegisub9212\AssDraw3.chm,vendor\aegisub9212\DirectShowSource.dll -Destination build-dir\Release -Recurse - name: upload artifact - uses: actions/upload-artifact@v2 + uses: actions/upload-artifact@v3 with: name: aegisub-win64-cmake-GHA path: build-dir/Release diff --git a/automation/autoload/cleantags-autoload.lua b/automation/autoload/cleantags-autoload.lua index 3d739dd55d..285cecdc72 100644 --- a/automation/autoload/cleantags-autoload.lua +++ b/automation/autoload/cleantags-autoload.lua @@ -1,4 +1,4 @@ ---[[ +--[[ "Clean Tags" -- An Auto4 LUA script for cleaning up ASS subtitle lines of badly-formed override blocks and redundant/duplicate tags * Designed to work for Aegisub 2.0 and above (only pre-release version was available at the time of diff --git a/automation/autoload/kara-templater.lua b/automation/autoload/kara-templater.lua index 8b6be10bb1..d1ea079727 100644 --- a/automation/autoload/kara-templater.lua +++ b/automation/autoload/kara-templater.lua @@ -115,7 +115,7 @@ function parse_code(meta, styles, line, templates, mods) aegisub.debug.out(3, "Unknown modifier in code template: %s\nIn template code line: %s\nEffect field: %s\n\n", m, line.text, line.effect) end end - + if not inserted then aegisub.debug.out(5, "Found implicit run-once code line: %s\n", line.text) table.insert(templates.once, template) @@ -145,7 +145,7 @@ function parse_template(meta, styles, line, templates, mods) noblank = false } local inserted = false - + local rest = mods while rest ~= "" do local m, t = string.headtail(rest) @@ -239,7 +239,7 @@ function parse_template(meta, styles, line, templates, mods) aegisub.debug.out(3, "Unknown modifier in template: %s\nIn template line: %s\nEffect field: %s\n\n", m, line.text, line.effect) end end - + if not inserted then table.insert(templates.syl, template) end @@ -299,9 +299,9 @@ function apply_templates(meta, styles, subs, templates) _G = _G } tenv.tenv = tenv - + -- Define helper functions in tenv - + tenv.retime = function(mode, addstart, addend) local line, syl = tenv.line, tenv.syl local newstart, newend = line.start_time, line.end_time @@ -346,20 +346,20 @@ function apply_templates(meta, styles, subs, templates) line.duration = newend - newstart return "" end - + tenv.fxgroup = {} - + tenv.relayer = function(layer) tenv.line.layer = layer return "" end - + tenv.restyle = function(style) tenv.line.style = style tenv.line.styleref = styles[style] return "" end - + tenv.maxloop = function(newmaxj) tenv.maxj = newmaxj return "" @@ -370,7 +370,7 @@ function apply_templates(meta, styles, subs, templates) tenv.maxj = newmaxj return "" end - + tenv.recall = {} setmetatable(tenv.recall, { decorators = {}, @@ -416,13 +416,13 @@ function apply_templates(meta, styles, subs, templates) end return value end - + -- run all run-once code snippets for k, t in pairs(templates.once) do assert(t.code, "WTF, a 'once' template without code?") run_code_template(t, tenv) end - + -- start processing lines local i, n = 0, #subs while i < n do @@ -438,7 +438,7 @@ function apply_templates(meta, styles, subs, templates) l.comment = true l.effect = "karaoke" subs[i] = l - end + end end end end @@ -499,7 +499,7 @@ end function apply_line(meta, styles, subs, line, templates, tenv) -- Tell whether any templates were applied to this line, needed to know whether the original line should be removed from input local applied_templates = false - + -- General variable replacement context local varctx = { layer = line.layer, @@ -527,7 +527,7 @@ function apply_line(meta, styles, subs, line, templates, tenv) lx = math.floor(line.x+0.5), ly = math.floor(line.y+0.5) } - + tenv.orgline = line tenv.line = nil tenv.syl = nil @@ -537,7 +537,7 @@ function apply_line(meta, styles, subs, line, templates, tenv) aegisub.debug.out(5, "Running line templates\n") for t in matching_templates(templates.line, line, tenv) do if aegisub.progress.is_cancelled() then break end - + -- Set varctx for per-line variables varctx["start"] = varctx.lstart varctx["end"] = varctx.lend @@ -555,7 +555,7 @@ function apply_line(meta, styles, subs, line, templates, tenv) varctx.height = varctx.lheight varctx.x = varctx.lx varctx.y = varctx.ly - + for j, maxj in template_loop(tenv, t.loops) do if t.code then aegisub.debug.out(5, "Code template, %s\n", t.code) @@ -605,29 +605,29 @@ function apply_line(meta, styles, subs, line, templates, tenv) end end aegisub.debug.out(5, "Done running line templates\n\n") - + -- Loop over syllables for i = 0, line.kara.n do if aegisub.progress.is_cancelled() then break end local syl = line.kara[i] - + aegisub.debug.out(5, "Applying templates to syllable: %s\n", syl.text) if apply_syllable_templates(syl, line, templates.syl, tenv, varctx, subs) then applied_templates = true end end - + -- Loop over furigana for i = 1, line.furi.n do if aegisub.progress.is_cancelled() then break end local furi = line.furi[i] - + aegisub.debug.out(5, "Applying templates to furigana: %s\n", furi.text) if apply_syllable_templates(furi, line, templates.furi, tenv, varctx, subs) then applied_templates = true end end - + return applied_templates end @@ -635,7 +635,6 @@ function run_code_template(template, tenv) local f, err = loadstring(template.code, "template code") if not f then aegisub.debug.out(2, "Failed to parse Lua code: %s\nCode that failed to parse: %s\n\n", err, template.code) - aegisub.cancel() else local pcall = pcall setfenv(f, tenv) @@ -643,7 +642,6 @@ function run_code_template(template, tenv) local res, err = pcall(f) if not res then aegisub.debug.out(2, "Runtime error in template code: %s\nCode producing error: %s\n\n", err, template.code) - aegisub.cancel() end end end @@ -652,7 +650,7 @@ end function run_text_template(template, tenv, varctx) local res = template aegisub.debug.out(5, "Running text template '%s'\n", res) - + -- Replace the variables in the string (this is probably faster than using a custom function, but doesn't provide error reporting) if varctx then aegisub.debug.out(5, "Has varctx, replacing variables\n") @@ -671,13 +669,13 @@ function run_text_template(template, tenv, varctx) res = string.gsub(res, "$([%a_]+)", var_replacer) aegisub.debug.out(5, "Done replacing variables, new template string is '%s'\n", res) end - + -- Function for evaluating expressions local function expression_evaluator(expression) f, err = loadstring(string.format("return (%s)", expression)) if (err) ~= nil then aegisub.debug.out(2, "Error parsing expression: %s\nExpression producing error: %s\nTemplate with expression: %s\n\n", err, expression, template) - aegisub.cancel() + return "!" .. expression .. "!" else setfenv(f, tenv) local res, val = pcall(f) @@ -685,7 +683,7 @@ function run_text_template(template, tenv, varctx) return val else aegisub.debug.out(2, "Runtime error in template expression: %s\nExpression producing error: %s\nTemplate with expression: %s\n\n", val, expression, template) - aegisub.cancel() + return "!" .. expression .. "!" end end end @@ -693,24 +691,24 @@ function run_text_template(template, tenv, varctx) aegisub.debug.out(5, "Now evaluating expressions\n") res = string.gsub(res , "!(.-)!", expression_evaluator) aegisub.debug.out(5, "After evaluation: %s\nDone handling template\n\n", res) - + return res end function apply_syllable_templates(syl, line, templates, tenv, varctx, subs) local applied = 0 - + -- Loop over all templates matching the line style for t in matching_templates(templates, line, tenv) do if aegisub.progress.is_cancelled() then break end - + tenv.syl = syl tenv.basesyl = syl set_ctx_syl(varctx, line, syl) - + applied = applied + apply_one_syllable_template(syl, line, t, tenv, varctx, subs, false, false) end - + return applied > 0 end @@ -718,7 +716,7 @@ function is_syl_blank(syl) if syl.duration <= 0 then return true end - + -- try to remove common spacing characters local t = syl.text_stripped if t:len() <= 0 then return true end @@ -731,26 +729,26 @@ function apply_one_syllable_template(syl, line, template, tenv, varctx, subs, sk if aegisub.progress.is_cancelled() then return 0 end local t = template local applied = 0 - + aegisub.debug.out(5, "Applying template to one syllable with text: %s\n", syl.text) - + -- Check for right inline_fx if t.fx and t.fx ~= syl.inline_fx then aegisub.debug.out(5, "Syllable has wrong inline-fx (wanted '%s', got '%s'), skipping.\n", t.fx, syl.inline_fx) return 0 end - + if t.noblank and is_syl_blank(syl) then aegisub.debug.out(5, "Syllable is blank, skipping.\n") return 0 end - + -- Recurse to per-char if required if not skip_perchar and t.perchar then aegisub.debug.out(5, "Doing per-character effects...\n") local charsyl = table.copy(syl) tenv.syl = charsyl - + local left, width = syl.left, 0 for c in unicode.chars(syl.text_stripped) do charsyl.text = c @@ -764,29 +762,29 @@ function apply_one_syllable_template(syl, line, template, tenv, varctx, subs, sk charsyl.prespacewidth, charsyl.postspacewidth = 0, 0 -- whatever... left = left + width set_ctx_syl(varctx, line, charsyl) - + applied = applied + apply_one_syllable_template(charsyl, line, t, tenv, varctx, subs, true, false) end - + return applied end - + -- Recurse to multi-hl if required if not skip_multi and t.multi then aegisub.debug.out(5, "Doing multi-highlight effects...\n") local hlsyl = table.copy(syl) tenv.syl = hlsyl - + for hl = 1, syl.highlights.n do local hldata = syl.highlights[hl] hlsyl.start_time = hldata.start_time hlsyl.end_time = hldata.end_time hlsyl.duration = hldata.duration set_ctx_syl(varctx, line, hlsyl) - + applied = applied + apply_one_syllable_template(hlsyl, line, t, tenv, varctx, subs, true, true) end - + return applied end @@ -815,7 +813,7 @@ function apply_one_syllable_template(syl, line, template, tenv, varctx, subs, sk applied = applied + 1 end end - + return applied end @@ -824,10 +822,10 @@ end function filter_apply_templates(subs, config) aegisub.progress.task("Collecting header data...") local meta, styles = karaskel.collect_head(subs, true) - + aegisub.progress.task("Parsing templates...") local templates = parse_templates(meta, styles, subs) - + aegisub.progress.task("Applying templates...") apply_templates(meta, styles, subs, templates) end diff --git a/automation/autoload/karaoke-auto-leadin.lua b/automation/autoload/karaoke-auto-leadin.lua index 7544f45a75..c3dfb09613 100644 --- a/automation/autoload/karaoke-auto-leadin.lua +++ b/automation/autoload/karaoke-auto-leadin.lua @@ -1,4 +1,4 @@ ---[[ +--[[ Copyright (c) 2007, Niels Martin Hansen All rights reserved. diff --git a/automation/autoload/macro-1-edgeblur.lua b/automation/autoload/macro-1-edgeblur.lua index 9c316c6955..f49321cd1f 100644 --- a/automation/autoload/macro-1-edgeblur.lua +++ b/automation/autoload/macro-1-edgeblur.lua @@ -1,4 +1,4 @@ --- Automation 4 demo script +-- Automation 4 demo script -- Macro that adds \be1 tags in front of every selected line local tr = aegisub.gettext diff --git a/automation/autoload/strip-tags.lua b/automation/autoload/strip-tags.lua index 62601e662e..ffe1b90d18 100644 --- a/automation/autoload/strip-tags.lua +++ b/automation/autoload/strip-tags.lua @@ -1,4 +1,4 @@ --- Copyright (c) 2010, Thomas Goyne +-- Copyright (c) 2010, Thomas Goyne -- -- Permission to use, copy, modify, and distribute this software for any -- purpose with or without fee is hereby granted, provided that the above diff --git a/automation/demos/future-windy-blur.lua b/automation/demos/future-windy-blur.lua index f67206ed94..00711a650a 100644 --- a/automation/demos/future-windy-blur.lua +++ b/automation/demos/future-windy-blur.lua @@ -1,4 +1,4 @@ --- This script is given to the public domain. +-- This script is given to the public domain. -- You can use and modify the effect freely, also without credit, although I would appreciate some. include("karaskel.lua") diff --git a/automation/demos/raytracer-test1.ass b/automation/demos/raytracer-test1.ass index 3cfc731c06..cc345220bf 100644 --- a/automation/demos/raytracer-test1.ass +++ b/automation/demos/raytracer-test1.ass @@ -1,25 +1,25 @@ -[Script Info] -; Script generated by Aegisub v2.00 PRE-RELEASE (SVN r965, jfs) -; http://www.aegisub.net -Title: Default Aegisub file -ScriptType: v4.00+ -WrapStyle: 0 -PlayResX: 640 -PlayResY: 480 -Video Aspect Ratio: 0 -Video Zoom: 6 -Video Position: 0 -Automation Scripts: ~raytracer.lua - -[V4+ Styles] -Format: Name, Fontname, Fontsize, PrimaryColour, SecondaryColour, OutlineColour, BackColour, Bold, Italic, Underline, StrikeOut, ScaleX, ScaleY, Spacing, Angle, BorderStyle, Outline, Shadow, Alignment, MarginL, MarginR, MarginV, Encoding -Style: Default,Arial,20,&H00FFFFFF,&H0000FFFF,&H00000000,&H00000000,0,0,0,0,100,100,0,0,1,2,2,2,10,10,10,0 - -[Events] -Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text -Dialogue: 0,0:00:00.00,0:00:05.00,Default,,0000,0000,0000,,light 2 2 2 100 0 0 -Dialogue: 0,0:00:05.00,0:00:10.00,Default,,0000,0000,0000,,light -2 2 2 0 200 0 -Dialogue: 0,0:00:10.00,0:00:15.00,Default,,0000,0000,0000,,light 2 -2 2 0 0 100 -Dialogue: 0,0:00:15.00,0:00:20.00,Default,,0000,0000,0000,,tri -10 -3 10 10 -3 10 -10 -3 -10 1 1 1 -Dialogue: 0,0:00:20.00,0:00:25.00,Default,,0000,0000,0000,,tri 0 0 1 1 0 1 0 1 1 1 1 1 -Dialogue: 0,0:00:25.00,0:00:30.00,Default,,0000,0000,0000,, +[Script Info] +; Script generated by Aegisub v2.00 PRE-RELEASE (SVN r965, jfs) +; http://www.aegisub.net +Title: Default Aegisub file +ScriptType: v4.00+ +WrapStyle: 0 +PlayResX: 640 +PlayResY: 480 +Video Aspect Ratio: 0 +Video Zoom: 6 +Video Position: 0 +Automation Scripts: ~raytracer.lua + +[V4+ Styles] +Format: Name, Fontname, Fontsize, PrimaryColour, SecondaryColour, OutlineColour, BackColour, Bold, Italic, Underline, StrikeOut, ScaleX, ScaleY, Spacing, Angle, BorderStyle, Outline, Shadow, Alignment, MarginL, MarginR, MarginV, Encoding +Style: Default,Arial,20,&H00FFFFFF,&H0000FFFF,&H00000000,&H00000000,0,0,0,0,100,100,0,0,1,2,2,2,10,10,10,0 + +[Events] +Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text +Dialogue: 0,0:00:00.00,0:00:05.00,Default,,0000,0000,0000,,light 2 2 2 100 0 0 +Dialogue: 0,0:00:05.00,0:00:10.00,Default,,0000,0000,0000,,light -2 2 2 0 200 0 +Dialogue: 0,0:00:10.00,0:00:15.00,Default,,0000,0000,0000,,light 2 -2 2 0 0 100 +Dialogue: 0,0:00:15.00,0:00:20.00,Default,,0000,0000,0000,,tri -10 -3 10 10 -3 10 -10 -3 -10 1 1 1 +Dialogue: 0,0:00:20.00,0:00:25.00,Default,,0000,0000,0000,,tri 0 0 1 1 0 1 0 1 1 1 1 1 +Dialogue: 0,0:00:25.00,0:00:30.00,Default,,0000,0000,0000,, diff --git a/automation/demos/raytracer.lua b/automation/demos/raytracer.lua index 16f1539198..ab684b149c 100644 --- a/automation/demos/raytracer.lua +++ b/automation/demos/raytracer.lua @@ -1,4 +1,4 @@ --- This script is given to the public domain +-- This script is given to the public domain -- It was originally intended as an april fool's joke in 2007, though it never really caught on. -- You're encouraged to experiment with this if you have plenty of time. -- Even if it's made as a joke, it does show that "anything is possible with ASS, although some things are insane to try." diff --git a/automation/include/aegisub/argcheck.moon b/automation/include/aegisub/argcheck.moon index b7492cdf76..417bb4fe16 100644 --- a/automation/include/aegisub/argcheck.moon +++ b/automation/include/aegisub/argcheck.moon @@ -1,78 +1,78 @@ --- Copyright (c) 2014, Thomas Goyne --- --- Permission to use, copy, modify, and distribute this software for any --- purpose with or without fee is hereby granted, provided that the above --- copyright notice and this permission notice appear in all copies. --- --- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES --- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF --- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR --- ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES --- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN --- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF --- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. --- --- Aegisub Project http://www.aegisub.org/ - -assert = assert -error = error -select = select -tostring = tostring -type = type - -is_type = (v, ty, expected) -> - ty == expected or (ty == 'table' and v.__class and v.__class.__name == expected) - -(argfmt) -> - assert type(argfmt) == 'string' - min_args = 0 - max_args = 0 - checks = {} - for arg in argfmt\gmatch '[^ ]+' - if arg == '...' - max_args = nil - break - - max_args += 1 - - optional = arg\sub(1, 1) == '?' - if optional - arg = arg\sub 2 - else - min_args += 1 - - if arg\find '|' - types = [ty for ty in arg\gmatch '[^|]+'] - checks[max_args] = (i, v) -> - if v == nil - return if optional - error "Argument ##{i} should be a #{arg}, is nil", 4 - ty = type v - for argtype in *types - return if is_type v, ty, argtype - error "Argument ##{i} should be a #{arg}, is #{ty} (#{v})", 3 - else - checks[max_args] = (i, v) -> - if v == nil - return if optional - error "Argument ##{i} should be a #{arg}, is nil", 4 - ty = type v - return if is_type v, ty, arg - error "Argument ##{i} should be a #{arg}, is #{ty} (#{v})", 3 - - (fn) -> (...) -> - arg_count = select '#', ... - if arg_count < min_args or (max_args and arg_count > max_args) - if min_args == max_args - error "Expected #{min_args} arguments, got #{arg_count}", 3 - else if max_args - error "Expected #{min_args}-#{max_args} arguments, got #{arg_count}", 3 - else - error "Expected at least #{min_args} arguments, got #{arg_count}", 3 - - for i=1,arg_count - if not checks[i] then break - checks[i] i, select i, ... - - fn ... - +-- Copyright (c) 2014, Thomas Goyne +-- +-- Permission to use, copy, modify, and distribute this software for any +-- purpose with or without fee is hereby granted, provided that the above +-- copyright notice and this permission notice appear in all copies. +-- +-- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +-- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +-- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +-- ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +-- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +-- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +-- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. +-- +-- Aegisub Project http://www.aegisub.org/ + +assert = assert +error = error +select = select +tostring = tostring +type = type + +is_type = (v, ty, expected) -> + ty == expected or (ty == 'table' and v.__class and v.__class.__name == expected) + +(argfmt) -> + assert type(argfmt) == 'string' + min_args = 0 + max_args = 0 + checks = {} + for arg in argfmt\gmatch '[^ ]+' + if arg == '...' + max_args = nil + break + + max_args += 1 + + optional = arg\sub(1, 1) == '?' + if optional + arg = arg\sub 2 + else + min_args += 1 + + if arg\find '|' + types = [ty for ty in arg\gmatch '[^|]+'] + checks[max_args] = (i, v) -> + if v == nil + return if optional + error "Argument ##{i} should be a #{arg}, is nil", 4 + ty = type v + for argtype in *types + return if is_type v, ty, argtype + error "Argument ##{i} should be a #{arg}, is #{ty} (#{v})", 3 + else + checks[max_args] = (i, v) -> + if v == nil + return if optional + error "Argument ##{i} should be a #{arg}, is nil", 4 + ty = type v + return if is_type v, ty, arg + error "Argument ##{i} should be a #{arg}, is #{ty} (#{v})", 3 + + (fn) -> (...) -> + arg_count = select '#', ... + if arg_count < min_args or (max_args and arg_count > max_args) + if min_args == max_args + error "Expected #{min_args} arguments, got #{arg_count}", 3 + else if max_args + error "Expected #{min_args}-#{max_args} arguments, got #{arg_count}", 3 + else + error "Expected at least #{min_args} arguments, got #{arg_count}", 3 + + for i=1,arg_count + if not checks[i] then break + checks[i] i, select i, ... + + fn ... + diff --git a/automation/include/aegisub/clipboard.lua b/automation/include/aegisub/clipboard.lua index 4978607205..00b156ce32 100644 --- a/automation/include/aegisub/clipboard.lua +++ b/automation/include/aegisub/clipboard.lua @@ -1,4 +1,4 @@ --- Copyright (c) 2012, Thomas Goyne +-- Copyright (c) 2012, Thomas Goyne -- -- Permission to use, copy, modify, and distribute this software for any -- purpose with or without fee is hereby granted, provided that the above @@ -12,11 +12,4 @@ -- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF -- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. -local check = require 'aegisub.argcheck' -local ffi_util = require 'aegisub.ffi' -local impl = aegisub.__init_clipboard() - -return { - get = function() return ffi_util.string(impl.get()) end, - set = check'string'(impl.set) -} +return aegisub.__init_clipboard() diff --git a/automation/include/aegisub/ffi.moon b/automation/include/aegisub/ffi.moon index b58d1e5e5b..e6b64f18bd 100644 --- a/automation/include/aegisub/ffi.moon +++ b/automation/include/aegisub/ffi.moon @@ -1,48 +1,48 @@ --- Copyright (c) 2014, Thomas Goyne --- --- Permission to use, copy, modify, and distribute this software for any --- purpose with or without fee is hereby granted, provided that the above --- copyright notice and this permission notice appear in all copies. --- --- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES --- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF --- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR --- ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES --- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN --- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF --- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. --- --- Aegisub Project http://www.aegisub.org/ - -ffi = require 'ffi' -ffi.cdef[[ - void free(void *ptr); -]] - -char_ptr = ffi.typeof 'char *' - --- Convert a const char * to a lua string, returning nil rather than crashing --- if it's NULL and freeing the input if it's non-const -string = (cdata) -> - return nil if cdata == nil - str = ffi.string cdata - if type(cdata) == char_ptr - ffi.C.free cdata - str - -err_buff = ffi.new 'char *[1]' - --- Convert a function which has an error out parameter to one which returns --- the original return value and the error as a string -err_arg_to_multiple_return = (f) -> (arg) -> - err_buff[0] = nil - result = if arg != nil - f arg, err_buff - else - f err_buff - errmsg = string err_buff[0] - if errmsg - return nil, errmsg - return result - -{:string, :err_arg_to_multiple_return} +-- Copyright (c) 2014, Thomas Goyne +-- +-- Permission to use, copy, modify, and distribute this software for any +-- purpose with or without fee is hereby granted, provided that the above +-- copyright notice and this permission notice appear in all copies. +-- +-- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +-- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +-- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +-- ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +-- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +-- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +-- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. +-- +-- Aegisub Project http://www.aegisub.org/ + +ffi = require 'ffi' +ffi.cdef[[ + void free(void *ptr); +]] + +char_ptr = ffi.typeof 'char *' + +-- Convert a const char * to a lua string, returning nil rather than crashing +-- if it's NULL and freeing the input if it's non-const +string = (cdata) -> + return nil if cdata == nil + str = ffi.string cdata + if type(cdata) == char_ptr + ffi.C.free cdata + str + +err_buff = ffi.new 'char *[1]' + +-- Convert a function which has an error out parameter to one which returns +-- the original return value and the error as a string +err_arg_to_multiple_return = (f) -> (arg) -> + err_buff[0] = nil + result = if arg != nil + f arg, err_buff + else + f err_buff + errmsg = string err_buff[0] + if errmsg + return nil, errmsg + return result + +{:string, :err_arg_to_multiple_return} diff --git a/automation/include/aegisub/lfs.moon b/automation/include/aegisub/lfs.moon index 5a70f023ff..e1feeb05c7 100644 --- a/automation/include/aegisub/lfs.moon +++ b/automation/include/aegisub/lfs.moon @@ -1,78 +1,78 @@ --- Copyright (c) 2014, Thomas Goyne --- --- Permission to use, copy, modify, and distribute this software for any --- purpose with or without fee is hereby granted, provided that the above --- copyright notice and this permission notice appear in all copies. --- --- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES --- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF --- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR --- ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES --- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN --- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF --- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. --- --- Aegisub Project http://www.aegisub.org/ - -impl = require 'aegisub.__lfs_impl' - -check = require 'aegisub.argcheck' -ffi = require 'ffi' -ffi_util = require 'aegisub.ffi' - -for k, v in pairs impl - impl[k] = ffi_util.err_arg_to_multiple_return v - -string_ret = (f) -> (...) -> - res, err = f ... - ffi_util.string(res), err - -attributes = check'string ?string' (path, field) -> - switch field - when 'mode' - res, err = impl.get_mode path - ffi_util.string(res), err - when 'modification' - res, err = impl.get_mtime path - tonumber(res), err - when 'size' - res, err = impl.get_size path - tonumber(res), err - else - mode, err = impl.get_mode path - if err or mode == nil then return nil, err - - mod, err = impl.get_mtime path - if err then return nil, err - - size, err = impl.get_size path - if err then return nil, err - - mode: ffi_util.string(mode), modification: tonumber(mod), size: tonumber(size) - -class dir_iter - new: (iter) => - @iter = ffi.gc iter, -> impl.dir_free iter - close: => - impl.dir_close @iter - next: => - str, err = impl.dir_next @iter - if err then error err, 2 - ffi_util.string str - -dir = check'string' (path) -> - obj, err = impl.dir_new path - if err - error 2, err - iter = dir_iter obj - iter.next, iter - -return { - :attributes - chdir: check'string' impl.chdir - currentdir: check'' string_ret impl.currentdir - :dir - mkdir: check'string' impl.mkdir - rmdir: check'string' impl.rmdir - touch: check'string' impl.touch -} +-- Copyright (c) 2014, Thomas Goyne +-- +-- Permission to use, copy, modify, and distribute this software for any +-- purpose with or without fee is hereby granted, provided that the above +-- copyright notice and this permission notice appear in all copies. +-- +-- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +-- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +-- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +-- ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +-- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +-- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +-- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. +-- +-- Aegisub Project http://www.aegisub.org/ + +impl = require 'aegisub.__lfs_impl' + +check = require 'aegisub.argcheck' +ffi = require 'ffi' +ffi_util = require 'aegisub.ffi' + +for k, v in pairs impl + impl[k] = ffi_util.err_arg_to_multiple_return v + +string_ret = (f) -> (...) -> + res, err = f ... + ffi_util.string(res), err + +attributes = check'string ?string' (path, field) -> + switch field + when 'mode' + res, err = impl.get_mode path + ffi_util.string(res), err + when 'modification' + res, err = impl.get_mtime path + tonumber(res), err + when 'size' + res, err = impl.get_size path + tonumber(res), err + else + mode, err = impl.get_mode path + if err or mode == nil then return nil, err + + mod, err = impl.get_mtime path + if err then return nil, err + + size, err = impl.get_size path + if err then return nil, err + + mode: ffi_util.string(mode), modification: tonumber(mod), size: tonumber(size) + +class dir_iter + new: (iter) => + @iter = ffi.gc iter, -> impl.dir_free iter + close: => + impl.dir_close @iter + next: => + str, err = impl.dir_next @iter + if err then error err, 2 + ffi_util.string str + +dir = check'string' (path) -> + obj, err = impl.dir_new path + if err + error 2, err + iter = dir_iter obj + iter.next, iter + +return { + :attributes + chdir: check'string' impl.chdir + currentdir: check'' string_ret impl.currentdir + :dir + mkdir: check'string' impl.mkdir + rmdir: check'string' impl.rmdir + touch: check'string' impl.touch +} diff --git a/automation/include/aegisub/re.moon b/automation/include/aegisub/re.moon index 28b825da80..b571b74bdd 100644 --- a/automation/include/aegisub/re.moon +++ b/automation/include/aegisub/re.moon @@ -17,80 +17,38 @@ next = next select = select type = type -bit = require 'bit' -ffi = require 'ffi' -ffi_util = require 'aegisub.ffi' -check = require 'aegisub.argcheck' - -ffi.cdef[[ - typedef struct agi_re_flag { - const char *name; - int value; - } agi_re_flag; -]] -regex_flag = ffi.typeof 'agi_re_flag' - --- Get the boost::eegex binding +-- Get the boost::regex binding regex = require 'aegisub.__re_impl' --- Wrappers to convert returned values from C types to Lua types -search = (re, str, start) -> - return unless start <= str\len() - res = regex.search re, str, str\len(), start - return unless res != nil - first, last = res[0], res[1] - ffi.gc(res, ffi.C.free) - first, last - -replace = (re, replacement, str, max_count) -> - ffi_util.string regex.replace re, replacement, str, str\len(), max_count - -match = (re, str, start) -> - assert start <= str\len() - m = regex.match re, str, str\len(), start - return unless m != nil - ffi.gc m, regex.match_free - -get_match = (m, idx) -> - res = regex.get_match m, idx - return unless res != nil - res[0], res[1] -- Result buffer is owned by match so no need to free - -err_buff = ffi.new 'char *[1]' -compile = (pattern, flags) -> - err_buff[0] = nil - re = regex.compile pattern, flags, err_buff - if err_buff[0] != nil - return ffi.string err_buff[0] - ffi.gc re, regex.regex_free - -- Return the first n elements from ... select_first = (n, a, ...) -> if n == 0 then return a, select_first n - 1, ... --- Bitwise-or together regex flags passed as arguments to a function -process_flags = (...) -> - flags = 0 - for i = 1, select '#', ... - v = select i, ... - if not ffi.istype regex_flag, v - error 'Flags must follow all non-flag arguments', 3 - flags = bit.bor flags, v.value - flags - -- Extract the flags from ..., bitwise OR them together, and move them to the -- front of ... unpack_args = (...) -> - flags_start = nil + userdata_start = nil for i = 1, select '#', ... v = select i, ... - if ffi.istype regex_flag, v - flags_start = i + if type(v) == 'userdata' + userdata_start = i break - return 0, ... unless flags_start - process_flags(select flags_start, ...), select_first flags_start - 1, ... + return 0, ... unless userdata_start + + flags = regex.process_flags select userdata_start, ... + if type(flags) == 'string' + error(flags, 3) + + flags, select_first userdata_start - 1, ... + + +-- Typecheck a variable and throw an error if it fails +check_arg = (arg, expected_type, argn, func_name, level) -> + if type(arg) != expected_type + error "Argument #{argn} to #{func_name} should be a '#{expected_type}', is '#{type(arg)}' (#{arg})", + level + 1 -- Replace a match with the value returned from func when passed the match replace_match = (match, func, str, last, acc) -> @@ -150,27 +108,30 @@ class RegEx new: (@_regex, @_level) => - gsplit: check'RegEx string ?boolean ?number' (str, skip_empty, max_split) => + start = 1 + gsplit: (str, skip_empty, max_split) => + @_check_self! + check_arg str, 'string', 2, 'gsplit', @_level if not max_split or max_split <= 0 then max_split = str\len() - start = 0 + start = 1 prev = 1 do_split = () -> if not str or str\len() == 0 then return local first, last if max_split > 0 - first, last = search @_regex, str, start + first, last = regex.search @_regex, str, start if not first or first > str\len() ret = str\sub prev, str\len() str = nil - return if skip_empty and ret\len() == 0 then nil else ret + return ret ret = str\sub prev, first - 1 prev = last + 1 - start = if start >= last then start + 1 else last + start = 1 + if start >= last then start else last if skip_empty and ret\len() == 0 do_split() @@ -180,38 +141,55 @@ class RegEx do_split - split: check'RegEx string ?boolean ?number' (str, skip_empty, max_split) => + split: (str, skip_empty, max_split) => + @_check_self! + check_arg str, 'string', 2, 'split', @_level [v for v in @gsplit str, skip_empty, max_split] - gfind: check'RegEx string' (str) => - start = 0 + gfind: (str) => + @_check_self! + check_arg str, 'string', 2, 'gfind', @_level + + start = 1 -> - first, last = search(@_regex, str, start) + first, last = regex.search(@_regex, str, start) return unless first - start = if last > start then last else start + 1 + start = if last >= start then last + 1 else start + 1 str\sub(first, last), first, last - find: check'RegEx string' (str) => + find: (str) => + @_check_self! + check_arg str, 'string', 2, 'find', @_level + ret = [str: s, first: f, last: l for s, f, l in @gfind(str)] next(ret) and ret - sub: check'RegEx string string|function ?number' (str, repl, max_count) => + sub: (str, repl, max_count) => + @_check_self! + check_arg str, 'string', 2, 'sub', @_level + if max_count != nil + check_arg max_count, 'number', 4, 'sub', @_level + max_count = str\len() + 1 if not max_count or max_count == 0 if type(repl) == 'function' do_replace_fun @, repl, str, max_count elseif type(repl) == 'string' - replace @_regex, repl, str, max_count + regex.replace @_regex, repl, str, max_count + else + error "Argument 2 to sub should be a string or function, is '#{type(repl)}' (#{repl})", @_level - gmatch: check'RegEx string ?number' (str, start) => + gmatch: (str, start) => + @_check_self! + check_arg str, 'string', 2, 'gmatch', @_level start = if start then start - 1 else 0 - m = match @_regex, str, start - i = 0 + match = regex.match @_regex, str, start + i = 1 -> - return unless m - first, last = get_match m, i + return unless match + first, last = regex.get_match match, i return unless first i += 1 @@ -221,7 +199,10 @@ class RegEx last: last + start } - match: check'RegEx string ?number' (str, start) => + match: (str, start) => + @_check_self! + check_arg(str, 'string', 2, 'match', @_level) + ret = [v for v in @gmatch str, start] -- Return nil rather than a empty table so that if re.match(...) works return nil if next(ret) == nil @@ -232,7 +213,7 @@ real_compile = (pattern, level, flags, stored_level) -> if pattern == '' error 'Regular expression must not be empty', level + 1 - re = compile pattern, flags + re = regex.compile pattern, flags if type(re) == 'string' error regex, level + 1 @@ -244,28 +225,25 @@ invoke = (str, pattern, fn, flags, ...) -> compiled_regex[fn](compiled_regex, str, ...) -- Generate a static version of a method with arg type checking -gen_wrapper = (impl_name) -> check'string string ...' (str, pattern, ...) -> - invoke str, pattern, impl_name, unpack_args ... +gen_wrapper = (impl_name) -> + (str, pattern, ...) -> + check_arg str, 'string', 1, impl_name, 2 + check_arg pattern, 'string', 2, impl_name, 2 + invoke str, pattern, impl_name, unpack_args ... -- And now at last the actual public API -do - re = { - compile: check'string ...' (pattern, ...) -> - real_compile pattern, 2, process_flags(...), 2 - - split: gen_wrapper 'split' - gsplit: gen_wrapper 'gsplit' - find: gen_wrapper 'find' - gfind: gen_wrapper 'gfind' - match: gen_wrapper 'match' - gmatch: gen_wrapper 'gmatch' - sub: gen_wrapper 'sub' - } - - i = 0 - flags = regex.get_flags() - while flags[i].name != nil - re[ffi.string flags[i].name] = flags[i] - i += 1 - - re +re = regex.init_flags(re) + +re.compile = (pattern, ...) -> + check_arg pattern, 'string', 1, 'compile', 2 + real_compile pattern, 2, regex.process_flags(...), 2 + +re.split = gen_wrapper 'split' +re.gsplit = gen_wrapper 'gsplit' +re.find = gen_wrapper 'find' +re.gfind = gen_wrapper 'gfind' +re.match = gen_wrapper 'match' +re.gmatch = gen_wrapper 'gmatch' +re.sub = gen_wrapper 'sub' + +re diff --git a/automation/include/aegisub/unicode.moon b/automation/include/aegisub/unicode.moon index c807d10438..672ff7a2ac 100644 --- a/automation/include/aegisub/unicode.moon +++ b/automation/include/aegisub/unicode.moon @@ -1,4 +1,4 @@ --- Copyright (c) 2007, Niels Martin Hansen, Rodrigo Braz Monteiro +-- Copyright (c) 2007, Niels Martin Hansen, Rodrigo Braz Monteiro -- All rights reserved. -- -- Redistribution and use in source and binary forms, with or without @@ -30,23 +30,10 @@ impl = require 'aegisub.__unicode_impl' -check = require 'aegisub.argcheck' -ffi = require 'ffi' -ffi_util = require 'aegisub.ffi' - -err_buff = ffi.new 'char *[1]' -conv_func = (f) -> check'string' (str) -> - err_buff[0] = nil - result = f str, err_buff - errmsg = ffi_util.string err_buff[0] - if errmsg - error errmsg, 2 - ffi_util.string result - local unicode unicode = -- Return the number of bytes occupied by the character starting at the i'th byte in s - charwidth: check'string ?number' (s, i) -> + charwidth: (s, i) -> b = s\byte i or 1 -- FIXME, something in karaskel results in this case, shouldn't happen -- What would "proper" behaviour be? Zero? Or just explode? @@ -57,7 +44,7 @@ unicode = else 4 -- Returns an iterator function for iterating over the characters in s - chars: check'string' (s) -> + chars: (s) -> curchar, i = 0, 1 -> return if i > s\len() @@ -69,13 +56,13 @@ unicode = -- Returns the number of characters in s -- Runs in O(s:len()) time! - len: check'string' (s) -> + len: (s) -> n = 0 n += 1 for c in unicode.chars s n -- Get codepoint of first char in s - codepoint: check'string' (s) -> + codepoint: (s) -> -- Basic case, ASCII b = s\byte 1 return b if b < 128 @@ -99,8 +86,8 @@ unicode = res = res*64 + s\byte(i) - 128 res - to_upper_case: conv_func impl.to_upper_case - to_lower_case: conv_func impl.to_lower_case - to_fold_case: conv_func impl.to_fold_case + to_upper_case: impl.to_upper_case + to_lower_case: impl.to_lower_case + to_fold_case: impl.to_fold_case return unicode diff --git a/automation/include/aegisub/util.moon b/automation/include/aegisub/util.moon index 42a642ee67..b9ff61ae3d 100644 --- a/automation/include/aegisub/util.moon +++ b/automation/include/aegisub/util.moon @@ -20,16 +20,14 @@ sformat = string.format tonumber = tonumber type = type -check = require 'aegisub.argcheck' - local * -- Make a shallow copy of a table -copy = check'table' (tbl) -> {k, v for k, v in pairs tbl} +copy = (tbl) -> {k, v for k, v in pairs tbl} -- Make a deep copy of a table -- Retains equality of table references inside the copy and handles self-referencing structures -deep_copy = check'table' (tbl) -> +deep_copy = (tbl) -> seen = {} copy = (val) -> return val if type(val) != 'table' @@ -46,7 +44,7 @@ ass_alpha = (a) -> sformat "&H%02X&", a ass_style_color = (r, g, b, a) -> sformat "&H%02X%02X%02X%02X", a, b, g, r -- Extract colour components of an ASS colour -extract_color = check'string' (s) -> +extract_color = (s) -> local a, b, g, r -- Try a style first @@ -70,10 +68,10 @@ extract_color = check'string' (s) -> return tonumber(r, 16), tonumber(g, 16) or 0, tonumber(b, 16) or 0, tonumber(a, 16) or 0 -- Create an alpha override code from a style definition colour code -alpha_from_style = check'string' (scolor) -> ass_alpha select 4, extract_color scolor +alpha_from_style = (scolor) -> ass_alpha select 4, extract_color scolor -- Create an colour override code from a style definition colour code -color_from_style = check'string' (scolor) -> +color_from_style = (scolor) -> r, g, b = extract_color scolor ass_color r or 0, g or 0, b or 0 diff --git a/automation/include/cleantags.lua b/automation/include/cleantags.lua index 51bc016b88..0ca65daa9b 100644 --- a/automation/include/cleantags.lua +++ b/automation/include/cleantags.lua @@ -1,4 +1,4 @@ ---[[ +--[[ "Clean Tags" -- An Auto4 LUA script for cleaning up ASS subtitle lines of badly-formed override blocks and redundant/duplicate tags * Designed to work for Aegisub 2.0 and above diff --git a/automation/include/clipboard.lua b/automation/include/clipboard.lua index d65b080c09..5033add1be 100644 --- a/automation/include/clipboard.lua +++ b/automation/include/clipboard.lua @@ -1,4 +1,4 @@ --- Copyright (c) 2012, Thomas Goyne +-- Copyright (c) 2012, Thomas Goyne -- -- Permission to use, copy, modify, and distribute this software for any -- purpose with or without fee is hereby granted, provided that the above @@ -12,5 +12,5 @@ -- ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF -- OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. -_G.clipboard = require 'aegisub.clipboard' +_G.clipboard = aegisub.__init_clipboard() return _G.clipboard diff --git a/automation/include/karaskel-auto4.lua b/automation/include/karaskel-auto4.lua index f5f17a708b..275cb56aa3 100644 --- a/automation/include/karaskel-auto4.lua +++ b/automation/include/karaskel-auto4.lua @@ -52,13 +52,12 @@ function karaskel.collect_head(subs, generate_furigana) end -- First pass: collect all existing styles and get resolution info - for i, l in ipairs(subs) do + for i = 1, #subs do if aegisub.progress.is_cancelled() then error("User cancelled") end + local l = subs[i] if l.class == "style" then - if not first_style_line then - first_style_line = i - end + if not first_style_line then first_style_line = i end -- Store styles into the style table styles.n = styles.n + 1 styles[styles.n] = l @@ -67,7 +66,7 @@ function karaskel.collect_head(subs, generate_furigana) -- And also generate furigana styles if wanted if generate_furigana and not l.name:match("furigana") then - aegisub.debug.out(5, "Creating furigana style for style: %s\n", l.name) + aegisub.debug.out(5, "Creating furigana style for style: " .. l.name .. "\n") local fs = table.copy(l) fs.fontsize = l.fontsize * karaskel.furigana_scale fs.outline = l.outline * karaskel.furigana_scale @@ -81,15 +80,9 @@ function karaskel.collect_head(subs, generate_furigana) -- Also look for script resolution local k = l.key:lower() meta[k] = l.value - else - break end end - if first_style_line == nil then - error("Karaskel error: No styles were found in the file, bug?!") - end - -- Second pass: insert all toinsert styles that don't already exist for i = 1, #toinsert do if not styles[toinsert[i].name] then @@ -103,7 +96,30 @@ function karaskel.collect_head(subs, generate_furigana) end -- Fix resolution data - meta.res_x, meta.res_y = subs.script_resolution() + if meta.playresx then + meta.res_x = math.floor(meta.playresx) + end + if meta.playresy then + meta.res_y = math.floor(meta.playresy) + end + if meta.res_x == 0 and meta_res_y == 0 then + meta.res_x = 384 + meta.res_y = 288 + elseif meta.res_x == 0 then + -- This is braindead, but it's how TextSub does things... + if meta.res_y == 1024 then + meta.res_x = 1280 + else + meta.res_x = meta.res_y / 3 * 4 + end + elseif meta.res_y == 0 then + -- As if 1280x960 didn't exist + if meta.res_x == 1280 then + meta.res_y = 1024 + else + meta.res_y = meta.res_x * 3 / 4 + end + end local video_x, video_y = aegisub.video_size() if video_y then diff --git a/automation/include/karaskel.lua b/automation/include/karaskel.lua index 3dea6b8cbd..7a1f2e4f67 100644 --- a/automation/include/karaskel.lua +++ b/automation/include/karaskel.lua @@ -1,4 +1,4 @@ ---[[ +--[[ Copyright (c) 2007, Niels Martin Hansen, Rodrigo Braz Monteiro All rights reserved. diff --git a/automation/include/lfs.lua b/automation/include/lfs.lua index 4b4e542f3e..407bdbfcb2 100644 --- a/automation/include/lfs.lua +++ b/automation/include/lfs.lua @@ -1,2 +1,2 @@ -lfs = require 'aegisub.lfs' -return lfs +lfs = require 'aegisub.lfs' +return lfs diff --git a/automation/include/moonscript.lua b/automation/include/moonscript.lua index 78806c8709..88257f5ca7 100644 --- a/automation/include/moonscript.lua +++ b/automation/include/moonscript.lua @@ -1,839 +1,219 @@ -package.preload['moonscript.transform.comprehension'] = function() - local is_value - is_value = require("moonscript.types").is_value - local construct_comprehension - construct_comprehension = function(inner, clauses) - local current_stms = inner - for i = #clauses, 1, -1 do - local clause = clauses[i] - local t = clause[1] - local _exp_0 = t - if "for" == _exp_0 then - local _, name, bounds - _, name, bounds = clause[1], clause[2], clause[3] - current_stms = { - "for", - name, - bounds, - current_stms - } - elseif "foreach" == _exp_0 then - local _, names, iter - _, names, iter = clause[1], clause[2], clause[3] - current_stms = { - "foreach", - names, - { - iter - }, - current_stms - } - elseif "when" == _exp_0 then - local _, cond - _, cond = clause[1], clause[2] - current_stms = { - "if", - cond, - current_stms - } - else - current_stms = error("Unknown comprehension clause: " .. t) +package.preload['moonscript.base'] = function() + local compile = require("moonscript.compile") + local parse = require("moonscript.parse") + local concat, insert, remove + do + local _obj_0 = table + concat, insert, remove = _obj_0.concat, _obj_0.insert, _obj_0.remove + end + local split, dump, get_options, unpack + do + local _obj_0 = require("moonscript.util") + split, dump, get_options, unpack = _obj_0.split, _obj_0.dump, _obj_0.get_options, _obj_0.unpack + end + local lua = { + loadstring = loadstring, + load = load + } + local dirsep, line_tables, create_moonpath, to_lua, moon_loader, loadstring, loadfile, dofile, insert_loader, remove_loader + dirsep = "/" + line_tables = require("moonscript.line_tables") + create_moonpath = function(package_path) + local paths = split(package_path, ";") + for i, path in ipairs(paths) do + local p = path:match("^(.-)%.lua$") + if p then + paths[i] = p .. ".moon" end - current_stms = { - current_stms - } end - return current_stms[1] + return concat(paths, ";") end - local comprehension_has_value - comprehension_has_value = function(comp) - return is_value(comp[2]) + to_lua = function(text, options) + if options == nil then + options = { } + end + if "string" ~= type(text) then + local t = type(text) + return nil, "expecting string (got " .. t .. ")", 2 + end + local tree, err = parse.string(text) + if not tree then + return nil, err + end + local code, ltable, pos = compile.tree(tree, options) + if not code then + return nil, compile.format_error(ltable, pos, text), 2 + end + return code, ltable + end + loadstring = function(...) + local options, str, chunk_name, mode, env = get_options(...) + chunk_name = chunk_name or "=(moonscript.loadstring)" + local code, ltable_or_err = to_lua(str, options) + if not (code) then + return nil, ltable_or_err + end + if chunk_name then + line_tables[chunk_name] = ltable_or_err + end + return (lua.loadstring or lua.load)(code, chunk_name, unpack({ + mode, + env + })) + end + loadfile = function(fname, ...) + local file, err = io.open(fname) + if not (file) then + return nil, err + end + local text = assert(file:read("*a")) + file:close() + return loadstring(text, fname, ...) + end + dofile = function(...) + local f = assert(loadfile(...)) + return f() end return { - construct_comprehension = construct_comprehension, - comprehension_has_value = comprehension_has_value + _NAME = "moonscript", + to_lua = to_lua, + moon_chunk = moon_chunk, + dirsep = dirsep, + dofile = dofile, + loadfile = loadfile, + loadstring = loadstring } end -package.preload['moonscript.transform.value'] = function() - local Transformer - Transformer = require("moonscript.transform.transformer").Transformer - local build, ntype, smart_node - do - local _obj_0 = require("moonscript.types") - build, ntype, smart_node = _obj_0.build, _obj_0.ntype, _obj_0.smart_node - end - local NameProxy - NameProxy = require("moonscript.transform.names").NameProxy - local Accumulator, default_accumulator - do - local _obj_0 = require("moonscript.transform.accumulator") - Accumulator, default_accumulator = _obj_0.Accumulator, _obj_0.default_accumulator - end - local lua_keywords - lua_keywords = require("moonscript.data").lua_keywords - local Run, transform_last_stm, implicitly_return, chain_is_stub - do - local _obj_0 = require("moonscript.transform.statements") - Run, transform_last_stm, implicitly_return, chain_is_stub = _obj_0.Run, _obj_0.transform_last_stm, _obj_0.implicitly_return, _obj_0.chain_is_stub +package.preload['moonscript.cmd.coverage'] = function() + local log + log = function(str) + if str == nil then + str = "" + end + return io.stderr:write(str .. "\n") end - local construct_comprehension - construct_comprehension = require("moonscript.transform.comprehension").construct_comprehension - local insert - insert = table.insert - local unpack - unpack = require("moonscript.util").unpack - return Transformer({ - ["for"] = default_accumulator, - ["while"] = default_accumulator, - foreach = default_accumulator, - ["do"] = function(self, node) - return build.block_exp(node[2]) - end, - decorated = function(self, node) - return self.transform.statement(node) - end, - class = function(self, node) - return build.block_exp({ - node - }) - end, - string = function(self, node) - local delim = node[2] - local convert_part - convert_part = function(part) - if type(part) == "string" or part == nil then - return { - "string", - delim, - part or "" - } - else - return build.chain({ - base = "tostring", - { - "call", - { - part[2] - } - } + local create_counter + create_counter = function() + return setmetatable({ }, { + __index = function(self, name) + do + local tbl = setmetatable({ }, { + __index = function(self) + return 0 + end }) + self[name] = tbl + return tbl end end - if #node <= 3 then - if type(node[3]) == "string" then - return node - else - return convert_part(node[3]) + }) + end + local position_to_lines + position_to_lines = function(file_content, positions) + local lines = { } + local current_pos = 0 + local line_no = 1 + for char in file_content:gmatch(".") do + do + local count = rawget(positions, current_pos) + if count then + lines[line_no] = count end end - local e = { - "exp", - convert_part(node[3]) - } - for i = 4, #node do - insert(e, "..") - insert(e, convert_part(node[i])) - end - return e - end, - comprehension = function(self, node) - local a = Accumulator() - node = self.transform.statement(node, function(exp) - return a:mutate_body({ - exp - }) - end) - return a:wrap(node) - end, - tblcomprehension = function(self, node) - local explist, clauses = unpack(node, 2) - local key_exp, value_exp = unpack(explist) - local accum = NameProxy("tbl") - local inner - if value_exp then - local dest = build.chain({ - base = accum, - { - "index", - key_exp - } - }) - inner = { - build.assign_one(dest, value_exp) - } - else - local key_name, val_name = NameProxy("key"), NameProxy("val") - local dest = build.chain({ - base = accum, - { - "index", - key_name - } - }) - inner = { - build.assign({ - names = { - key_name, - val_name - }, - values = { - key_exp - } - }), - build.assign_one(dest, val_name) - } + if char == "\n" then + line_no = line_no + 1 end - return build.block_exp({ - build.assign_one(accum, build.table()), - construct_comprehension(inner, clauses), - accum - }) - end, - fndef = function(self, node) - smart_node(node) - node.body = transform_last_stm(node.body, implicitly_return(self)) - node.body = { - Run(function(self) - return self:listen("varargs", function() end) - end), - unpack(node.body) - } - return node - end, - ["if"] = function(self, node) - return build.block_exp({ - node - }) - end, - unless = function(self, node) - return build.block_exp({ - node - }) - end, - with = function(self, node) - return build.block_exp({ - node - }) - end, - switch = function(self, node) - return build.block_exp({ - node - }) - end, - chain = function(self, node) - for i = 2, #node do - local part = node[i] - if ntype(part) == "dot" and lua_keywords[part[2]] then - node[i] = { - "index", - { - "string", - '"', - part[2] - } - } - end - end - if ntype(node[2]) == "string" then - node[2] = { - "parens", - node[2] - } - end - if chain_is_stub(node) then - local base_name = NameProxy("base") - local fn_name = NameProxy("fn") - local colon = table.remove(node) - local is_super = ntype(node[2]) == "ref" and node[2][2] == "super" - return build.block_exp({ - build.assign({ - names = { - base_name - }, - values = { - node - } - }), - build.assign({ - names = { - fn_name - }, - values = { - build.chain({ - base = base_name, - { - "dot", - colon[2] - } - }) - } - }), - build.fndef({ - args = { - { - "..." - } - }, - body = { - build.chain({ - base = fn_name, - { - "call", - { - is_super and "self" or base_name, - "..." - } - } - }) - } - }) - }) - end - end, - block_exp = function(self, node) - local body = unpack(node, 2) - local fn = nil - local arg_list = { } - fn = smart_node(build.fndef({ - body = { - Run(function(self) - return self:listen("varargs", function() - insert(arg_list, "...") - insert(fn.args, { - "..." - }) - return self:unlisten("varargs") - end) - end), - unpack(body) - } - })) - return build.chain({ - base = { - "parens", - fn - }, - { - "call", - arg_list - } - }) + current_pos = current_pos + 1 end - }) - -end -package.preload['moonscript.transform.class'] = function() - local NameProxy, LocalName - do - local _obj_0 = require("moonscript.transform.names") - NameProxy, LocalName = _obj_0.NameProxy, _obj_0.LocalName - end - local Run - Run = require("moonscript.transform.statements").Run - local CONSTRUCTOR_NAME = "new" - local insert - insert = table.insert - local build, ntype, NOOP - do - local _obj_0 = require("moonscript.types") - build, ntype, NOOP = _obj_0.build, _obj_0.ntype, _obj_0.NOOP + return lines end - local unpack - unpack = require("moonscript.util").unpack - local transform_super - transform_super = function(cls_name, on_base, block, chain) - if on_base == nil then - on_base = true - end - local relative_parent = { - "chain", - cls_name, - { - "dot", - "__parent" - } - } - if not (chain) then - return relative_parent - end - local chain_tail = { - unpack(chain, 3) - } - local head = chain_tail[1] - if head == nil then - return relative_parent - end - local new_chain = relative_parent - local _exp_0 = head[1] - if "call" == _exp_0 then - if on_base then - insert(new_chain, { - "dot", - "__base" - }) - end - local calling_name = block:get("current_method") - assert(calling_name, "missing calling name") - chain_tail[1] = { - "call", - { - "self", - unpack(head[2]) - } - } - if ntype(calling_name) == "key_literal" then - insert(new_chain, { - "dot", - calling_name[2] - }) - else - insert(new_chain, { - "index", - calling_name - }) - end - elseif "colon" == _exp_0 then - local call = chain_tail[2] - if call and call[1] == "call" then - chain_tail[1] = { - "dot", - head[2] - } - chain_tail[2] = { - "call", - { - "self", - unpack(call[2]) - } - } - end - end - for _index_0 = 1, #chain_tail do - local item = chain_tail[_index_0] - insert(new_chain, item) + local format_file + format_file = function(fname, positions) + local file = assert(io.open(fname)) + local content = file:read("*a") + file:close() + local lines = position_to_lines(content, positions) + log("------| @" .. tostring(fname)) + local line_no = 1 + for line in (content .. "\n"):gmatch("(.-)\n") do + local foramtted_no = ("% 5d"):format(line_no) + local sym = lines[line_no] and "*" or " " + log(tostring(sym) .. tostring(foramtted_no) .. "| " .. tostring(line)) + line_no = line_no + 1 end - return new_chain - end - local super_scope - super_scope = function(value, t, key) - local prev_method - return { - "scoped", - Run(function(self) - prev_method = self:get("current_method") - self:set("current_method", key) - return self:set("super", t) - end), - value, - Run(function(self) - return self:set("current_method", prev_method) - end) - } + return log() end - return function(self, node, ret, parent_assign) - local name, parent_val, body = unpack(node, 2) - if parent_val == "" then - parent_val = nil - end - local parent_cls_name = NameProxy("parent") - local base_name = NameProxy("base") - local self_name = NameProxy("self") - local cls_name = NameProxy("class") - local cls_instance_super - cls_instance_super = function(...) - return transform_super(cls_name, true, ...) - end - local cls_super - cls_super = function(...) - return transform_super(cls_name, false, ...) - end - local statements = { } - local properties = { } - for _index_0 = 1, #body do - local item = body[_index_0] - local _exp_0 = item[1] - if "stm" == _exp_0 then - insert(statements, item[2]) - elseif "props" == _exp_0 then - for _index_1 = 2, #item do - local tuple = item[_index_1] - if ntype(tuple[1]) == "self" then - local k, v - k, v = tuple[1], tuple[2] - v = super_scope(v, cls_super, { - "key_literal", - k[2] - }) - insert(statements, build.assign_one(k, v)) - else - insert(properties, tuple) + local CodeCoverage + do + local _base_0 = { + reset = function(self) + self.line_counts = create_counter() + end, + start = function(self) + return debug.sethook((function() + local _base_1 = self + local _fn_0 = _base_1.process_line + return function(...) + return _fn_0(_base_1, ...) end - end - end - end - local constructor - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #properties do - local _continue_0 = false - repeat - local tuple = properties[_index_0] - local key = tuple[1] - local _value_0 - if key[1] == "key_literal" and key[2] == CONSTRUCTOR_NAME then - constructor = tuple[2] + end)(), "l") + end, + stop = function(self) + return debug.sethook() + end, + print_results = function(self) + return self:format_results() + end, + process_line = function(self, _, line_no) + local debug_data = debug.getinfo(2, "S") + local source = debug_data.source + self.line_counts[source][line_no] = self.line_counts[source][line_no] + 1 + end, + format_results = function(self) + local line_table = require("moonscript.line_tables") + local positions = create_counter() + for file, lines in pairs(self.line_counts) do + local _continue_0 = false + repeat + local file_table = line_table[file] + if not (file_table) then + _continue_0 = true + break + end + for line, count in pairs(lines) do + local _continue_1 = false + repeat + local position = file_table[line] + if not (position) then + _continue_1 = true + break + end + positions[file][position] = positions[file][position] + count + _continue_1 = true + until true + if not _continue_1 then + break + end + end _continue_0 = true + until true + if not _continue_0 then break - else - local val - key, val = tuple[1], tuple[2] - _value_0 = { - key, - super_scope(val, cls_instance_super, key) - } end - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - _continue_0 = true - until true - if not _continue_0 then - break + end + for file, ps in pairs(positions) do + format_file(file, ps) end end - properties = _accum_0 - end - if not (constructor) then - if parent_val then - constructor = build.fndef({ - args = { - { - "..." - } - }, - arrow = "fat", - body = { - build.chain({ - base = "super", - { - "call", - { - "..." - } - } - }) - } - }) - else - constructor = build.fndef() - end - end - local real_name = name or parent_assign and parent_assign[2][1] - local _exp_0 = ntype(real_name) - if "chain" == _exp_0 then - local last = real_name[#real_name] - local _exp_1 = ntype(last) - if "dot" == _exp_1 then - real_name = { - "string", - '"', - last[2] - } - elseif "index" == _exp_1 then - real_name = last[2] - else - real_name = "nil" - end - elseif "nil" == _exp_0 then - real_name = "nil" - else - local name_t = type(real_name) - local flattened_name - if name_t == "string" then - flattened_name = real_name - elseif name_t == "table" and real_name[1] == "ref" then - flattened_name = real_name[2] - else - flattened_name = error("don't know how to extract name from " .. tostring(name_t)) - end - real_name = { - "string", - '"', - flattened_name - } - end - local cls = build.table({ - { - "__init", - super_scope(constructor, cls_super, { - "key_literal", - "__init" - }) - }, - { - "__base", - base_name - }, - { - "__name", - real_name - }, - parent_val and { - "__parent", - parent_cls_name - } or nil - }) - local class_index - if parent_val then - local class_lookup = build["if"]({ - cond = { - "exp", - { - "ref", - "val" - }, - "==", - "nil" - }, - ["then"] = { - build.assign_one(LocalName("parent"), build.chain({ - base = "rawget", - { - "call", - { - { - "ref", - "cls" - }, - { - "string", - '"', - "__parent" - } - } - } - })), - build["if"]({ - cond = LocalName("parent"), - ["then"] = { - build.chain({ - base = LocalName("parent"), - { - "index", - "name" - } - }) - } - }) - } - }) - insert(class_lookup, { - "else", - { - "val" - } - }) - class_index = build.fndef({ - args = { - { - "cls" - }, - { - "name" - } - }, - body = { - build.assign_one(LocalName("val"), build.chain({ - base = "rawget", - { - "call", - { - base_name, - { - "ref", - "name" - } - } - } - })), - class_lookup - } - }) - else - class_index = base_name - end - local cls_mt = build.table({ - { - "__index", - class_index - }, - { - "__call", - build.fndef({ - args = { - { - "cls" - }, - { - "..." - } - }, - body = { - build.assign_one(self_name, build.chain({ - base = "setmetatable", - { - "call", - { - "{}", - base_name - } - } - })), - build.chain({ - base = "cls.__init", - { - "call", - { - self_name, - "..." - } - } - }), - self_name - } - }) - } - }) - cls = build.chain({ - base = "setmetatable", - { - "call", - { - cls, - cls_mt - } - } - }) - local value = nil - do - local out_body = { - Run(function(self) - if name then - return self:put_name(name) - end - end), - { - "declare", - { - cls_name - } - }, - { - "declare_glob", - "*" - }, - parent_val and build.assign_one(parent_cls_name, parent_val) or NOOP, - build.assign_one(base_name, { - "table", - properties - }), - build.assign_one(base_name:chain("__index"), base_name), - parent_val and build.chain({ - base = "setmetatable", - { - "call", - { - base_name, - build.chain({ - base = parent_cls_name, - { - "dot", - "__base" - } - }) - } - } - }) or NOOP, - build.assign_one(cls_name, cls), - build.assign_one(base_name:chain("__class"), cls_name), - build.group((function() - if #statements > 0 then - return { - build.assign_one(LocalName("self"), cls_name), - build.group(statements) - } - end - end)()), - parent_val and build["if"]({ - cond = { - "exp", - parent_cls_name:chain("__inherited") - }, - ["then"] = { - parent_cls_name:chain("__inherited", { - "call", - { - parent_cls_name, - cls_name - } - }) - } - }) or NOOP, - build.group((function() - if name then - return { - build.assign_one(name, cls_name) - } - end - end)()), - (function() - if ret then - return ret(cls_name) - end - end)() - } - value = build.group({ - build.group((function() - if ntype(name) == "value" then - return { - build.declare({ - names = { - name - } - }) - } - end - end)()), - build["do"](out_body) - }) - end - return value - end - -end -package.preload['moonscript.transform.statements'] = function() - local types = require("moonscript.types") - local ntype, mtype, is_value, NOOP - ntype, mtype, is_value, NOOP = types.ntype, types.mtype, types.is_value, types.NOOP - local comprehension_has_value - comprehension_has_value = require("moonscript.transform.comprehension").comprehension_has_value - local Run - do - local _class_0 - local _base_0 = { - call = function(self, state) - return self.fn(state) - end } _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, fn) - self.fn = fn - self[1] = "run" + local _class_0 = setmetatable({ + __init = function(self) + return self:reset() end, __base = _base_0, - __name = "Run" + __name = "CodeCoverage" }, { __index = _base_0, __call = function(cls, ...) @@ -843,1361 +223,939 @@ package.preload['moonscript.transform.statements'] = function() end }) _base_0.__class = _class_0 - Run = _class_0 - end - local last_stm - last_stm = function(stms) - local last_exp_id = 0 - for i = #stms, 1, -1 do - local stm = stms[i] - if stm and mtype(stm) ~= Run then - if ntype(stm) == "group" then - return last_stm(stm[2]) - end - last_exp_id = i - break - end - end - return stms[last_exp_id], last_exp_id, stms - end - local transform_last_stm - transform_last_stm = function(stms, fn) - local _, last_idx, _stms = last_stm(stms) - if _stms ~= stms then - error("cannot transform last node in group") - end - return (function() - local _accum_0 = { } - local _len_0 = 1 - for i, stm in ipairs(stms) do - if i == last_idx then - _accum_0[_len_0] = { - "transform", - stm, - fn - } - else - _accum_0[_len_0] = stm - end - _len_0 = _len_0 + 1 - end - return _accum_0 - end)() - end - local chain_is_stub - chain_is_stub = function(chain) - local stub = chain[#chain] - return stub and ntype(stub) == "colon" - end - local implicitly_return - implicitly_return = function(scope) - local is_top = true - local fn - fn = function(stm) - local t = ntype(stm) - if t == "decorated" then - stm = scope.transform.statement(stm) - t = ntype(stm) - end - if types.cascading[t] then - is_top = false - return scope.transform.statement(stm, fn) - elseif types.manual_return[t] or not is_value(stm) then - if is_top and t == "return" and stm[2] == "" then - return NOOP - else - return stm - end - else - if t == "comprehension" and not comprehension_has_value(stm) then - return stm - else - return { - "return", - stm - } - end - end - end - return fn + CodeCoverage = _class_0 end return { - Run = Run, - last_stm = last_stm, - transform_last_stm = transform_last_stm, - chain_is_stub = chain_is_stub, - implicitly_return = implicitly_return + CodeCoverage = CodeCoverage } end -package.preload['moonscript.transform.destructure'] = function() - local ntype, mtype, build +package.preload['moonscript.cmd.lint'] = function() + local insert do - local _obj_0 = require("moonscript.types") - ntype, mtype, build = _obj_0.ntype, _obj_0.mtype, _obj_0.build + local _obj_0 = table + insert = _obj_0.insert end - local NameProxy - NameProxy = require("moonscript.transform.names").NameProxy - local insert - insert = table.insert - local unpack - unpack = require("moonscript.util").unpack - local user_error - user_error = require("moonscript.errors").user_error - local join - join = function(...) - do - local out = { } - local i = 1 - local _list_0 = { - ... - } - for _index_0 = 1, #_list_0 do - local tbl = _list_0[_index_0] - for _index_1 = 1, #tbl do - local v = tbl[_index_1] - out[i] = v - i = i + 1 - end - end - return out - end + local Set + do + local _obj_0 = require("moonscript.data") + Set = _obj_0.Set end - local has_destructure - has_destructure = function(names) - for _index_0 = 1, #names do - local n = names[_index_0] - if ntype(n) == "table" then - return true + local Block + do + local _obj_0 = require("moonscript.compile") + Block = _obj_0.Block + end + local default_whitelist = Set({ + '_G', + '_VERSION', + 'assert', + 'bit32', + 'collectgarbage', + 'coroutine', + 'debug', + 'dofile', + 'error', + 'getfenv', + 'getmetatable', + 'io', + 'ipairs', + 'load', + 'loadfile', + 'loadstring', + 'math', + 'module', + 'next', + 'os', + 'package', + 'pairs', + 'pcall', + 'print', + 'rawequal', + 'rawget', + 'rawlen', + 'rawset', + 'require', + 'select', + 'setfenv', + 'setmetatable', + 'string', + 'table', + 'tonumber', + 'tostring', + 'type', + 'unpack', + 'xpcall', + "nil", + "true", + "false" + }) + local LinterBlock + do + local _parent_0 = Block + local _base_0 = { + block = function(self, ...) + do + local _with_0 = _parent_0.block(self, ...) + _with_0.block = self.block + _with_0.value_compilers = self.value_compilers + return _with_0 + end + end + } + _base_0.__index = _base_0 + setmetatable(_base_0, _parent_0.__base) + local _class_0 = setmetatable({ + __init = function(self, whitelist_globals, ...) + if whitelist_globals == nil then + whitelist_globals = default_whitelist + end + _parent_0.__init(self, ...) + self.lint_errors = { } + local vc = self.value_compilers + self.value_compilers = setmetatable({ + ref = function(block, val) + local name = val[2] + if not (block:has_name(name) or whitelist_globals[name] or name:match("%.")) then + insert(self.lint_errors, { + "accessing global " .. tostring(name), + val[-1] + }) + end + return vc.ref(block, val) + end + }, { + __index = vc + }) + end, + __base = _base_0, + __name = "LinterBlock", + __parent = _parent_0 + }, { + __index = function(cls, name) + local val = rawget(_base_0, name) + if val == nil then + return _parent_0[name] + else + return val + end + end, + __call = function(cls, ...) + local _self_0 = setmetatable({}, _base_0) + cls.__init(_self_0, ...) + return _self_0 end + }) + _base_0.__class = _class_0 + if _parent_0.__inherited then + _parent_0.__inherited(_parent_0, _class_0) end - return false + LinterBlock = _class_0 end - local extract_assign_names - extract_assign_names = function(name, accum, prefix) - if accum == nil then - accum = { } + local format_lint + format_lint = function(errors, code, header) + if not (next(errors)) then + return end - if prefix == nil then - prefix = { } + local pos_to_line, get_line + do + local _obj_0 = require("moonscript.util") + pos_to_line, get_line = _obj_0.pos_to_line, _obj_0.get_line end - local i = 1 - local _list_0 = name[2] - for _index_0 = 1, #_list_0 do - local tuple = _list_0[_index_0] - local value, suffix - if #tuple == 1 then - local s = { - "index", - { - "number", - i - } - } - i = i + 1 - value, suffix = tuple[1], s - else - local key = tuple[1] - local s - if ntype(key) == "key_literal" then - local key_name = key[2] - if ntype(key_name) == "colon" then - s = key_name - else - s = { - "dot", - key_name - } - end + local formatted + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #errors do + local _des_0 = errors[_index_0] + local msg, pos + msg, pos = _des_0[1], _des_0[2] + if pos then + local line = pos_to_line(code, pos) + msg = "line " .. tostring(line) .. ": " .. tostring(msg) + local line_text = "> " .. get_line(code, line) + local sep_len = math.max(#msg, #line_text) + _accum_0[_len_0] = table.concat({ + msg, + ("="):rep(sep_len), + line_text + }, "\n") else - s = { - "index", - key - } + _accum_0[_len_0] = msg end - value, suffix = tuple[2], s - end - suffix = join(prefix, { - suffix - }) - local _exp_0 = ntype(value) - if "value" == _exp_0 or "ref" == _exp_0 or "chain" == _exp_0 or "self" == _exp_0 then - insert(accum, { - value, - suffix - }) - elseif "table" == _exp_0 then - extract_assign_names(value, accum, suffix) - else - user_error("Can't destructure value of type: " .. tostring(ntype(value))) + _len_0 = _len_0 + 1 end + formatted = _accum_0 end - return accum + if header then + table.insert(formatted, 1, header) + end + return table.concat(formatted, "\n\n") end - local build_assign - build_assign = function(scope, destruct_literal, receiver) - local extracted_names = extract_assign_names(destruct_literal) - local names = { } - local values = { } - local inner = { - "assign", - names, - values - } - local obj - if scope:is_local(receiver) or #extracted_names == 1 then - obj = receiver - else - do - obj = NameProxy("obj") - inner = build["do"]({ - build.assign_one(obj, receiver), - { - "assign", - names, - values - } - }) - obj = obj + local whitelist_for_file + do + local lint_config + whitelist_for_file = function(fname) + if not (lint_config) then + lint_config = { } + pcall(function() + lint_config = require("lint_config") + end) end - end - for _index_0 = 1, #extracted_names do - local tuple = extracted_names[_index_0] - insert(names, tuple[1]) - local chain - if obj then - chain = NameProxy.chain(obj, unpack(tuple[2])) - else - chain = "nil" + if not (lint_config.whitelist_globals) then + return default_whitelist + end + local final_list = { } + for pattern, list in pairs(lint_config.whitelist_globals) do + if fname:match(pattern) then + for _index_0 = 1, #list do + local item = list[_index_0] + insert(final_list, item) + end + end end - insert(values, chain) + return setmetatable(Set(final_list), { + __index = default_whitelist + }) end - return build.group({ - { - "declare", - names - }, - inner - }) end - local split_assign - split_assign = function(scope, assign) - local names, values = unpack(assign, 2) - local g = { } - local total_names = #names - local total_values = #values - local start = 1 - for i, n in ipairs(names) do - if ntype(n) == "table" then - if i > start then - local stop = i - 1 - insert(g, { - "assign", - (function() - local _accum_0 = { } - local _len_0 = 1 - for i = start, stop do - _accum_0[_len_0] = names[i] - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), - (function() - local _accum_0 = { } - local _len_0 = 1 - for i = start, stop do - _accum_0[_len_0] = values[i] - _len_0 = _len_0 + 1 - end - return _accum_0 - end)() - }) - end - insert(g, build_assign(scope, n, values[i])) - start = i + 1 - end + local lint_code + lint_code = function(code, name, whitelist_globals) + if name == nil then + name = "string input" end - if total_names >= start or total_values >= start then - local name_slice - if total_names < start then - name_slice = { - "_" - } - else + local parse = require("moonscript.parse") + local tree, err = parse.string(code) + if not (tree) then + return nil, err + end + local scope = LinterBlock(whitelist_globals) + scope:stms(tree) + return format_lint(scope.lint_errors, code, name) + end + local lint_file + lint_file = function(fname) + local f, err = io.open(fname) + if not (f) then + return nil, err + end + return lint_code(f:read("*a"), fname, whitelist_for_file(fname)) + end + return { + lint_code = lint_code, + lint_file = lint_file + } + +end +package.preload['moonscript.compile.statement'] = function() + local util = require("moonscript.util") + local data = require("moonscript.data") + local reversed, unpack + reversed, unpack = util.reversed, util.unpack + local ntype + do + local _obj_0 = require("moonscript.types") + ntype = _obj_0.ntype + end + local concat, insert + do + local _obj_0 = table + concat, insert = _obj_0.concat, _obj_0.insert + end + return { + raw = function(self, node) + return self:add(node[2]) + end, + lines = function(self, node) + local _list_0 = node[2] + for _index_0 = 1, #_list_0 do + local line = _list_0[_index_0] + self:add(line) + end + end, + declare = function(self, node) + local names = node[2] + local undeclared = self:declare(names) + if #undeclared > 0 then do + local _with_0 = self:line("local ") + _with_0:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #undeclared do + local name = undeclared[_index_0] + _accum_0[_len_0] = self:name(name) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ") + return _with_0 + end + end + end, + declare_with_shadows = function(self, node) + local names = node[2] + self:declare(names) + do + local _with_0 = self:line("local ") + _with_0:append_list((function() local _accum_0 = { } local _len_0 = 1 - for i = start, total_names do - _accum_0[_len_0] = names[i] + for _index_0 = 1, #names do + local name = names[_index_0] + _accum_0[_len_0] = self:name(name) _len_0 = _len_0 + 1 end - name_slice = _accum_0 + return _accum_0 + end)(), ", ") + return _with_0 + end + end, + assign = function(self, node) + local _, names, values = unpack(node) + local undeclared = self:declare(names) + local declare = "local " .. concat(undeclared, ", ") + local has_fndef = false + local i = 1 + while i <= #values do + if ntype(values[i]) == "fndef" then + has_fndef = true end + i = i + 1 end - local value_slice - if total_values < start then - value_slice = { - "nil" - } - else - do + do + local _with_0 = self:line() + if #undeclared == #names and not has_fndef then + _with_0:append(declare) + else + if #undeclared > 0 then + self:add(declare) + end + _with_0:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #names do + local name = names[_index_0] + _accum_0[_len_0] = self:value(name) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ") + end + _with_0:append(" = ") + _with_0:append_list((function() local _accum_0 = { } local _len_0 = 1 - for i = start, total_values do - _accum_0[_len_0] = values[i] + for _index_0 = 1, #values do + local v = values[_index_0] + _accum_0[_len_0] = self:value(v) _len_0 = _len_0 + 1 end - value_slice = _accum_0 + return _accum_0 + end)(), ", ") + return _with_0 + end + end, + ["return"] = function(self, node) + return self:line("return ", (function() + if node[2] ~= "" then + return self:value(node[2]) end + end)()) + end, + ["break"] = function(self, node) + return "break" + end, + ["if"] = function(self, node) + local cond, block = node[2], node[3] + local root + do + local _with_0 = self:block(self:line("if ", self:value(cond), " then")) + _with_0:stms(block) + root = _with_0 end - insert(g, { - "assign", - name_slice, - value_slice - }) - end - return build.group(g) - end - return { - has_destructure = has_destructure, - split_assign = split_assign, - build_assign = build_assign, - extract_assign_names = extract_assign_names - } - -end -package.preload['moonscript.transform.names'] = function() - local build - build = require("moonscript.types").build - local unpack - unpack = require("moonscript.util").unpack - local LocalName - do - local _class_0 - local _base_0 = { - get_name = function(self) - return self.name + local current = root + local add_clause + add_clause = function(clause) + local type = clause[1] + local i = 2 + local next + if type == "else" then + next = self:block("else") + else + i = i + 1 + next = self:block(self:line("elseif ", self:value(clause[2]), " then")) + end + next:stms(clause[i]) + current.next = next + current = next end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, name) - self.name = name - self[1] = "temp_name" - end, - __base = _base_0, - __name = "LocalName" - }, { - __index = _base_0, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + for _index_0 = 4, #node do + cond = node[_index_0] + add_clause(cond) end - }) - _base_0.__class = _class_0 - LocalName = _class_0 - end - local NameProxy - do - local _class_0 - local _base_0 = { - get_name = function(self, scope, dont_put) - if dont_put == nil then - dont_put = true - end - if not self.name then - self.name = scope:free_name(self.prefix, dont_put) - end - return self.name - end, - chain = function(self, ...) - local items = { - base = self, - ... - } - for k, v in ipairs(items) do - if type(v) == "string" then - items[k] = { - "dot", - v - } - else - items[k] = v - end - end - return build.chain(items) - end, - index = function(self, key) - if type(key) == "string" then - key = { - "ref", - key - } - end - return build.chain({ - base = self, - { - "index", - key - } + return root + end, + ["repeat"] = function(self, node) + local cond, block = unpack(node, 2) + do + local _with_0 = self:block("repeat", self:line("until ", self:value(cond))) + _with_0:stms(block) + return _with_0 + end + end, + ["while"] = function(self, node) + local _, cond, block = unpack(node) + do + local _with_0 = self:block(self:line("while ", self:value(cond), " do")) + _with_0:stms(block) + return _with_0 + end + end, + ["for"] = function(self, node) + local _, name, bounds, block = unpack(node) + local loop = self:line("for ", self:name(name), " = ", self:value({ + "explist", + unpack(bounds) + }), " do") + do + local _with_0 = self:block(loop) + _with_0:declare({ + name }) - end, - __tostring = function(self) - if self.name then - return ("name<%s>"):format(self.name) - else - return ("name"):format(self.prefix) + _with_0:stms(block) + return _with_0 + end + end, + foreach = function(self, node) + local _, names, exps, block = unpack(node) + local loop + do + local _with_0 = self:line() + _with_0:append("for ") + loop = _with_0 + end + do + local _with_0 = self:block(loop) + loop:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #names do + local name = names[_index_0] + _accum_0[_len_0] = _with_0:name(name, false) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ") + loop:append(" in ") + loop:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #exps do + local exp = exps[_index_0] + _accum_0[_len_0] = self:value(exp) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ",") + loop:append(" do") + _with_0:declare(names) + _with_0:stms(block) + return _with_0 + end + end, + export = function(self, node) + local _, names = unpack(node) + if type(names) == "string" then + if names == "*" then + self.export_all = true + elseif names == "^" then + self.export_proper = true end + else + self:declare(names) end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, prefix) - self.prefix = prefix - self[1] = "temp_name" - end, - __base = _base_0, - __name = "NameProxy" - }, { - __index = _base_0, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + return nil + end, + run = function(self, code) + code:call(self) + return nil + end, + group = function(self, node) + return self:stms(node[2]) + end, + ["do"] = function(self, node) + do + local _with_0 = self:block() + _with_0:stms(node[2]) + return _with_0 end - }) - _base_0.__class = _class_0 - NameProxy = _class_0 - end - local is_name_proxy - is_name_proxy = function(v) - if not (type(v) == "table") then - return false - end - local _exp_0 = v.__class - if LocalName == _exp_0 or NameProxy == _exp_0 then - return true - end - end - return { - NameProxy = NameProxy, - LocalName = LocalName, - is_name_proxy = is_name_proxy + end, + noop = function(self) end } end -package.preload['moonscript.transform.statement'] = function() - local Transformer - Transformer = require("moonscript.transform.transformer").Transformer - local NameProxy, LocalName, is_name_proxy +package.preload['moonscript.compile.value'] = function() + local util = require("moonscript.util") + local data = require("moonscript.data") + local ntype do - local _obj_0 = require("moonscript.transform.names") - NameProxy, LocalName, is_name_proxy = _obj_0.NameProxy, _obj_0.LocalName, _obj_0.is_name_proxy + local _obj_0 = require("moonscript.types") + ntype = _obj_0.ntype end - local Run, transform_last_stm, implicitly_return, last_stm + local user_error do - local _obj_0 = require("moonscript.transform.statements") - Run, transform_last_stm, implicitly_return, last_stm = _obj_0.Run, _obj_0.transform_last_stm, _obj_0.implicitly_return, _obj_0.last_stm + local _obj_0 = require("moonscript.errors") + user_error = _obj_0.user_error + end + local concat, insert + do + local _obj_0 = table + concat, insert = _obj_0.concat, _obj_0.insert end - local types = require("moonscript.types") - local build, ntype, is_value, smart_node, value_is_singular, is_slice, NOOP - build, ntype, is_value, smart_node, value_is_singular, is_slice, NOOP = types.build, types.ntype, types.is_value, types.smart_node, types.value_is_singular, types.is_slice, types.NOOP - local insert - insert = table.insert - local destructure = require("moonscript.transform.destructure") - local construct_comprehension - construct_comprehension = require("moonscript.transform.comprehension").construct_comprehension local unpack - unpack = require("moonscript.util").unpack - local with_continue_listener - with_continue_listener = function(body) - local continue_name = nil - return { - Run(function(self) - return self:listen("continue", function() - if not (continue_name) then - continue_name = NameProxy("continue") - self:put_name(continue_name) - end - return continue_name - end) - end), - build.group(body), - Run(function(self) - if not (continue_name) then - return + unpack = util.unpack + local table_delim = "," + local string_chars = { + ["\r"] = "\\r", + ["\n"] = "\\n" + } + return { + exp = function(self, node) + local _comp + _comp = function(i, value) + if i % 2 == 1 and value == "!=" then + value = "~=" end - local last = last_stm(body) - local enclose_lines = types.terminating[last and ntype(last)] - self:put_name(continue_name, nil) - return self:splice(function(lines) - if enclose_lines then - lines = { - "do", - { - lines - } - } + return self:value(value) + end + do + local _with_0 = self:line() + _with_0:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for i, v in ipairs(node) do + if i > 1 then + _accum_0[_len_0] = _comp(i, v) + _len_0 = _len_0 + 1 + end end - return { - { - "assign", - { - continue_name - }, - { - "false" - } - }, - { - "repeat", - "true", - { - lines, - { - "assign", - { - continue_name - }, - { - "true" - } - } - } - }, - { - "if", - { - "not", - continue_name - }, - { - { - "break" - } - } - } - } - end) - end) - } - end - local extract_declarations - extract_declarations = function(self, body, start, out) - if body == nil then - body = self.current_stms - end - if start == nil then - start = self.current_stm_i + 1 - end - if out == nil then - out = { } - end - for i = start, #body do - local _continue_0 = false - repeat - local stm = body[i] - if stm == nil then - _continue_0 = true - break + return _accum_0 + end)(), " ") + return _with_0 + end + end, + explist = function(self, node) + do + local _with_0 = self:line() + _with_0:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 2, #node do + local v = node[_index_0] + _accum_0[_len_0] = self:value(v) + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), ", ") + return _with_0 + end + end, + parens = function(self, node) + return self:line("(", self:value(node[2]), ")") + end, + string = function(self, node) + local _, delim, inner = unpack(node) + local end_delim = delim:gsub("%[", "]") + if delim == "'" or delim == '"' then + inner = inner:gsub("[\r\n]", string_chars) + end + return delim .. inner .. end_delim + end, + chain = function(self, node) + local callee = node[2] + local callee_type = ntype(callee) + if callee == -1 then + callee = self:get("scope_var") + if not callee then + user_error("Short-dot syntax must be called within a with block") end - stm = self.transform.statement(stm) - body[i] = stm - local _exp_0 = stm[1] - if "assign" == _exp_0 or "declare" == _exp_0 then - local _list_0 = stm[2] - for _index_0 = 1, #_list_0 do - local name = _list_0[_index_0] - if ntype(name) == "ref" then - insert(out, name) - elseif type(name) == "string" then - insert(out, name) - end + end + if callee_type == "ref" and callee[2] == "super" or callee == "super" then + do + local sup = self:get("super") + if sup then + return self:value(sup(self, node)) end - elseif "group" == _exp_0 then - extract_declarations(self, stm[2], 1, out) end - _continue_0 = true - until true - if not _continue_0 then - break end - end - return out - end - local expand_elseif_assign - expand_elseif_assign = function(ifstm) - for i = 4, #ifstm do - local case = ifstm[i] - if ntype(case) == "elseif" and ntype(case[2]) == "assign" then - local split = { - unpack(ifstm, 1, i - 1) - } - insert(split, { - "else", - { - { - "if", - case[2], - case[3], - unpack(ifstm, i + 1) - } - } - }) - return split + local chain_item + chain_item = function(node) + local t, arg = unpack(node) + if t == "call" then + return "(", self:values(arg), ")" + elseif t == "index" then + return "[", self:value(arg), "]" + elseif t == "dot" then + return ".", tostring(arg) + elseif t == "colon" then + return ":", arg, chain_item(node[3]) + elseif t == "colon_stub" then + return user_error("Uncalled colon stub") + else + return error("Unknown chain action: " .. tostring(t)) + end end - end - return ifstm - end - return Transformer({ - transform = function(self, tuple) - local _, node, fn - _, node, fn = tuple[1], tuple[2], tuple[3] - return fn(node) - end, - root_stms = function(self, body) - return transform_last_stm(body, implicitly_return(self)) - end, - ["return"] = function(self, node) - local ret_val = node[2] - local ret_val_type = ntype(ret_val) - if ret_val_type == "explist" and #ret_val == 2 then - ret_val = ret_val[2] - ret_val_type = ntype(ret_val) + if (callee_type == "self" or callee_type == "self_class") and node[3] and ntype(node[3]) == "call" then + callee[1] = callee_type .. "_colon" end - if types.cascading[ret_val_type] then - return implicitly_return(self)(ret_val) + local callee_value = self:value(callee) + if ntype(callee) == "exp" then + callee_value = self:line("(", callee_value, ")") end - if ret_val_type == "chain" or ret_val_type == "comprehension" or ret_val_type == "tblcomprehension" then - local Value = require("moonscript.transform.value") - ret_val = Value:transform_once(self, ret_val) - if ntype(ret_val) == "block_exp" then - return build.group(transform_last_stm(ret_val[2], function(stm) - return { - "return", - stm - } - end)) + local actions + do + local _with_0 = self:line() + for _index_0 = 3, #node do + local action = node[_index_0] + _with_0:append(chain_item(action)) end + actions = _with_0 end - node[2] = ret_val - return node + return self:line(callee_value, actions) end, - declare_glob = function(self, node) - local names = extract_declarations(self) - if node[2] == "^" then - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #names do - local _continue_0 = false - repeat - local name = names[_index_0] - local str_name - if ntype(name) == "ref" then - str_name = name[2] - else - str_name = name - end - if not (str_name:match("^%u")) then - _continue_0 = true - break - end - local _value_0 = name - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - _continue_0 = true - until true - if not _continue_0 then - break + fndef = function(self, node) + local _, args, whitelist, arrow, block = unpack(node) + local default_args = { } + local self_args = { } + local arg_names + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #args do + local arg = args[_index_0] + local name, default_value = unpack(arg) + if type(name) == "string" then + name = name + else + if name[1] == "self" or name[1] == "self_class" then + insert(self_args, name) end + name = name[2] end - names = _accum_0 + if default_value then + insert(default_args, arg) + end + local _value_0 = name + _accum_0[_len_0] = _value_0 + _len_0 = _len_0 + 1 end + arg_names = _accum_0 end - return { - "declare", - names - } - end, - assign = function(self, node) - local names, values = unpack(node, 2) - local num_values = #values - local num_names = #values - if num_names == 1 and num_values == 1 then - local first_value = values[1] - local first_name = names[1] - local first_type = ntype(first_value) - if first_type == "chain" then - local Value = require("moonscript.transform.value") - first_value = Value:transform_once(self, first_value) - first_type = ntype(first_value) + if arrow == "fat" then + insert(arg_names, 1, "self") + end + do + local _with_0 = self:block() + if #whitelist > 0 then + _with_0:whitelist_names(whitelist) end - local _exp_0 = ntype(first_value) - if "block_exp" == _exp_0 then - local block_body = first_value[2] - local idx = #block_body - block_body[idx] = build.assign_one(first_name, block_body[idx]) - return build.group({ + for _index_0 = 1, #arg_names do + local name = arg_names[_index_0] + _with_0:put_name(name) + end + for _index_0 = 1, #default_args do + local default = default_args[_index_0] + local name, value = unpack(default) + if type(name) == "table" then + name = name[2] + end + _with_0:stm({ + 'if', { - "declare", + 'exp', { - first_name - } + "ref", + name + }, + '==', + 'nil' }, { - "do", - block_body - } - }) - elseif "comprehension" == _exp_0 or "tblcomprehension" == _exp_0 or "foreach" == _exp_0 or "for" == _exp_0 or "while" == _exp_0 then - local Value = require("moonscript.transform.value") - return build.assign_one(first_name, Value:transform_once(self, first_value)) - else - values[1] = first_value - end - end - local transformed - if num_values == 1 then - local value = values[1] - local t = ntype(value) - if t == "decorated" then - value = self.transform.statement(value) - t = ntype(value) - end - if types.cascading[t] then - local ret - ret = function(stm) - if is_value(stm) then - return { - "assign", - names, + { + 'assign', { - stm + name + }, + { + value } } - else - return stm - end - end - transformed = build.group({ - { - "declare", - names - }, - self.transform.statement(value, ret, node) - }) - end - end - node = transformed or node - if destructure.has_destructure(names) then - return destructure.split_assign(self, node) - end - return node - end, - continue = function(self, node) - local continue_name = self:send("continue") - if not (continue_name) then - error("continue must be inside of a loop") - end - return build.group({ - build.assign_one(continue_name, "true"), - { - "break" - } - }) - end, - export = function(self, node) - if #node > 2 then - if node[2] == "class" then - local cls = smart_node(node[3]) - return build.group({ - { - "export", - { - cls.name - } - }, - cls - }) - else - return build.group({ - { - "export", - node[2] - }, - build.assign({ - names = node[2], - values = node[3] - }) + } }) end - else - return nil - end - end, - update = function(self, node) - local name, op, exp = unpack(node, 2) - local op_final = op:match("^(.+)=$") - if not op_final then - error("Unknown op: " .. op) - end - local lifted - if ntype(name) == "chain" then - lifted = { } - local new_chain + local self_arg_values do local _accum_0 = { } local _len_0 = 1 - for _index_0 = 3, #name do - local part = name[_index_0] - if ntype(part) == "index" then - local proxy = NameProxy("update") - table.insert(lifted, { - proxy, - part[2] - }) - _accum_0[_len_0] = { - "index", - proxy - } - else - _accum_0[_len_0] = part - end + for _index_0 = 1, #self_args do + local arg = self_args[_index_0] + _accum_0[_len_0] = arg[2] _len_0 = _len_0 + 1 end - new_chain = _accum_0 - end - if next(lifted) then - name = { - name[1], - name[2], - unpack(new_chain) - } + self_arg_values = _accum_0 end - end - if not (value_is_singular(exp)) then - exp = { - "parens", - exp - } - end - local out = build.assign_one(name, { - "exp", - name, - op_final, - exp - }) - if lifted and next(lifted) then - local names - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #lifted do - local l = lifted[_index_0] - _accum_0[_len_0] = l[1] - _len_0 = _len_0 + 1 - end - names = _accum_0 + if #self_args > 0 then + _with_0:stm({ + "assign", + self_args, + self_arg_values + }) end - local values - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #lifted do - local l = lifted[_index_0] - _accum_0[_len_0] = l[2] - _len_0 = _len_0 + 1 + _with_0:stms(block) + if #args > #arg_names then + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #args do + local arg = args[_index_0] + _accum_0[_len_0] = arg[1] + _len_0 = _len_0 + 1 + end + arg_names = _accum_0 end - values = _accum_0 end - out = build.group({ - { - "assign", - names, - values - }, - out - }) + _with_0.header = "function(" .. concat(arg_names, ", ") .. ")" + return _with_0 end - return out end, - import = function(self, node) - local names, source = unpack(node, 2) - local table_values + table = function(self, node) + local _, items = unpack(node) do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #names do - local name = names[_index_0] - local dest_name - if ntype(name) == "colon" then - dest_name = name[2] + local _with_0 = self:block("{", "}") + local format_line + format_line = function(tuple) + if #tuple == 2 then + local key, value = unpack(tuple) + if ntype(key) == "key_literal" and data.lua_keywords[key[2]] then + key = { + "string", + '"', + key[2] + } + end + local assign + if ntype(key) == "key_literal" then + assign = key[2] + else + assign = self:line("[", _with_0:value(key), "]") + end + _with_0:set("current_block", key) + local out = self:line(assign, " = ", _with_0:value(value)) + _with_0:set("current_block", nil) + return out else - dest_name = name + return self:line(_with_0:value(tuple[1])) end - local _value_0 = { - { - "key_literal", - name - }, - dest_name - } - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 end - table_values = _accum_0 + if items then + local count = #items + for i, tuple in ipairs(items) do + local line = format_line(tuple) + if not (count == i) then + line:append(table_delim) + end + _with_0:add(line) + end + end + return _with_0 end - local dest = { - "table", - table_values - } - return { - "assign", - { - dest - }, - { - source - }, - [-1] = node[-1] - } end, - comprehension = function(self, node, action) - local exp, clauses = unpack(node, 2) - action = action or function(exp) - return { - exp - } - end - return construct_comprehension(action(exp), clauses) + minus = function(self, node) + return self:line("-", self:value(node[2])) end, - ["do"] = function(self, node, ret) - if ret then - node[2] = transform_last_stm(node[2], ret) - end - return node + temp_name = function(self, node, ...) + return node:get_name(self, ...) end, - decorated = function(self, node) - local stm, dec = unpack(node, 2) - local wrapped - local _exp_0 = dec[1] - if "if" == _exp_0 then - local cond, fail = unpack(dec, 2) - if fail then - fail = { - "else", - { - fail - } - } - end - wrapped = { - "if", - cond, - { - stm - }, - fail - } - elseif "unless" == _exp_0 then - wrapped = { - "unless", - dec[2], - { - stm - } - } - elseif "comprehension" == _exp_0 then - wrapped = { - "comprehension", - stm, - dec[2] - } - else - wrapped = error("Unknown decorator " .. dec[1]) - end - if ntype(stm) == "assign" then - wrapped = build.group({ - build.declare({ - names = (function() - local _accum_0 = { } - local _len_0 = 1 - local _list_0 = stm[2] - for _index_0 = 1, #_list_0 do - local name = _list_0[_index_0] - if ntype(name) == "ref" then - _accum_0[_len_0] = name - _len_0 = _len_0 + 1 - end - end - return _accum_0 - end)() - }), - wrapped - }) - end - return wrapped + number = function(self, node) + return node[2] end, - unless = function(self, node) - local clause = node[2] - if ntype(clause) == "assign" then - if destructure.has_destructure(clause[2]) then - error("destructure not allowed in unless assignment") + length = function(self, node) + return self:line("#", self:value(node[2])) + end, + ["not"] = function(self, node) + return self:line("not ", self:value(node[2])) + end, + self = function(self, node) + return "self." .. self:name(node[2]) + end, + self_class = function(self, node) + return "self.__class." .. self:name(node[2]) + end, + self_colon = function(self, node) + return "self:" .. self:name(node[2]) + end, + self_class_colon = function(self, node) + return "self.__class:" .. self:name(node[2]) + end, + ref = function(self, value) + do + local sup = value[2] == "super" and self:get("super") + if sup then + return self:value(sup(self)) end - return build["do"]({ - clause, - { - "if", - { - "not", - clause[2][1] - }, - unpack(node, 3) - } - }) - else - return { - "if", - { - "not", - { - "parens", - clause - } - }, - unpack(node, 3) - } end + return tostring(value[2]) end, - ["if"] = function(self, node, ret) - if ntype(node[2]) == "assign" then - local assign, body = unpack(node, 2) - if destructure.has_destructure(assign[2]) then - local name = NameProxy("des") - body = { - destructure.build_assign(self, assign[2][1], name), - build.group(node[3]) - } - return build["do"]({ - build.assign_one(name, assign[3][1]), - { - "if", - name, - body, - unpack(node, 4) - } - }) - else - local name = assign[2][1] - return build["do"]({ - assign, - { - "if", - name, - unpack(node, 3) - } - }) - end + raw_value = function(self, value) + if value == "..." then + self:send("varargs") end - node = expand_elseif_assign(node) - if ret then - smart_node(node) - node['then'] = transform_last_stm(node['then'], ret) - for i = 4, #node do - local case = node[i] - local body_idx = #node[i] - case[body_idx] = transform_last_stm(case[body_idx], ret) + return tostring(value) + end + } + +end +package.preload['moonscript.compile'] = function() + local util = require("moonscript.util") + local dump = require("moonscript.dump") + local transform = require("moonscript.transform") + local NameProxy, LocalName + do + local _obj_0 = require("moonscript.transform.names") + NameProxy, LocalName = _obj_0.NameProxy, _obj_0.LocalName + end + local Set + do + local _obj_0 = require("moonscript.data") + Set = _obj_0.Set + end + local ntype, has_value + do + local _obj_0 = require("moonscript.types") + ntype, has_value = _obj_0.ntype, _obj_0.has_value + end + local statement_compilers = require("moonscript.compile.statement") + local value_compilers = require("moonscript.compile.value") + local concat, insert + do + local _obj_0 = table + concat, insert = _obj_0.concat, _obj_0.insert + end + local pos_to_line, get_closest_line, trim, unpack + pos_to_line, get_closest_line, trim, unpack = util.pos_to_line, util.get_closest_line, util.trim, util.unpack + local mtype = util.moon.type + local indent_char = " " + local Line, DelayedLine, Lines, Block, RootBlock + do + local _base_0 = { + mark_pos = function(self, pos, line) + if line == nil then + line = #self end - end - return node - end, - with = function(self, node, ret) - local exp, block = unpack(node, 2) - local copy_scope = true - local scope_name, named_assign - do - local last = last_stm(block) - if last then - if types.terminating[ntype(last)] then - ret = false - end + if not (self.posmap[line]) then + self.posmap[line] = pos end - end - if ntype(exp) == "assign" then - local names, values = unpack(exp, 2) - local first_name = names[1] - if ntype(first_name) == "ref" then - scope_name = first_name - named_assign = exp - exp = values[1] - copy_scope = false + end, + add = function(self, item) + local _exp_0 = mtype(item) + if Line == _exp_0 then + item:render(self) + elseif Block == _exp_0 then + item:render(self) else - scope_name = NameProxy("with") - exp = values[1] - values[1] = scope_name - named_assign = { - "assign", - names, - values - } + self[#self + 1] = item end - elseif self:is_local(exp) then - scope_name = exp - copy_scope = false - end - scope_name = scope_name or NameProxy("with") - local out = build["do"]({ - copy_scope and build.assign_one(scope_name, exp) or NOOP, - named_assign or NOOP, - Run(function(self) - return self:set("scope_var", scope_name) - end), - unpack(block) - }) - if ret then - table.insert(out[2], ret(scope_name)) - end - return out - end, - foreach = function(self, node, _) - smart_node(node) - local source = unpack(node.iter) - local destructures = { } - do - local _accum_0 = { } - local _len_0 = 1 - for i, name in ipairs(node.names) do - if ntype(name) == "table" then - do - local proxy = NameProxy("des") - insert(destructures, destructure.build_assign(self, name, proxy)) - _accum_0[_len_0] = proxy + return self + end, + flatten_posmap = function(self, line_no, out) + if line_no == nil then + line_no = 0 + end + if out == nil then + out = { } + end + local posmap = self.posmap + for i, l in ipairs(self) do + local _exp_0 = mtype(l) + if "string" == _exp_0 or DelayedLine == _exp_0 then + line_no = line_no + 1 + out[line_no] = posmap[i] + for _ in l:gmatch("\n") do + line_no = line_no + 1 end + out[line_no] = posmap[i] + elseif Lines == _exp_0 then + local _ + _, line_no = l:flatten_posmap(line_no, out) else - _accum_0[_len_0] = name + error("Unknown item in Lines: " .. tostring(l)) end - _len_0 = _len_0 + 1 end - node.names = _accum_0 - end - if next(destructures) then - insert(destructures, build.group(node.body)) - node.body = destructures - end - if ntype(source) == "unpack" then - local list = source[2] - local index_name = NameProxy("index") - local list_name = self:is_local(list) and list or NameProxy("list") - local slice_var = nil - local bounds - if is_slice(list) then - local slice = list[#list] - table.remove(list) - table.remove(slice, 1) - if self:is_local(list) then - list_name = list - end - if slice[2] and slice[2] ~= "" then - local max_tmp_name = NameProxy("max") - slice_var = build.assign_one(max_tmp_name, slice[2]) - slice[2] = { - "exp", - max_tmp_name, - "<", - 0, - "and", - { - "length", - list_name - }, - "+", - max_tmp_name, - "or", - max_tmp_name - } - else - slice[2] = { - "length", - list_name - } - end - bounds = slice - else - bounds = { - 1, - { - "length", - list_name - } - } + return out, line_no + end, + flatten = function(self, indent, buffer) + if indent == nil then + indent = nil end - local names - do - local _accum_0 = { } - local _len_0 = 1 - local _list_0 = node.names - for _index_0 = 1, #_list_0 do - local n = _list_0[_index_0] - _accum_0[_len_0] = is_name_proxy(n) and n or LocalName(n) or n - _len_0 = _len_0 + 1 - end - names = _accum_0 + if buffer == nil then + buffer = { } end - return build.group({ - list_name ~= list and build.assign_one(list_name, list) or NOOP, - slice_var or NOOP, - build["for"]({ - name = index_name, - bounds = bounds, - body = { - { - "assign", - names, - { - NameProxy.index(list_name, index_name) - } - }, - build.group(node.body) - } - }) - }) - end - node.body = with_continue_listener(node.body) - end, - ["while"] = function(self, node) - smart_node(node) - node.body = with_continue_listener(node.body) - end, - ["for"] = function(self, node) - smart_node(node) - node.body = with_continue_listener(node.body) - end, - switch = function(self, node, ret) - local exp, conds = unpack(node, 2) - local exp_name = NameProxy("exp") - local convert_cond - convert_cond = function(cond) - local t, case_exps, body = unpack(cond) - local out = { } - insert(out, t == "case" and "elseif" or "else") - if t ~= "else" then - local cond_exp = { } - for i, case in ipairs(case_exps) do - if i == 1 then - insert(cond_exp, "exp") - else - insert(cond_exp, "or") + for i = 1, #self do + local l = self[i] + local t = mtype(l) + if t == DelayedLine then + l = l:render() + t = "string" + end + local _exp_0 = t + if "string" == _exp_0 then + if indent then + insert(buffer, indent) end - if not (value_is_singular(case)) then - case = { - "parens", - case - } + insert(buffer, l) + if "string" == type(self[i + 1]) then + local lc = l:sub(-1) + if (lc == ")" or lc == "]") and self[i + 1]:sub(1, 1) == "(" then + insert(buffer, ";") + end end - insert(cond_exp, { - "exp", - case, - "==", - exp_name - }) + insert(buffer, "\n") + local last = l + elseif Lines == _exp_0 then + l:flatten(indent and indent .. indent_char or indent_char, buffer) + else + error("Unknown item in Lines: " .. tostring(l)) end - insert(out, cond_exp) - else - body = case_exps - end - if ret then - body = transform_last_stm(body, ret) - end - insert(out, body) - return out - end - local first = true - local if_stm = { - "if" - } - for _index_0 = 1, #conds do - local cond = conds[_index_0] - local if_cond = convert_cond(cond) - if first then - first = false - insert(if_stm, if_cond[2]) - insert(if_stm, if_cond[3]) - else - insert(if_stm, if_cond) - end - end - return build.group({ - build.assign_one(exp_name, exp), - if_stm - }) - end, - class = require("moonscript.transform.class") - }) - -end -package.preload['moonscript.transform.transformer'] = function() - local ntype - ntype = require("moonscript.types").ntype - local Transformer - do - local _class_0 - local _base_0 = { - transform_once = function(self, scope, node, ...) - if self.seen_nodes[node] then - return node - end - self.seen_nodes[node] = true - local transformer = self.transformers[ntype(node)] - if transformer then - return transformer(scope, node, ...) or node - else - return node end + return buffer end, - transform = function(self, scope, node, ...) - if self.seen_nodes[node] then - return node - end - self.seen_nodes[node] = true - while true do - local transformer = self.transformers[ntype(node)] - local res - if transformer then - res = transformer(scope, node, ...) or node + __tostring = function(self) + local strip + strip = function(t) + if "table" == type(t) then + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #t do + local v = t[_index_0] + _accum_0[_len_0] = strip(v) + _len_0 = _len_0 + 1 + end + return _accum_0 else - res = node - end - if res == node then - return node + return t end - node = res - end - return node - end, - bind = function(self, scope) - return function(...) - return self:transform(scope, ...) end - end, - __call = function(self, ...) - return self:transform(...) - end, - can_transform = function(self, node) - return self.transformers[ntype(node)] ~= nil + return "Lines<" .. tostring(util.dump(strip(self)):sub(1, -2)) .. ">" end } _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, transformers) - self.transformers = transformers - self.seen_nodes = setmetatable({ }, { - __mode = "k" - }) + local _class_0 = setmetatable({ + __init = function(self) + self.posmap = { } end, __base = _base_0, - __name = "Transformer" + __name = "Lines" }, { __index = _base_0, __call = function(cls, ...) @@ -2207,105 +1165,76 @@ package.preload['moonscript.transform.transformer'] = function() end }) _base_0.__class = _class_0 - Transformer = _class_0 - end - return { - Transformer = Transformer - } - -end -package.preload['moonscript.transform.accumulator'] = function() - local types = require("moonscript.types") - local build, ntype, NOOP - build, ntype, NOOP = types.build, types.ntype, types.NOOP - local NameProxy - NameProxy = require("moonscript.transform.names").NameProxy - local insert - insert = table.insert - local is_singular - is_singular = function(body) - if #body ~= 1 then - return false - end - if "group" == ntype(body) then - return is_singular(body[2]) - else - return body[1] - end + Lines = _class_0 end - local transform_last_stm - transform_last_stm = require("moonscript.transform.statements").transform_last_stm - local Accumulator do - local _class_0 local _base_0 = { - body_idx = { - ["for"] = 4, - ["while"] = 3, - foreach = 4 - }, - convert = function(self, node) - local index = self.body_idx[ntype(node)] - node[index] = self:mutate_body(node[index]) - return self:wrap(node) + pos = nil, + append_list = function(self, items, delim) + for i = 1, #items do + self:append(items[i]) + if i < #items then + insert(self, delim) + end + end + return nil end, - wrap = function(self, node, group_type) - if group_type == nil then - group_type = "block_exp" + append = function(self, first, ...) + if Line == mtype(first) then + if not (self.pos) then + self.pos = first.pos + end + for _index_0 = 1, #first do + local value = first[_index_0] + self:append(value) + end + else + insert(self, first) + end + if ... then + return self:append(...) end - return build[group_type]({ - build.assign_one(self.accum_name, build.table()), - build.assign_one(self.len_name, 1), - node, - group_type == "block_exp" and self.accum_name or NOOP - }) end, - mutate_body = function(self, body) - local single_stm = is_singular(body) - local val - if single_stm and types.is_value(single_stm) then - body = { } - val = single_stm - else - body = transform_last_stm(body, function(n) - if types.is_value(n) then - return build.assign_one(self.value_name, n) - else - return build.group({ - { - "declare", - { - self.value_name - } - }, - n - }) + render = function(self, buffer) + local current = { } + local add_current + add_current = function() + buffer:add(concat(current)) + return buffer:mark_pos(self.pos) + end + for _index_0 = 1, #self do + local chunk = self[_index_0] + local _exp_0 = mtype(chunk) + if Block == _exp_0 then + local _list_0 = chunk:render(Lines()) + for _index_1 = 1, #_list_0 do + local block_chunk = _list_0[_index_1] + if "string" == type(block_chunk) then + insert(current, block_chunk) + else + add_current() + buffer:add(block_chunk) + current = { } + end end - end) - val = self.value_name + else + insert(current, chunk) + end end - local update = { - build.assign_one(NameProxy.index(self.accum_name, self.len_name), val), - { - "update", - self.len_name, - "+=", - 1 - } - } - insert(body, build.group(update)) - return body + if current[1] then + add_current() + end + return buffer + end, + __tostring = function(self) + return "Line<" .. tostring(util.dump(self):sub(1, -2)) .. ">" end } _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, accum_name) - self.accum_name = NameProxy("accum") - self.value_name = NameProxy("value") - self.len_name = NameProxy("len") - end, + local _class_0 = setmetatable({ + __init = function() end, __base = _base_0, - __name = "Accumulator" + __name = "Line" }, { __index = _base_0, __call = function(cls, ...) @@ -2315,730 +1244,375 @@ package.preload['moonscript.transform.accumulator'] = function() end }) _base_0.__class = _class_0 - Accumulator = _class_0 - end - local default_accumulator - default_accumulator = function(self, node) - return Accumulator():convert(node) + Line = _class_0 end - return { - Accumulator = Accumulator, - default_accumulator = default_accumulator - } - -end -package.preload['moonscript.dump'] = function() - local flat_value - flat_value = function(op, depth) - if depth == nil then - depth = 1 - end - if type(op) == "string" then - return '"' .. op .. '"' - end - if type(op) ~= "table" then - return tostring(op) - end - local items - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #op do - local item = op[_index_0] - _accum_0[_len_0] = flat_value(item, depth + 1) - _len_0 = _len_0 + 1 + do + local _base_0 = { + prepare = function() end, + render = function(self) + self:prepare() + return concat(self) end - items = _accum_0 - end - local pos = op[-1] - return "{" .. (pos and "[" .. pos .. "] " or "") .. table.concat(items, ", ") .. "}" - end - local value - value = function(op) - return flat_value(op) - end - local tree - tree = function(block) - local _list_0 = block - for _index_0 = 1, #_list_0 do - local value = _list_0[_index_0] - print(flat_value(value)) - end - end - return { - value = value, - tree = tree - } - -end -package.preload['moonscript.util'] = function() - local concat - concat = table.concat - local unpack = unpack or table.unpack - local type = type - local moon = { - is_object = function(value) - return type(value) == "table" and value.__class - end, - is_a = function(thing, t) - if not (type(thing) == "table") then - return false + } + _base_0.__index = _base_0 + local _class_0 = setmetatable({ + __init = function(self, fn) + self.prepare = fn + end, + __base = _base_0, + __name = "DelayedLine" + }, { + __index = _base_0, + __call = function(cls, ...) + local _self_0 = setmetatable({}, _base_0) + cls.__init(_self_0, ...) + return _self_0 end - local cls = thing.__class - while cls do - if cls == t then - return true + }) + _base_0.__class = _class_0 + DelayedLine = _class_0 + end + do + local _base_0 = { + header = "do", + footer = "end", + export_all = false, + export_proper = false, + value_compilers = value_compilers, + __tostring = function(self) + local h + if "string" == type(self.header) then + h = self.header + else + h = unpack(self.header:render({ })) end - cls = cls.__parent - end - return false - end, - type = function(value) - local base_type = type(value) - if base_type == "table" then - local cls = value.__class - if cls then - return cls + return "Block<" .. tostring(h) .. "> <- " .. tostring(self.parent) + end, + set = function(self, name, value) + self._state[name] = value + end, + get = function(self, name) + return self._state[name] + end, + get_current = function(self, name) + return rawget(self._state, name) + end, + listen = function(self, name, fn) + self._listeners[name] = fn + end, + unlisten = function(self, name) + self._listeners[name] = nil + end, + send = function(self, name, ...) + do + local fn = self._listeners[name] + if fn then + return fn(self, ...) + end end - end - return base_type - end - } - local pos_to_line - pos_to_line = function(str, pos) - local line = 1 - for _ in str:sub(1, pos):gmatch("\n") do - line = line + 1 - end - return line - end - local trim - trim = function(str) - return str:match("^%s*(.-)%s*$") - end - local get_line - get_line = function(str, line_num) - for line in str:gmatch("([^\n]*)\n?") do - if line_num == 1 then - return line - end - line_num = line_num - 1 - end - end - local get_closest_line - get_closest_line = function(str, line_num) - local line = get_line(str, line_num) - if (not line or trim(line) == "") and line_num > 1 then - return get_closest_line(str, line_num - 1) - else - return line, line_num - end - end - local split - split = function(str, delim) - if str == "" then - return { } - end - str = str .. delim - local _accum_0 = { } - local _len_0 = 1 - for m in str:gmatch("(.-)" .. delim) do - _accum_0[_len_0] = m - _len_0 = _len_0 + 1 - end - return _accum_0 - end - local dump - dump = function(what) - local seen = { } - local _dump - _dump = function(what, depth) - if depth == nil then - depth = 0 - end - local t = type(what) - if t == "string" then - return '"' .. what .. '"\n' - elseif t == "table" then - if seen[what] then - return "recursion(" .. tostring(what) .. ")...\n" - end - seen[what] = true - depth = depth + 1 - local lines + end, + declare = function(self, names) + local undeclared do local _accum_0 = { } local _len_0 = 1 - for k, v in pairs(what) do - _accum_0[_len_0] = (" "):rep(depth * 4) .. "[" .. tostring(k) .. "] = " .. _dump(v, depth) - _len_0 = _len_0 + 1 + for _index_0 = 1, #names do + local _continue_0 = false + repeat + local name = names[_index_0] + local is_local = false + local real_name + local _exp_0 = mtype(name) + if LocalName == _exp_0 then + is_local = true + real_name = name:get_name(self) + elseif NameProxy == _exp_0 then + real_name = name:get_name(self) + elseif "table" == _exp_0 then + real_name = name[1] == "ref" and name[2] + elseif "string" == _exp_0 then + real_name = name + end + if not (is_local or real_name and not self:has_name(real_name, true)) then + _continue_0 = true + break + end + self:put_name(real_name) + if self:name_exported(real_name) then + _continue_0 = true + break + end + local _value_0 = real_name + _accum_0[_len_0] = _value_0 + _len_0 = _len_0 + 1 + _continue_0 = true + until true + if not _continue_0 then + break + end end - lines = _accum_0 + undeclared = _accum_0 end - seen[what] = false - return "{\n" .. concat(lines) .. (" "):rep((depth - 1) * 4) .. "}\n" - else - return tostring(what) .. "\n" - end - end - return _dump(what) - end - local debug_posmap - debug_posmap = function(posmap, moon_code, lua_code) - local tuples - do - local _accum_0 = { } - local _len_0 = 1 - for k, v in pairs(posmap) do - _accum_0[_len_0] = { - k, - v - } - _len_0 = _len_0 + 1 - end - tuples = _accum_0 - end - table.sort(tuples, function(a, b) - return a[1] < b[1] - end) - local lines - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #tuples do - local pair = tuples[_index_0] - local lua_line, pos = unpack(pair) - local moon_line = pos_to_line(moon_code, pos) - local lua_text = get_line(lua_code, lua_line) - local moon_text = get_closest_line(moon_code, moon_line) - local _value_0 = tostring(pos) .. "\t " .. tostring(lua_line) .. ":[ " .. tostring(trim(lua_text)) .. " ] >> " .. tostring(moon_line) .. ":[ " .. tostring(trim(moon_text)) .. " ]" - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - end - lines = _accum_0 - end - return concat(lines, "\n") - end - local setfenv = setfenv or function(fn, env) - local name - local i = 1 - while true do - name = debug.getupvalue(fn, i) - if not name or name == "_ENV" then - break - end - i = i + 1 - end - if name then - debug.upvaluejoin(fn, i, (function() - return env - end), 1) - end - return fn - end - local getfenv = getfenv or function(fn) - local i = 1 - while true do - local name, val = debug.getupvalue(fn, i) - if not (name) then - break - end - if name == "_ENV" then - return val - end - i = i + 1 - end - return nil - end - local get_options - get_options = function(...) - local count = select("#", ...) - local opts = select(count, ...) - if type(opts) == "table" then - return opts, unpack({ - ... - }, nil, count - 1) - else - return { }, ... - end - end - local safe_module - safe_module = function(name, tbl) - return setmetatable(tbl, { - __index = function(self, key) - return error("Attempted to import non-existent `" .. tostring(key) .. "` from " .. tostring(name)) - end - }) - end - return { - moon = moon, - pos_to_line = pos_to_line, - get_closest_line = get_closest_line, - get_line = get_line, - trim = trim, - split = split, - dump = dump, - debug_posmap = debug_posmap, - getfenv = getfenv, - setfenv = setfenv, - get_options = get_options, - unpack = unpack, - safe_module = safe_module - } - -end -package.preload['moonscript.cmd.lint'] = function() - local insert - insert = table.insert - local Set - Set = require("moonscript.data").Set - local Block - Block = require("moonscript.compile").Block - local mtype - mtype = require("moonscript.util").moon.type - local default_whitelist = Set({ - '_G', - '_VERSION', - 'assert', - 'bit32', - 'collectgarbage', - 'coroutine', - 'debug', - 'dofile', - 'error', - 'getfenv', - 'getmetatable', - 'io', - 'ipairs', - 'load', - 'loadfile', - 'loadstring', - 'math', - 'module', - 'next', - 'os', - 'package', - 'pairs', - 'pcall', - 'print', - 'rawequal', - 'rawget', - 'rawlen', - 'rawset', - 'require', - 'select', - 'setfenv', - 'setmetatable', - 'string', - 'table', - 'tonumber', - 'tostring', - 'type', - 'unpack', - 'xpcall', - "nil", - "true", - "false" - }) - local LinterBlock - do - local _class_0 - local _parent_0 = Block - local _base_0 = { - lint_mark_used = function(self, name) - if self.lint_unused_names and self.lint_unused_names[name] then - self.lint_unused_names[name] = false - return + return undeclared + end, + whitelist_names = function(self, names) + self._name_whitelist = Set(names) + end, + name_exported = function(self, name) + if self.export_all then + return true end - if self.parent then - return self.parent:lint_mark_used(name) + if self.export_proper and name:match("^%u") then + return true end end, - lint_check_unused = function(self) - if not (self.lint_unused_names and next(self.lint_unused_names)) then - return + put_name = function(self, name, ...) + local value = ... + if select("#", ...) == 0 then + value = true end - local names_by_position = { } - for name, pos in pairs(self.lint_unused_names) do - local _continue_0 = false - repeat - if not (pos) then - _continue_0 = true - break - end - names_by_position[pos] = names_by_position[pos] or { } - insert(names_by_position[pos], name) - _continue_0 = true - until true - if not _continue_0 then - break - end + if NameProxy == mtype(name) then + name = name:get_name(self) end - local tuples - do - local _accum_0 = { } - local _len_0 = 1 - for pos, names in pairs(names_by_position) do - _accum_0[_len_0] = { - pos, - names - } - _len_0 = _len_0 + 1 + self._names[name] = value + end, + has_name = function(self, name, skip_exports) + if not skip_exports and self:name_exported(name) then + return true + end + local yes = self._names[name] + if yes == nil and self.parent then + if not self._name_whitelist or self._name_whitelist[name] then + return self.parent:has_name(name, true) end - tuples = _accum_0 + else + return yes end - table.sort(tuples, function(a, b) - return a[1] < b[1] - end) - for _index_0 = 1, #tuples do - local _des_0 = tuples[_index_0] - local pos, names - pos, names = _des_0[1], _des_0[2] - insert(self:get_root_block().lint_errors, { - "assigned but unused " .. tostring(table.concat((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_1 = 1, #names do - local n = names[_index_1] - _accum_0[_len_0] = "`" .. tostring(n) .. "`" - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ")), - pos - }) + end, + is_local = function(self, node) + local t = mtype(node) + if t == "string" then + return self:has_name(node, false) + end + if t == NameProxy or t == LocalName then + return true end + if t == "table" then + if node[1] == "ref" or (node[1] == "chain" and #node == 2) then + return self:is_local(node[2]) + end + end + return false end, - render = function(self, ...) - self:lint_check_unused() - return _class_0.__parent.__base.render(self, ...) + free_name = function(self, prefix, dont_put) + prefix = prefix or "moon" + local searching = true + local name, i = nil, 0 + while searching do + name = concat({ + "", + prefix, + i + }, "_") + i = i + 1 + searching = self:has_name(name, true) + end + if not dont_put then + self:put_name(name) + end + return name end, - block = function(self, ...) + init_free_var = function(self, prefix, value) + local name = self:free_name(prefix, true) + self:stm({ + "assign", + { + name + }, + { + value + } + }) + return name + end, + add = function(self, item) + self._lines:add(item) + return item + end, + render = function(self, buffer) + buffer:add(self.header) + buffer:mark_pos(self.pos) + if self.next then + buffer:add(self._lines) + self.next:render(buffer) + else + if #self._lines == 0 and "string" == type(buffer[#buffer]) then + buffer[#buffer] = buffer[#buffer] .. (" " .. (unpack(Lines():add(self.footer)))) + else + buffer:add(self._lines) + buffer:add(self.footer) + buffer:mark_pos(self.pos) + end + end + return buffer + end, + block = function(self, header, footer) + return Block(self, header, footer) + end, + line = function(self, ...) do - local _with_0 = _class_0.__parent.__base.block(self, ...) - _with_0.block = self.block - _with_0.render = self.render - _with_0.get_root_block = self.get_root_block - _with_0.lint_check_unused = self.lint_check_unused - _with_0.lint_mark_used = self.lint_mark_used - _with_0.value_compilers = self.value_compilers - _with_0.statement_compilers = self.statement_compilers + local _with_0 = Line() + _with_0:append(...) return _with_0 end - end - } - _base_0.__index = _base_0 - setmetatable(_base_0, _parent_0.__base) - _class_0 = setmetatable({ - __init = function(self, whitelist_globals, ...) - if whitelist_globals == nil then - whitelist_globals = default_whitelist + end, + is_stm = function(self, node) + return statement_compilers[ntype(node)] ~= nil + end, + is_value = function(self, node) + local t = ntype(node) + return self.value_compilers[t] ~= nil or t == "value" + end, + name = function(self, node, ...) + if type(node) == "string" then + return node + else + return self:value(node, ...) end - _class_0.__parent.__init(self, ...) - self.get_root_block = function() - return self + end, + value = function(self, node, ...) + node = self.transform.value(node) + local action + if type(node) ~= "table" then + action = "raw_value" + else + action = node[1] end - self.lint_errors = { } - local vc = self.value_compilers - self.value_compilers = setmetatable({ - ref = function(block, val) - local name = val[2] - if not (block:has_name(name) or whitelist_globals[name] or name:match("%.")) then - insert(self.lint_errors, { - "accessing global `" .. tostring(name) .. "`", - val[-1] - }) + local fn = self.value_compilers[action] + if not (fn) then + error({ + "compile-error", + "Failed to find value compiler for: " .. dump.value(node), + node[-1] + }) + end + local out = fn(self, node, ...) + if type(node) == "table" and node[-1] then + if type(out) == "string" then + do + local _with_0 = Line() + _with_0:append(out) + out = _with_0 end - block:lint_mark_used(name) - return vc.ref(block, val) end - }, { - __index = vc - }) - local sc = self.statement_compilers - self.statement_compilers = setmetatable({ - assign = function(block, node) - local names = node[2] - for _index_0 = 1, #names do - local _continue_0 = false - repeat - local name = names[_index_0] - if type(name) == "table" and name[1] == "temp_name" then - _continue_0 = true - break - end - local real_name, is_local = block:extract_assign_name(name) - if not (is_local or real_name and not block:has_name(real_name, true)) then - _continue_0 = true - break - end - if real_name == "_" then - _continue_0 = true - break - end - block.lint_unused_names = block.lint_unused_names or { } - block.lint_unused_names[real_name] = node[-1] or 0 - _continue_0 = true - until true - if not _continue_0 then - break - end + out.pos = node[-1] + end + return out + end, + values = function(self, values, delim) + delim = delim or ', ' + do + local _with_0 = Line() + _with_0:append_list((function() + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #values do + local v = values[_index_0] + _accum_0[_len_0] = self:value(v) + _len_0 = _len_0 + 1 end - return sc.assign(block, node) - end - }, { - __index = sc - }) + return _accum_0 + end)(), delim) + return _with_0 + end end, - __base = _base_0, - __name = "LinterBlock", - __parent = _parent_0 - }, { - __index = function(cls, name) - local val = rawget(_base_0, name) - if val == nil then - local parent = rawget(cls, "__parent") - if parent then - return parent[name] + stm = function(self, node, ...) + if not node then + return + end + node = self.transform.statement(node) + local result + do + local fn = statement_compilers[ntype(node)] + if fn then + result = fn(self, node, ...) + else + if has_value(node) then + result = self:stm({ + "assign", + { + "_" + }, + { + node + } + }) + else + result = self:value(node) + end end - else - return val end + if result then + if type(node) == "table" and type(result) == "table" and node[-1] then + result.pos = node[-1] + end + self:add(result) + end + return nil + end, + stms = function(self, stms, ret) + if ret then + error("deprecated stms call, use transformer") + end + local current_stms, current_stm_i + current_stms, current_stm_i = self.current_stms, self.current_stm_i + self.current_stms = stms + for i = 1, #stms do + self.current_stm_i = i + self:stm(stms[i]) + end + self.current_stms = current_stms + self.current_stm_i = current_stm_i + return nil end, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + splice = function(self, fn) + local lines = { + "lines", + self._lines + } + self._lines = Lines() + return self:stms(fn(lines)) end - }) - _base_0.__class = _class_0 - if _parent_0.__inherited then - _parent_0.__inherited(_parent_0, _class_0) - end - LinterBlock = _class_0 - end - local format_lint - format_lint = function(errors, code, header) - if not (next(errors)) then - return - end - local pos_to_line, get_line - do - local _obj_0 = require("moonscript.util") - pos_to_line, get_line = _obj_0.pos_to_line, _obj_0.get_line - end - local formatted - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #errors do - local _des_0 = errors[_index_0] - local msg, pos - msg, pos = _des_0[1], _des_0[2] - if pos then - local line = pos_to_line(code, pos) - msg = "line " .. tostring(line) .. ": " .. tostring(msg) - local line_text = "> " .. get_line(code, line) - local sep_len = math.max(#msg, #line_text) - _accum_0[_len_0] = table.concat({ - msg, - ("="):rep(sep_len), - line_text - }, "\n") - else - _accum_0[_len_0] = msg - end - _len_0 = _len_0 + 1 - end - formatted = _accum_0 - end - if header then - table.insert(formatted, 1, header) - end - return table.concat(formatted, "\n\n") - end - local whitelist_for_file - do - local lint_config - whitelist_for_file = function(fname) - if not (lint_config) then - lint_config = { } - pcall(function() - lint_config = require("lint_config") - end) - end - if not (lint_config.whitelist_globals) then - return default_whitelist - end - local final_list = { } - for pattern, list in pairs(lint_config.whitelist_globals) do - if fname:match(pattern) then - for _index_0 = 1, #list do - local item = list[_index_0] - insert(final_list, item) - end - end - end - return setmetatable(Set(final_list), { - __index = default_whitelist - }) - end - end - local lint_code - lint_code = function(code, name, whitelist_globals) - if name == nil then - name = "string input" - end - local parse = require("moonscript.parse") - local tree, err = parse.string(code) - if not (tree) then - return nil, err - end - local scope = LinterBlock(whitelist_globals) - scope:stms(tree) - scope:lint_check_unused() - return format_lint(scope.lint_errors, code, name) - end - local lint_file - lint_file = function(fname) - local f, err = io.open(fname) - if not (f) then - return nil, err - end - return lint_code(f:read("*a"), fname, whitelist_for_file(fname)) - end - return { - lint_code = lint_code, - lint_file = lint_file - } - -end -package.preload['moonscript.cmd.args'] = function() - local unpack - unpack = require("moonscript.util").unpack - local parse_spec - parse_spec = function(spec) - local flags, words - if type(spec) == "table" then - flags, words = unpack(spec), spec - else - flags, words = spec, { } - end - assert("no flags for arguments") - local out = { } - for part in flags:gmatch("%w:?") do - if part:match(":$") then - out[part:sub(1, 1)] = { - value = true - } - else - out[part] = { } - end - end - return out - end - local parse_arguments - parse_arguments = function(spec, args) - spec = parse_spec(spec) - local out = { } - local remaining = { } - local last_flag = nil - for _index_0 = 1, #args do - local _continue_0 = false - repeat - local arg = args[_index_0] - local group = { } - if last_flag then - out[last_flag] = arg - _continue_0 = true - break - end + } + _base_0.__index = _base_0 + local _class_0 = setmetatable({ + __init = function(self, parent, header, footer) + self.parent, self.header, self.footer = parent, header, footer + self._lines = Lines() + self._names = { } + self._state = { } + self._listeners = { } do - local flag = arg:match("-(%w+)") - if flag then - do - local short_name = spec[flag] - if short_name then - out[short_name] = true - else - for char in flag:gmatch(".") do - out[char] = true - end - end - end - _continue_0 = true - break - end + self.transform = { + value = transform.Value:bind(self), + statement = transform.Statement:bind(self) + } end - table.insert(remaining, arg) - _continue_0 = true - until true - if not _continue_0 then - break - end - end - return out, remaining - end - return { - parse_arguments = parse_arguments, - parse_spec = parse_spec - } - -end -package.preload['moonscript.cmd.watchers'] = function() - local remove_dupes - remove_dupes = function(list, key_fn) - local seen = { } - return (function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #list do - local _continue_0 = false - repeat - local item = list[_index_0] - local key - if key_fn then - key = key_fn(item) - else - key = item - end - if seen[key] then - _continue_0 = true - break - end - seen[key] = true - local _value_0 = item - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - _continue_0 = true - until true - if not _continue_0 then - break + if self.parent then + self.root = self.parent.root + self.indent = self.parent.indent + 1 + setmetatable(self._state, { + __index = self.parent._state + }) + return setmetatable(self._listeners, { + __index = self.parent._listeners + }) + else + self.indent = 0 end - end - return _accum_0 - end)() - end - local plural - plural = function(count, word) - return tostring(count) .. " " .. tostring(word) .. tostring(count == 1 and "" or "s") - end - local Watcher - do - local _class_0 - local _base_0 = { - start_msg = "Starting watch loop (Ctrl-C to exit)", - print_start = function(self, mode, misc) - return io.stderr:write(tostring(self.start_msg) .. " with " .. tostring(mode) .. " [" .. tostring(misc) .. "]\n") - end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, file_list) - self.file_list = file_list end, __base = _base_0, - __name = "Watcher" + __name = "Block" }, { __index = _base_0, __call = function(cls, ...) @@ -3048,95 +1622,44 @@ package.preload['moonscript.cmd.watchers'] = function() end }) _base_0.__class = _class_0 - Watcher = _class_0 + Block = _class_0 end - local InotifyWacher do - local _class_0 - local _parent_0 = Watcher + local _parent_0 = Block local _base_0 = { - get_dirs = function(self) - local parse_dir - parse_dir = require("moonscript.cmd.moonc").parse_dir - local dirs - do - local _accum_0 = { } - local _len_0 = 1 - local _list_0 = self.file_list - for _index_0 = 1, #_list_0 do - local _des_0 = _list_0[_index_0] - local file_path - file_path = _des_0[1] - local dir = parse_dir(file_path) - if dir == "" then - dir = "./" - end - local _value_0 = dir - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - end - dirs = _accum_0 + __tostring = function(self) + return "RootBlock<>" + end, + root_stms = function(self, stms) + if not (self.options.implicitly_return_root == false) then + stms = transform.Statement.transformers.root_stms(self, stms) end - return remove_dupes(dirs) + return self:stms(stms) end, - each_update = function(self) - return coroutine.wrap(function() - local dirs = self:get_dirs() - self:print_start("inotify", plural(#dirs, "dir")) - local wd_table = { } - local inotify = require("inotify") - local handle = inotify.init() - for _index_0 = 1, #dirs do - local dir = dirs[_index_0] - local wd = handle:addwatch(dir, inotify.IN_CLOSE_WRITE, inotify.IN_MOVED_TO) - wd_table[wd] = dir - end - while true do - local events = handle:read() - if not (events) then - break - end - for _index_0 = 1, #events do - local _continue_0 = false - repeat - local ev = events[_index_0] - local fname = ev.name - if not (fname:match("%.moon$")) then - _continue_0 = true - break - end - local dir = wd_table[ev.wd] - if dir ~= "./" then - fname = dir .. fname - end - coroutine.yield(fname) - _continue_0 = true - until true - if not _continue_0 then - break - end - end - end - end) + render = function(self) + local buffer = self._lines:flatten() + if buffer[#buffer] == "\n" then + buffer[#buffer] = nil + end + return table.concat(buffer) end } _base_0.__index = _base_0 setmetatable(_base_0, _parent_0.__base) - _class_0 = setmetatable({ - __init = function(self, ...) - return _class_0.__parent.__init(self, ...) + local _class_0 = setmetatable({ + __init = function(self, options) + self.options = options + self.root = self + return _parent_0.__init(self) end, __base = _base_0, - __name = "InotifyWacher", + __name = "RootBlock", __parent = _parent_0 }, { __index = function(cls, name) local val = rawget(_base_0, name) if val == nil then - local parent = rawget(cls, "__parent") - if parent then - return parent[name] - end + return _parent_0[name] else return val end @@ -3148,245 +1671,132 @@ package.preload['moonscript.cmd.watchers'] = function() end }) _base_0.__class = _class_0 - local self = _class_0 - self.available = function(self) - return pcall(function() - return require("inotify") - end) - end if _parent_0.__inherited then _parent_0.__inherited(_parent_0, _class_0) end - InotifyWacher = _class_0 + RootBlock = _class_0 end - local SleepWatcher - do - local _class_0 - local _parent_0 = Watcher - local _base_0 = { - polling_rate = 1.0, - get_sleep_func = function(self) - local sleep - pcall(function() - sleep = require("socket").sleep - end) - sleep = sleep or require("moonscript")._sleep - if not (sleep) then - error("Missing sleep function; install LuaSocket") - end - return sleep - end, - each_update = function(self) - return coroutine.wrap(function() - local lfs = require("lfs") - local sleep = self:get_sleep_func() - self:print_start("polling", plural(#self.file_list, "files")) - local mod_time = { } - while true do - local _list_0 = self.file_list - for _index_0 = 1, #_list_0 do - local _continue_0 = false - repeat - local _des_0 = _list_0[_index_0] - local file - file = _des_0[1] - local time = lfs.attributes(file, "modification") - if not (time) then - mod_time[file] = nil - _continue_0 = true - break - end - if not (mod_time[file]) then - mod_time[file] = time - _continue_0 = true - break - end - if time > mod_time[file] then - mod_time[file] = time - coroutine.yield(file) - end - _continue_0 = true - until true - if not _continue_0 then - break - end - end - sleep(self.polling_rate) - end - end) - end - } - _base_0.__index = _base_0 - setmetatable(_base_0, _parent_0.__base) - _class_0 = setmetatable({ - __init = function(self, ...) - return _class_0.__parent.__init(self, ...) - end, - __base = _base_0, - __name = "SleepWatcher", - __parent = _parent_0 - }, { - __index = function(cls, name) - local val = rawget(_base_0, name) - if val == nil then - local parent = rawget(cls, "__parent") - if parent then - return parent[name] - end - else - return val - end - end, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 - end - }) - _base_0.__class = _class_0 - if _parent_0.__inherited then - _parent_0.__inherited(_parent_0, _class_0) + local format_error + format_error = function(msg, pos, file_str) + local line_message + if pos then + local line = pos_to_line(file_str, pos) + local line_str + line_str, line = get_closest_line(file_str, line) + line_str = line_str or "" + line_message = (" [%d] >> %s"):format(line, trim(line_str)) end - SleepWatcher = _class_0 + return concat({ + "Compile error: " .. msg, + line_message + }, "\n") end - return { - Watcher = Watcher, - SleepWatcher = SleepWatcher, - InotifyWacher = InotifyWacher - } - -end -package.preload['moonscript.cmd.coverage'] = function() - local log - log = function(str) - if str == nil then - str = "" + local value + value = function(value) + local out = nil + do + local _with_0 = RootBlock() + _with_0:add(_with_0:value(value)) + out = _with_0:render() end - return io.stderr:write(str .. "\n") + return out end - local create_counter - create_counter = function() - return setmetatable({ }, { - __index = function(self, name) - do - local tbl = setmetatable({ }, { - __index = function(self) - return 0 - end - }) - self[name] = tbl - return tbl + local tree + tree = function(tree, options) + if options == nil then + options = { } + end + assert(tree, "missing tree") + local scope = (options.scope or RootBlock)(options) + local runner = coroutine.create(function() + return scope:root_stms(tree) + end) + local success, err = coroutine.resume(runner) + if not (success) then + local error_msg, error_pos + if type(err) == "table" then + local error_type = err[1] + local _exp_0 = err[1] + if "user-error" == _exp_0 or "compile-error" == _exp_0 then + error_msg, error_pos = unpack(err, 2) + else + error_msg, error_pos = error("Unknown error thrown", util.dump(error_msg)) end + else + error_msg, error_pos = concat({ + err, + debug.traceback(runner) + }, "\n") end - }) + return nil, error_msg, error_pos or scope.last_pos + end + local lua_code = scope:render() + local posmap = scope._lines:flatten_posmap() + return lua_code, posmap end - local position_to_lines - position_to_lines = function(file_content, positions) - local lines = { } - local current_pos = 0 - local line_no = 1 - for char in file_content:gmatch(".") do - do - local count = rawget(positions, current_pos) - if count then - lines[line_no] = count - end - end - if char == "\n" then - line_no = line_no + 1 - end - current_pos = current_pos + 1 + do + local data = require("moonscript.data") + for name, cls in pairs({ + Line = Line, + Lines = Lines, + DelayedLine = DelayedLine + }) do + data[name] = cls end - return lines end - local format_file - format_file = function(fname, positions) - fname = fname:gsub("^@", "") - local file = assert(io.open(fname)) - local content = file:read("*a") - file:close() - local lines = position_to_lines(content, positions) - log("------| @" .. tostring(fname)) - local line_no = 1 - for line in (content .. "\n"):gmatch("(.-)\n") do - local foramtted_no = ("% 5d"):format(line_no) - local sym = lines[line_no] and "*" or " " - log(tostring(sym) .. tostring(foramtted_no) .. "| " .. tostring(line)) - line_no = line_no + 1 + return { + tree = tree, + value = value, + format_error = format_error, + Block = Block, + RootBlock = RootBlock + } + +end +package.preload['moonscript.data'] = function() + local concat, remove, insert + do + local _obj_0 = table + concat, remove, insert = _obj_0.concat, _obj_0.remove, _obj_0.insert + end + local Set + Set = function(items) + local _tbl_0 = { } + for _index_0 = 1, #items do + local k = items[_index_0] + _tbl_0[k] = true end - return log() + return _tbl_0 end - local CodeCoverage + local Stack do - local _class_0 local _base_0 = { - reset = function(self) - self.line_counts = create_counter() - end, - start = function(self) - return debug.sethook((function() - local _base_1 = self - local _fn_0 = _base_1.process_line - return function(...) - return _fn_0(_base_1, ...) - end - end)(), "l") - end, - stop = function(self) - return debug.sethook() - end, - print_results = function(self) - return self:format_results() + __tostring = function(self) + return "" end, - process_line = function(self, _, line_no) - local debug_data = debug.getinfo(2, "S") - local source = debug_data.source - self.line_counts[source][line_no] = self.line_counts[source][line_no] + 1 + pop = function(self) + return remove(self) end, - format_results = function(self) - local line_table = require("moonscript.line_tables") - local positions = create_counter() - for file, lines in pairs(self.line_counts) do - local _continue_0 = false - repeat - local file_table = line_table[file] - if not (file_table) then - _continue_0 = true - break - end - for line, count in pairs(lines) do - local _continue_1 = false - repeat - local position = file_table[line] - if not (position) then - _continue_1 = true - break - end - positions[file][position] = positions[file][position] + count - _continue_1 = true - until true - if not _continue_1 then - break - end - end - _continue_0 = true - until true - if not _continue_0 then - break - end - end - for file, ps in pairs(positions) do - format_file(file, ps) + push = function(self, value, ...) + insert(self, value) + if ... then + return self:push(...) + else + return value end + end, + top = function(self) + return self[#self] end } _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self) - return self:reset() + local _class_0 = setmetatable({ + __init = function(self, ...) + self:push(...) + return nil end, __base = _base_0, - __name = "CodeCoverage" + __name = "Stack" }, { __index = _base_0, __call = function(cls, ...) @@ -3396,1734 +1806,1500 @@ package.preload['moonscript.cmd.coverage'] = function() end }) _base_0.__class = _class_0 - CodeCoverage = _class_0 + Stack = _class_0 end + local lua_keywords = Set({ + 'and', + 'break', + 'do', + 'else', + 'elseif', + 'end', + 'false', + 'for', + 'function', + 'if', + 'in', + 'local', + 'nil', + 'not', + 'or', + 'repeat', + 'return', + 'then', + 'true', + 'until', + 'while' + }) return { - CodeCoverage = CodeCoverage + Set = Set, + Stack = Stack, + lua_keywords = lua_keywords } end -package.preload['moonscript.cmd.moonc'] = function() - local lfs = require("lfs") - local split - split = require("moonscript.util").split - local dirsep, dirsep_chars, mkdir, normalize_dir, parse_dir, parse_file, convert_path, format_time, gettime, compile_file_text, write_file, compile_and_write, is_abs_path, path_to_target - dirsep = package.config:sub(1, 1) - if dirsep == "\\" then - dirsep_chars = "\\/" - else - dirsep_chars = dirsep - end - mkdir = function(path) - local chunks = split(path, dirsep) - local accum - for _index_0 = 1, #chunks do - local dir = chunks[_index_0] - accum = accum and tostring(accum) .. tostring(dirsep) .. tostring(dir) or dir - lfs.mkdir(accum) - end - return lfs.attributes(path, "mode") - end - normalize_dir = function(path) - return path:match("^(.-)[" .. tostring(dirsep_chars) .. "]*$") .. dirsep - end - parse_dir = function(path) - return (path:match("^(.-)[^" .. tostring(dirsep_chars) .. "]*$")) - end - parse_file = function(path) - return (path:match("^.-([^" .. tostring(dirsep_chars) .. "]*)$")) - end - convert_path = function(path) - local new_path = path:gsub("%.moon$", ".lua") - if new_path == path then - new_path = path .. ".lua" - end - return new_path - end - format_time = function(time) - return ("%.3fms"):format(time * 1000) - end - do - local socket - gettime = function() - if socket == nil then - pcall(function() - socket = require("socket") - end) - if not (socket) then - socket = false - end - end - if socket then - return socket.gettime() - else - return nil, "LuaSocket needed for benchmark" - end - end - end - compile_file_text = function(text, opts) - if opts == nil then - opts = { } - end - local parse = require("moonscript.parse") - local compile = require("moonscript.compile") - local parse_time - if opts.benchmark then - parse_time = assert(gettime()) - end - local tree, err = parse.string(text) - if not (tree) then - return nil, err - end - if parse_time then - parse_time = gettime() - parse_time +package.preload['moonscript.dump'] = function() + local flat_value + flat_value = function(op, depth) + if depth == nil then + depth = 1 end - if opts.show_parse_tree then - local dump = require("moonscript.dump") - dump.tree(tree) - return true + if type(op) == "string" then + return '"' .. op .. '"' end - local compile_time - if opts.benchmark then - compile_time = gettime() + if type(op) ~= "table" then + return tostring(op) end + local items do - local mod = opts.transform_module - if mod then - tree = assert(require(mod)(tree)) + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #op do + local item = op[_index_0] + _accum_0[_len_0] = flat_value(item, depth + 1) + _len_0 = _len_0 + 1 end + items = _accum_0 end - local code, posmap_or_err, err_pos = compile.tree(tree) - if not (code) then - return nil, compile.format_error(posmap_or_err, err_pos, text) - end - if compile_time then - compile_time = gettime() - compile_time - end - if opts.show_posmap then - local debug_posmap - debug_posmap = require("moonscript.util").debug_posmap - print("Pos", "Lua", ">>", "Moon") - print(debug_posmap(posmap_or_err, text, code)) - return true - end - if opts.benchmark then - print(table.concat({ - opts.fname or "stdin", - "Parse time \t" .. format_time(parse_time), - "Compile time\t" .. format_time(compile_time), - "" - }, "\n")) - return true - end - return code + local pos = op[-1] + return "{" .. (pos and "[" .. pos .. "] " or "") .. table.concat(items, ", ") .. "}" end - write_file = function(fname, code) - mkdir(parse_dir(fname)) - local f, err = io.open(fname, "w") - if not (f) then - return nil, err - end - assert(f:write(code)) - assert(f:write("\n")) - f:close() - return "build" + local value + value = function(op) + return flat_value(op) end - compile_and_write = function(src, dest, opts) - if opts == nil then - opts = { } - end - local f = io.open(src) - if not (f) then - return nil, "Can't find file" - end - local text = assert(f:read("*a")) - f:close() - local code, err = compile_file_text(text, opts) - if not code then - return nil, err - end - if code == true then - return true - end - if opts.print then - print(code) - return true + local tree + tree = function(block) + local _list_0 = block + for _index_0 = 1, #_list_0 do + value = _list_0[_index_0] + print(flat_value(value)) end - return write_file(dest, code) end - is_abs_path = function(path) - local first = path:sub(1, 1) - if dirsep == "\\" then - return first == "/" or first == "\\" or path:sub(2, 1) == ":" - else - return first == dirsep + return { + value = value, + tree = tree + } + +end +package.preload['moonscript.errors'] = function() + local util = require("moonscript.util") + local lpeg = require("lpeg") + local concat, insert + do + local _obj_0 = table + concat, insert = _obj_0.concat, _obj_0.insert + end + local split, pos_to_line + split, pos_to_line = util.split, util.pos_to_line + local user_error + user_error = function(...) + return error({ + "user-error", + ... + }) + end + local lookup_line + lookup_line = function(fname, pos, cache) + if not cache[fname] then + do + local _with_0 = io.open(fname) + cache[fname] = _with_0:read("*a") + _with_0:close() + end end + return pos_to_line(cache[fname], pos) end - path_to_target = function(path, target_dir, base_dir) - if target_dir == nil then - target_dir = nil + local reverse_line_number + reverse_line_number = function(fname, line_table, line_num, cache) + for i = line_num, 0, -1 do + if line_table[i] then + return lookup_line(fname, line_table[i], cache) + end end - if base_dir == nil then - base_dir = nil + return "unknown" + end + local truncate_traceback + truncate_traceback = function(traceback, chunk_func) + if chunk_func == nil then + chunk_func = "moonscript_chunk" end - local target = convert_path(path) - if target_dir then - target_dir = normalize_dir(target_dir) + traceback = split(traceback, "\n") + local stop = #traceback + while stop > 1 do + if traceback[stop]:match(chunk_func) then + break + end + stop = stop - 1 end - if base_dir and target_dir then - local head = base_dir:match("^(.-)[^" .. tostring(dirsep_chars) .. "]*[" .. tostring(dirsep_chars) .. "]?$") - if head then - local start, stop = target:find(head, 1, true) - if start == 1 then - target = target:sub(stop + 1) - end + do + local _accum_0 = { } + local _len_0 = 1 + local _max_0 = stop + for _index_0 = 1, _max_0 < 0 and #traceback + _max_0 or _max_0 do + local t = traceback[_index_0] + _accum_0[_len_0] = t + _len_0 = _len_0 + 1 end + traceback = _accum_0 end - if target_dir then - if is_abs_path(target) then - target = parse_file(target) + local rep = "function '" .. chunk_func .. "'" + traceback[#traceback] = traceback[#traceback]:gsub(rep, "main chunk") + return concat(traceback, "\n") + end + local rewrite_traceback + rewrite_traceback = function(text, err) + local line_tables = require("moonscript.line_tables") + local V, S, Ct, C + V, S, Ct, C = lpeg.V, lpeg.S, lpeg.Ct, lpeg.C + local header_text = "stack traceback:" + local Header, Line = V("Header"), V("Line") + local Break = lpeg.S("\n") + local g = lpeg.P({ + Header, + Header = header_text * Break * Ct(Line ^ 1), + Line = "\t" * C((1 - Break) ^ 0) * (Break + -1) + }) + local cache = { } + local rewrite_single + rewrite_single = function(trace) + local fname, line, msg = trace:match('^%[string "(.-)"]:(%d+): (.*)$') + local tbl = line_tables[fname] + if fname and tbl then + return concat({ + fname, + ":", + reverse_line_number(fname, tbl, line, cache), + ": ", + "(", + line, + ") ", + msg + }) + else + return trace end - target = target_dir .. target end - return target + err = rewrite_single(err) + local match = g:match(text) + if not (match) then + return nil + end + for i, trace in ipairs(match) do + match[i] = rewrite_single(trace) + end + return concat({ + "moon: " .. err, + header_text, + "\t" .. concat(match, "\n\t") + }, "\n") end return { - dirsep = dirsep, - mkdir = mkdir, - normalize_dir = normalize_dir, - parse_dir = parse_dir, - parse_file = parse_file, - convert_path = convert_path, - gettime = gettime, - format_time = format_time, - path_to_target = path_to_target, - compile_file_text = compile_file_text, - compile_and_write = compile_and_write + rewrite_traceback = rewrite_traceback, + truncate_traceback = truncate_traceback, + user_error = user_error, + reverse_line_number = reverse_line_number } end +package.preload['moonscript'] = function() + return require("moonscript.base") +end package.preload['moonscript.line_tables'] = function() return { } end -package.preload['moonscript.compile.value'] = function() - local util = require("moonscript.util") - local data = require("moonscript.data") - local ntype - ntype = require("moonscript.types").ntype - local user_error - user_error = require("moonscript.errors").user_error - local concat, insert - do - local _obj_0 = table - concat, insert = _obj_0.concat, _obj_0.insert +package.preload['moonscript.parse'] = function() + + local util = require"moonscript.util" + + local lpeg = require"lpeg" + + local debug_grammar = false + + local data = require"moonscript.data" + local types = require"moonscript.types" + + local ntype = types.ntype + + local dump = util.dump + local trim = util.trim + + local getfenv = util.getfenv + local setfenv = util.setfenv + local unpack = util.unpack + + local Stack = data.Stack + + local function count_indent(str) + local sum = 0 + for v in str:gmatch("[\t ]") do + if v == ' ' then sum = sum + 1 end + if v == '\t' then sum = sum + 4 end + end + return sum end - local unpack - unpack = util.unpack - local table_delim = "," - local string_chars = { - ["\r"] = "\\r", - ["\n"] = "\\n" - } - return { - scoped = function(self, node) - local _, before, value, after - _, before, value, after = node[1], node[2], node[3], node[4] - _ = before and before:call(self) - do - local _with_0 = self:value(value) - _ = after and after:call(self) - return _with_0 - end - end, - exp = function(self, node) - local _comp - _comp = function(i, value) - if i % 2 == 1 and value == "!=" then - value = "~=" - end - return self:value(value) + + local R, S, V, P = lpeg.R, lpeg.S, lpeg.V, lpeg.P + local C, Ct, Cmt, Cg, Cb, Cc = lpeg.C, lpeg.Ct, lpeg.Cmt, lpeg.Cg, lpeg.Cb, lpeg.Cc + + lpeg.setmaxstack(10000) + + local White = S" \t\r\n"^0 + local _Space = S" \t"^0 + local Break = P"\r"^-1 * P"\n" + local Stop = Break + -1 + local Indent = C(S"\t "^0) / count_indent + + local Comment = P"--" * (1 - S"\r\n")^0 * #Stop + local Space = _Space * Comment^-1 + local SomeSpace = S" \t"^1 * Comment^-1 + + local SpaceBreak = Space * Break + local EmptyLine = SpaceBreak + + local AlphaNum = R("az", "AZ", "09", "__") + + local _Name = C(R("az", "AZ", "__") * AlphaNum^0) + local Name = Space * _Name + + local Num = P"0x" * R("09", "af", "AF")^1 * (S"uU"^-1 * S"lL"^2)^-1 + + R"09"^1 * (S"uU"^-1 * S"lL"^2) + + ( + R"09"^1 * (P"." * R"09"^1)^-1 + + P"." * R"09"^1 + ) * (S"eE" * P"-"^-1 * R"09"^1)^-1 + + Num = Space * (Num / function(value) return {"number", value} end) + + local FactorOp = Space * C(S"+-") + local TermOp = Space * C(S"*/%^") + + local Shebang = P"#!" * P(1 - Stop)^0 + + -- can't have P(false) because it causes preceding patterns not to run + local Cut = P(function() return false end) + + local function ensure(patt, finally) + return patt * finally + finally * Cut + end + + -- auto declare Proper variables with lpeg.V + local function wrap_env(fn) + local env = getfenv(fn) + local wrap_name = V + + if debug_grammar then + local indent = 0 + local indent_char = " " + + local function iprint(...) + local args = {...} + for i=1,#args do + args[i] = tostring(args[i]) + end + + io.stdout:write(indent_char:rep(indent) .. table.concat(args, ", ") .. "\n") + end + + wrap_name = function(name) + local v = V(name) + v = Cmt("", function() + iprint("* " .. name) + indent = indent + 1 + return true + end) * Cmt(v, function(str, pos, ...) + iprint(name, true) + indent = indent - 1 + return true, ... + end) + Cmt("", function() + iprint(name, false) + indent = indent - 1 + return false + end) + return v + end + end + + return setfenv(fn, setmetatable({}, { + __index = function(self, name) + local value = env[name] + if value ~= nil then return value end + + if name:match"^[A-Z][A-Za-z0-9]*$" then + local v = wrap_name(name) + rawset(self, name, v) + return v + end + error("unknown variable referenced: "..name) + end + })) + end + + local function extract_line(str, start_pos) + str = str:sub(start_pos) + m = str:match"^(.-)\n" + if m then return m end + return str:match"^.-$" + end + + local function mark(name) + return function(...) + return {name, ...} + end + end + + local function insert_pos(pos, value) + if type(value) == "table" then + value[-1] = pos end - do - local _with_0 = self:line() - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for i, v in ipairs(node) do - if i > 1 then - _accum_0[_len_0] = _comp(i, v) - _len_0 = _len_0 + 1 - end - end - return _accum_0 - end)(), " ") - return _with_0 - end - end, - explist = function(self, node) - do - local _with_0 = self:line() - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 2, #node do - local v = node[_index_0] - _accum_0[_len_0] = self:value(v) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") - return _with_0 - end - end, - parens = function(self, node) - return self:line("(", self:value(node[2]), ")") - end, - string = function(self, node) - local delim, inner = unpack(node, 2) - local end_delim = delim:gsub("%[", "]") - if delim == "'" or delim == '"' then - inner = inner:gsub("[\r\n]", string_chars) - end - return delim .. inner .. end_delim - end, - chain = function(self, node) - local callee = node[2] - local callee_type = ntype(callee) - local item_offset = 3 - if callee_type == "dot" or callee_type == "colon" or callee_type == "index" then - callee = self:get("scope_var") - if not (callee) then - user_error("Short-dot syntax must be called within a with block") - end - item_offset = 2 - end - if callee_type == "ref" and callee[2] == "super" or callee == "super" then - do - local sup = self:get("super") - if sup then - return self:value(sup(self, node)) - end - end - end - local chain_item - chain_item = function(node) - local t, arg = unpack(node) - if t == "call" then - return "(", self:values(arg), ")" - elseif t == "index" then - return "[", self:value(arg), "]" - elseif t == "dot" then - return ".", tostring(arg) - elseif t == "colon" then - return ":", tostring(arg) - elseif t == "colon_stub" then - return user_error("Uncalled colon stub") - else - return error("Unknown chain action: " .. tostring(t)) - end - end - if (callee_type == "self" or callee_type == "self_class") and node[3] and ntype(node[3]) == "call" then - callee[1] = callee_type .. "_colon" - end - local callee_value = self:value(callee) - if ntype(callee) == "exp" then - callee_value = self:line("(", callee_value, ")") - end - local actions - do - local _with_0 = self:line() - for _index_0 = item_offset, #node do - local action = node[_index_0] - _with_0:append(chain_item(action)) - end - actions = _with_0 - end - return self:line(callee_value, actions) - end, - fndef = function(self, node) - local args, whitelist, arrow, block = unpack(node, 2) - local default_args = { } - local self_args = { } - local arg_names - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #args do - local arg = args[_index_0] - local name, default_value = unpack(arg) - if type(name) == "string" then - name = name - else - if name[1] == "self" or name[1] == "self_class" then - insert(self_args, name) - end - name = name[2] - end - if default_value then - insert(default_args, arg) - end - local _value_0 = name - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - end - arg_names = _accum_0 - end - if arrow == "fat" then - insert(arg_names, 1, "self") - end - do - local _with_0 = self:block() - if #whitelist > 0 then - _with_0:whitelist_names(whitelist) - end - for _index_0 = 1, #arg_names do - local name = arg_names[_index_0] - _with_0:put_name(name) - end - for _index_0 = 1, #default_args do - local default = default_args[_index_0] - local name, value = unpack(default) - if type(name) == "table" then - name = name[2] - end - _with_0:stm({ - 'if', - { - 'exp', - { - "ref", - name - }, - '==', - 'nil' - }, - { - { - 'assign', - { - name - }, - { - value - } - } - } - }) - end - local self_arg_values - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #self_args do - local arg = self_args[_index_0] - _accum_0[_len_0] = arg[2] - _len_0 = _len_0 + 1 - end - self_arg_values = _accum_0 - end - if #self_args > 0 then - _with_0:stm({ - "assign", - self_args, - self_arg_values - }) - end - _with_0:stms(block) - if #args > #arg_names then - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #args do - local arg = args[_index_0] - _accum_0[_len_0] = arg[1] - _len_0 = _len_0 + 1 - end - arg_names = _accum_0 - end - end - _with_0.header = "function(" .. concat(arg_names, ", ") .. ")" - return _with_0 - end - end, - table = function(self, node) - local items = unpack(node, 2) - do - local _with_0 = self:block("{", "}") - local format_line - format_line = function(tuple) - if #tuple == 2 then - local key, value = unpack(tuple) - if ntype(key) == "key_literal" and data.lua_keywords[key[2]] then - key = { - "string", - '"', - key[2] - } - end - local assign - if ntype(key) == "key_literal" then - assign = key[2] - else - assign = self:line("[", _with_0:value(key), "]") - end - local out = self:line(assign, " = ", _with_0:value(value)) - return out - else - return self:line(_with_0:value(tuple[1])) - end - end - if items then - local count = #items - for i, tuple in ipairs(items) do - local line = format_line(tuple) - if not (count == i) then - line:append(table_delim) - end - _with_0:add(line) - end - end - return _with_0 - end - end, - minus = function(self, node) - return self:line("-", self:value(node[2])) - end, - temp_name = function(self, node, ...) - return node:get_name(self, ...) - end, - number = function(self, node) - return node[2] - end, - bitnot = function(self, node) - return self:line("~", self:value(node[2])) - end, - length = function(self, node) - return self:line("#", self:value(node[2])) - end, - ["not"] = function(self, node) - return self:line("not ", self:value(node[2])) - end, - self = function(self, node) - return "self." .. self:name(node[2]) - end, - self_class = function(self, node) - return "self.__class." .. self:name(node[2]) - end, - self_colon = function(self, node) - return "self:" .. self:name(node[2]) - end, - self_class_colon = function(self, node) - return "self.__class:" .. self:name(node[2]) - end, - ref = function(self, value) - do - local sup = value[2] == "super" and self:get("super") - if sup then - return self:value(sup(self)) - end - end - return tostring(value[2]) - end, - raw_value = function(self, value) - if value == "..." then - self:send("varargs") - end - return tostring(value) - end - } - -end -package.preload['moonscript.compile.statement'] = function() - local ntype - ntype = require("moonscript.types").ntype - local concat, insert - do - local _obj_0 = table - concat, insert = _obj_0.concat, _obj_0.insert - end - local unpack - unpack = require("moonscript.util").unpack - return { - raw = function(self, node) - return self:add(node[2]) - end, - lines = function(self, node) - local _list_0 = node[2] - for _index_0 = 1, #_list_0 do - local line = _list_0[_index_0] - self:add(line) - end - end, - declare = function(self, node) - local names = node[2] - local undeclared = self:declare(names) - if #undeclared > 0 then - do - local _with_0 = self:line("local ") - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #undeclared do - local name = undeclared[_index_0] - _accum_0[_len_0] = self:name(name) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") - return _with_0 - end - end - end, - declare_with_shadows = function(self, node) - local names = node[2] - self:declare(names) - do - local _with_0 = self:line("local ") - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #names do - local name = names[_index_0] - _accum_0[_len_0] = self:name(name) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") - return _with_0 - end - end, - assign = function(self, node) - local names, values = unpack(node, 2) - local undeclared = self:declare(names) - local declare = "local " .. concat(undeclared, ", ") - local has_fndef = false - local i = 1 - while i <= #values do - if ntype(values[i]) == "fndef" then - has_fndef = true - end - i = i + 1 - end - do - local _with_0 = self:line() - if #undeclared == #names and not has_fndef then - _with_0:append(declare) - else - if #undeclared > 0 then - self:add(declare, node[-1]) - end - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #names do - local name = names[_index_0] - _accum_0[_len_0] = self:value(name) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") - end - _with_0:append(" = ") - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #values do - local v = values[_index_0] - _accum_0[_len_0] = self:value(v) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") - return _with_0 - end - end, - ["return"] = function(self, node) - return self:line("return ", (function() - if node[2] ~= "" then - return self:value(node[2]) - end - end)()) - end, - ["break"] = function(self, node) - return "break" - end, - ["if"] = function(self, node) - local cond, block = node[2], node[3] - local root - do - local _with_0 = self:block(self:line("if ", self:value(cond), " then")) - _with_0:stms(block) - root = _with_0 - end - local current = root - local add_clause - add_clause = function(clause) - local type = clause[1] - local i = 2 - local next - if type == "else" then - next = self:block("else") - else - i = i + 1 - next = self:block(self:line("elseif ", self:value(clause[2]), " then")) - end - next:stms(clause[i]) - current.next = next - current = next - end - for _index_0 = 4, #node do - local cond = node[_index_0] - add_clause(cond) - end - return root - end, - ["repeat"] = function(self, node) - local cond, block = unpack(node, 2) - do - local _with_0 = self:block("repeat", self:line("until ", self:value(cond))) - _with_0:stms(block) - return _with_0 - end - end, - ["while"] = function(self, node) - local cond, block = unpack(node, 2) - do - local _with_0 = self:block(self:line("while ", self:value(cond), " do")) - _with_0:stms(block) - return _with_0 - end - end, - ["for"] = function(self, node) - local name, bounds, block = unpack(node, 2) - local loop = self:line("for ", self:name(name), " = ", self:value({ - "explist", - unpack(bounds) - }), " do") - do - local _with_0 = self:block(loop) - _with_0:declare({ - name - }) - _with_0:stms(block) - return _with_0 - end - end, - foreach = function(self, node) - local names, exps, block = unpack(node, 2) - local loop - do - local _with_0 = self:line() - _with_0:append("for ") - loop = _with_0 - end - do - local _with_0 = self:block(loop) - loop:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #names do - local name = names[_index_0] - _accum_0[_len_0] = _with_0:name(name, false) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ", ") - loop:append(" in ") - loop:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #exps do - local exp = exps[_index_0] - _accum_0[_len_0] = self:value(exp) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(), ",") - loop:append(" do") - _with_0:declare(names) - _with_0:stms(block) - return _with_0 - end - end, - export = function(self, node) - local names = unpack(node, 2) - if type(names) == "string" then - if names == "*" then - self.export_all = true - elseif names == "^" then - self.export_proper = true - end - else - self:declare(names) - end - return nil - end, - run = function(self, code) - code:call(self) - return nil - end, - group = function(self, node) - return self:stms(node[2]) - end, - ["do"] = function(self, node) - do - local _with_0 = self:block() - _with_0:stms(node[2]) - return _with_0 - end - end, - noop = function(self) end - } - -end -package.preload['moonscript.types'] = function() - local util = require("moonscript.util") - local Set - Set = require("moonscript.data").Set - local insert - insert = table.insert - local unpack - unpack = util.unpack - local manual_return = Set({ - "foreach", - "for", - "while", - "return" - }) - local cascading = Set({ - "if", - "unless", - "with", - "switch", - "class", - "do" - }) - local terminating = Set({ - "return", - "break" - }) - local ntype - ntype = function(node) - local _exp_0 = type(node) - if "nil" == _exp_0 then - return "nil" - elseif "table" == _exp_0 then - return node[1] - else - return "value" - end - end - local mtype - do - local moon_type = util.moon.type - mtype = function(val) - local mt = getmetatable(val) - if mt and mt.smart_node then - return "table" - end - return moon_type(val) - end + return value end - local value_can_be_statement - value_can_be_statement = function(node) - if not (ntype(node) == "chain") then - return false - end - return ntype(node[#node]) == "call" + + local function pos(patt) + return (lpeg.Cp() * patt) / insert_pos end - local is_value - is_value = function(stm) - local compile = require("moonscript.compile") - local transform = require("moonscript.transform") - return compile.Block:is_value(stm) or transform.Value:can_transform(stm) + + local function got(what) + return Cmt("", function(str, pos, ...) + local cap = {...} + print("++ got "..what, "["..extract_line(str, pos).."]") + return true + end) end - local value_is_singular - value_is_singular = function(node) - return type(node) ~= "table" or node[1] ~= "exp" or #node == 2 + + local function flatten(tbl) + if #tbl == 1 then + return tbl[1] + end + return tbl end - local is_slice - is_slice = function(node) - return ntype(node) == "chain" and ntype(node[#node]) == "slice" + + local function flatten_or_mark(name) + return function(tbl) + if #tbl == 1 then return tbl[1] end + table.insert(tbl, 1, name) + return tbl + end end - local t = { } - local node_types = { - class = { - { - "name", - "Tmp" - }, - { - "body", - t - } - }, - fndef = { - { - "args", - t - }, - { - "whitelist", - t - }, - { - "arrow", - "slim" - }, - { - "body", - t - } - }, - foreach = { - { - "names", - t - }, - { - "iter" - }, - { - "body", - t - } - }, - ["for"] = { - { - "name" - }, - { - "bounds", - t - }, - { - "body", - t - } - }, - ["while"] = { - { - "cond", - t - }, - { - "body", - t - } - }, - assign = { - { - "names", - t - }, - { - "values", - t - } - }, - declare = { - { - "names", - t - } - }, - ["if"] = { - { - "cond", - t - }, - { - "then", - t - } - } - } - local build_table - build_table = function() - local key_table = { } - for node_name, args in pairs(node_types) do - local index = { } - for i, tuple in ipairs(args) do - local prop_name = tuple[1] - index[prop_name] = i + 1 - end - key_table[node_name] = index - end - return key_table + + -- makes sure the last item in a chain is an index + local _chain_assignable = { index = true, dot = true, slice = true } + + local function is_assignable(node) + if node == "..." then + return false + end + + local t = ntype(node) + return t == "ref" or t == "self" or t == "value" or t == "self_class" or + t == "chain" and _chain_assignable[ntype(node[#node])] or + t == "table" end - local key_table = build_table() - local make_builder - make_builder = function(name) - local spec = node_types[name] - if not spec then - error("don't know how to build node: " .. name) - end - return function(props) - if props == nil then - props = { } - end - local node = { - name - } - for i, arg in ipairs(spec) do - local key, default_value = unpack(arg) - local val - if props[key] then - val = props[key] - else - val = default_value - end - if val == t then - val = { } - end - node[i + 1] = val - end - return node - end + + local function check_assignable(str, pos, value) + if is_assignable(value) then + return true, value + end + return false end - local build = nil - build = setmetatable({ - group = function(body) - if body == nil then - body = { } - end - return { - "group", - body - } - end, - ["do"] = function(body) - return { - "do", - body - } - end, - assign_one = function(name, value) - return build.assign({ - names = { - name - }, - values = { - value - } - }) - end, - table = function(tbl) - if tbl == nil then - tbl = { } - end - for _index_0 = 1, #tbl do - local tuple = tbl[_index_0] - if type(tuple[1]) == "string" then - tuple[1] = { - "key_literal", - tuple[1] - } - end - end - return { - "table", - tbl - } - end, - block_exp = function(body) - return { - "block_exp", - body - } - end, - chain = function(parts) - local base = parts.base or error("expecting base property for chain") - if type(base) == "string" then - base = { - "ref", - base - } - end - local node = { - "chain", - base - } - for _index_0 = 1, #parts do - local part = parts[_index_0] - insert(node, part) - end - return node - end - }, { - __index = function(self, name) - self[name] = make_builder(name) - return rawget(self, name) - end - }) - local smart_node_mt = setmetatable({ }, { - __index = function(self, node_type) - local index = key_table[node_type] - local mt = { - smart_node = true, - __index = function(node, key) - if index[key] then - return rawget(node, index[key]) - elseif type(key) == "string" then - return error("unknown key: `" .. key .. "` on node type: `" .. ntype(node) .. "`") - end - end, - __newindex = function(node, key, value) - if index[key] then - key = index[key] - end - return rawset(node, key, value) - end - } - self[node_type] = mt - return mt - end - }) - local smart_node - smart_node = function(node) - return setmetatable(node, smart_node_mt[ntype(node)]) + + local flatten_explist = flatten_or_mark"explist" + local function format_assign(lhs_exps, assign) + if not assign then + return flatten_explist(lhs_exps) + end + + for _, assign_exp in ipairs(lhs_exps) do + if not is_assignable(assign_exp) then + error {assign_exp, "left hand expression is not assignable"} + end + end + + local t = ntype(assign) + if t == "assign" then + return {"assign", lhs_exps, unpack(assign, 2)} + elseif t == "update" then + return {"update", lhs_exps[1], unpack(assign, 2)} + end + + error "unknown assign expression" end - local NOOP = { - "noop" - } - return { - ntype = ntype, - smart_node = smart_node, - build = build, - is_value = is_value, - is_slice = is_slice, - manual_return = manual_return, - cascading = cascading, - value_is_singular = value_is_singular, - value_can_be_statement = value_can_be_statement, - mtype = mtype, - terminating = terminating, - NOOP = NOOP - } -end -package.preload['moonscript.version'] = function() - local version = "0.5.0" + -- the if statement only takes a single lhs, so we wrap in table to git to + -- "assign" tuple format + local function format_single_assign(lhs, assign) + if assign then + return format_assign({lhs}, assign) + end + return lhs + end + + local function sym(chars) + return Space * chars + end + + local function symx(chars) + return chars + end + + local function simple_string(delim, allow_interpolation) + local inner = P('\\'..delim) + "\\\\" + (1 - P(delim)) + if allow_interpolation then + inter = symx"#{" * V"Exp" * sym"}" + inner = (C((inner - inter)^1) + inter / mark"interpolate")^0 + else + inner = C(inner^0) + end + + return C(symx(delim)) * + inner * sym(delim) / mark"string" + end + + local function wrap_func_arg(value) + return {"call", {value}} + end + + -- DOCME + local function flatten_func(callee, args) + if #args == 0 then return callee end + + args = {"call", args} + if ntype(callee) == "chain" then + -- check for colon stub that needs arguments + if ntype(callee[#callee]) == "colon_stub" then + local stub = callee[#callee] + stub[1] = "colon" + table.insert(stub, args) + else + table.insert(callee, args) + end + + return callee + end + + return {"chain", callee, args} + end + + local function flatten_string_chain(str, chain, args) + if not chain then return str end + return flatten_func({"chain", str, unpack(chain)}, args) + end + + -- transforms a statement that has a line decorator + local function wrap_decorator(stm, dec) + if not dec then return stm end + return { "decorated", stm, dec } + end + + -- wrap if statement if there is a conditional decorator + local function wrap_if(stm, cond) + if cond then + local pass, fail = unpack(cond) + if fail then fail = {"else", {fail}} end + return {"if", cond[2], {stm}, fail} + end + return stm + end + + local function check_lua_string(str, pos, right, left) + return #left == #right + end + + -- :name in table literal + local function self_assign(name) + return {{"key_literal", name}, name} + end + + local err_msg = "Failed to parse:%s\n [%d] >> %s" + + local build_grammar = wrap_env(function() + local _indent = Stack(0) -- current indent + local _do_stack = Stack(0) + + local last_pos = 0 -- used to know where to report error + local function check_indent(str, pos, indent) + last_pos = pos + return _indent:top() == indent + end + + local function advance_indent(str, pos, indent) + local top = _indent:top() + if top ~= -1 and indent > _indent:top() then + _indent:push(indent) + return true + end + end + + local function push_indent(str, pos, indent) + _indent:push(indent) + return true + end + + local function pop_indent(str, pos) + if not _indent:pop() then error("unexpected outdent") end + return true + end + + + local function check_do(str, pos, do_node) + local top = _do_stack:top() + if top == nil or top then + return true, do_node + end + return false + end + + local function disable_do(str_pos) + _do_stack:push(false) + return true + end + + local function enable_do(str_pos) + _do_stack:push(true) + return true + end + + local function pop_do(str, pos) + if nil == _do_stack:pop() then error("unexpected do pop") end + return true + end + + local DisableDo = Cmt("", disable_do) + local EnableDo = Cmt("", enable_do) + local PopDo = Cmt("", pop_do) + + local keywords = {} + local function key(chars) + keywords[chars] = true + return Space * chars * -AlphaNum + end + + local function op(word) + local patt = Space * C(word) + if word:match("^%w*$") then + keywords[word] = true + patt = patt * -AlphaNum + end + return patt + end + + -- make sure name is not a keyword + local Name = Cmt(Name, function(str, pos, name) + if keywords[name] then return false end + return true + end) / trim + + local SelfName = Space * "@" * ( + "@" * (_Name / mark"self_class" + Cc"self.__class") + + _Name / mark"self" + Cc"self") + + local KeyName = SelfName + Space * _Name / mark"key_literal" + local VarArg = Space * P"..." / trim + + local g = lpeg.P{ + File, + File = Shebang^-1 * (Block + Ct""), + Block = Ct(Line * (Break^1 * Line)^0), + CheckIndent = Cmt(Indent, check_indent), -- validates line is in correct indent + Line = (CheckIndent * Statement + Space * #Stop), + + Statement = pos( + Import + While + With + For + ForEach + Switch + Return + + Local + Export + BreakLoop + + Ct(ExpList) * (Update + Assign)^-1 / format_assign + ) * Space * (( + -- statement decorators + key"if" * Exp * (key"else" * Exp)^-1 * Space / mark"if" + + key"unless" * Exp / mark"unless" + + CompInner / mark"comprehension" + ) * Space)^-1 / wrap_decorator, + + Body = Space^-1 * Break * EmptyLine^0 * InBlock + Ct(Statement), -- either a statement, or an indented block + + Advance = #Cmt(Indent, advance_indent), -- Advances the indent, gives back whitespace for CheckIndent + PushIndent = Cmt(Indent, push_indent), + PreventIndent = Cmt(Cc(-1), push_indent), + PopIndent = Cmt("", pop_indent), + InBlock = Advance * Block * PopIndent, + + Local = key"local" * ((op"*" + op"^") / mark"declare_glob" + Ct(NameList) / mark"declare_with_shadows"), + + Import = key"import" * Ct(ImportNameList) * SpaceBreak^0 * key"from" * Exp / mark"import", + ImportName = (sym"\\" * Ct(Cc"colon_stub" * Name) + Name), + ImportNameList = SpaceBreak^0 * ImportName * ((SpaceBreak^1 + sym"," * SpaceBreak^0) * ImportName)^0, + + BreakLoop = Ct(key"break"/trim) + Ct(key"continue"/trim), + + Return = key"return" * (ExpListLow/mark"explist" + C"") / mark"return", + + WithExp = Ct(ExpList) * Assign^-1 / format_assign, + With = key"with" * DisableDo * ensure(WithExp, PopDo) * key"do"^-1 * Body / mark"with", + + Switch = key"switch" * DisableDo * ensure(Exp, PopDo) * key"do"^-1 * Space^-1 * Break * SwitchBlock / mark"switch", + + SwitchBlock = EmptyLine^0 * Advance * Ct(SwitchCase * (Break^1 * SwitchCase)^0 * (Break^1 * SwitchElse)^-1) * PopIndent, + SwitchCase = key"when" * Ct(ExpList) * key"then"^-1 * Body / mark"case", + SwitchElse = key"else" * Body / mark"else", + + IfCond = Exp * Assign^-1 / format_single_assign, + + If = key"if" * IfCond * key"then"^-1 * Body * + ((Break * CheckIndent)^-1 * EmptyLine^0 * key"elseif" * pos(IfCond) * key"then"^-1 * Body / mark"elseif")^0 * + ((Break * CheckIndent)^-1 * EmptyLine^0 * key"else" * Body / mark"else")^-1 / mark"if", + + Unless = key"unless" * IfCond * key"then"^-1 * Body * + ((Break * CheckIndent)^-1 * EmptyLine^0 * key"else" * Body / mark"else")^-1 / mark"unless", + + While = key"while" * DisableDo * ensure(Exp, PopDo) * key"do"^-1 * Body / mark"while", + + For = key"for" * DisableDo * ensure(Name * sym"=" * Ct(Exp * sym"," * Exp * (sym"," * Exp)^-1), PopDo) * + key"do"^-1 * Body / mark"for", + + ForEach = key"for" * Ct(AssignableNameList) * key"in" * DisableDo * ensure(Ct(sym"*" * Exp / mark"unpack" + ExpList), PopDo) * key"do"^-1 * Body / mark"foreach", + + Do = key"do" * Body / mark"do", + + Comprehension = sym"[" * Exp * CompInner * sym"]" / mark"comprehension", + + TblComprehension = sym"{" * Ct(Exp * (sym"," * Exp)^-1) * CompInner * sym"}" / mark"tblcomprehension", + + CompInner = Ct((CompForEach + CompFor) * CompClause^0), + CompForEach = key"for" * Ct(NameList) * key"in" * (sym"*" * Exp / mark"unpack" + Exp) / mark"foreach", + CompFor = key "for" * Name * sym"=" * Ct(Exp * sym"," * Exp * (sym"," * Exp)^-1) / mark"for", + CompClause = CompFor + CompForEach + key"when" * Exp / mark"when", + + Assign = sym"=" * (Ct(With + If + Switch) + Ct(TableBlock + ExpListLow)) / mark"assign", + Update = ((sym"..=" + sym"+=" + sym"-=" + sym"*=" + sym"/=" + sym"%=" + sym"or=" + sym"and=") / trim) * Exp / mark"update", + + -- we can ignore precedence for now + OtherOps = op"or" + op"and" + op"<=" + op">=" + op"~=" + op"!=" + op"==" + op".." + op"<" + op">", + + Assignable = Cmt(DotChain + Chain, check_assignable) + Name + SelfName, + + Exp = Ct(Value * ((OtherOps + FactorOp + TermOp) * Value)^0) / flatten_or_mark"exp", + + -- Exp = Ct(Factor * (OtherOps * Factor)^0) / flatten_or_mark"exp", + -- Factor = Ct(Term * (FactorOp * Term)^0) / flatten_or_mark"exp", + -- Term = Ct(Value * (TermOp * Value)^0) / flatten_or_mark"exp", + + SimpleValue = + If + Unless + + Switch + + With + + ClassDecl + + ForEach + For + While + + Cmt(Do, check_do) + + sym"-" * -SomeSpace * Exp / mark"minus" + + sym"#" * Exp / mark"length" + + key"not" * Exp / mark"not" + + TblComprehension + + TableLit + + Comprehension + + FunLit + + Num, + + ChainValue = -- a function call or an object access + StringChain + + ((Chain + DotChain + Callable) * Ct(InvokeArgs^-1)) / flatten_func, + + Value = pos( + SimpleValue + + Ct(KeyValueList) / mark"table" + + ChainValue), + + SliceValue = SimpleValue + ChainValue, + + StringChain = String * + (Ct((ColonCall + ColonSuffix) * ChainTail^-1) * Ct(InvokeArgs^-1))^-1 / flatten_string_chain, + + String = Space * DoubleString + Space * SingleString + LuaString, + SingleString = simple_string("'"), + DoubleString = simple_string('"', true), + + LuaString = Cg(LuaStringOpen, "string_open") * Cb"string_open" * Break^-1 * + C((1 - Cmt(C(LuaStringClose) * Cb"string_open", check_lua_string))^0) * + LuaStringClose / mark"string", + + LuaStringOpen = sym"[" * P"="^0 * "[" / trim, + LuaStringClose = "]" * P"="^0 * "]", + + Callable = pos(Name / mark"ref") + SelfName + VarArg + Parens / mark"parens", + Parens = sym"(" * Exp * sym")", + + FnArgs = symx"(" * Ct(ExpList^-1) * sym")" + sym"!" * -P"=" * Ct"", + + ChainTail = ChainItem^1 * ColonSuffix^-1 + ColonSuffix, + + -- a list of funcalls and indexes on a callable + Chain = Callable * ChainTail / mark"chain", + + -- shorthand dot call for use in with statement + DotChain = + (sym"." * Cc(-1) * (_Name / mark"dot") * ChainTail^-1) / mark"chain" + + (sym"\\" * Cc(-1) * ( + (_Name * Invoke / mark"colon") * ChainTail^-1 + + (_Name / mark"colon_stub") + )) / mark"chain", + + ChainItem = + Invoke + + Slice + + symx"[" * Exp/mark"index" * sym"]" + + symx"." * _Name/mark"dot" + + ColonCall, + + Slice = symx"[" * (SliceValue + Cc(1)) * sym"," * (SliceValue + Cc"") * + (sym"," * SliceValue)^-1 *sym"]" / mark"slice", + + ColonCall = symx"\\" * (_Name * Invoke) / mark"colon", + ColonSuffix = symx"\\" * _Name / mark"colon_stub", + + Invoke = FnArgs/mark"call" + + SingleString / wrap_func_arg + + DoubleString / wrap_func_arg, + + TableValue = KeyValue + Ct(Exp), + + TableLit = sym"{" * Ct( + TableValueList^-1 * sym","^-1 * + (SpaceBreak * TableLitLine * (sym","^-1 * SpaceBreak * TableLitLine)^0 * sym","^-1)^-1 + ) * White * sym"}" / mark"table", + + TableValueList = TableValue * (sym"," * TableValue)^0, + TableLitLine = PushIndent * ((TableValueList * PopIndent) + (PopIndent * Cut)) + Space, + + -- the unbounded table + TableBlockInner = Ct(KeyValueLine * (SpaceBreak^1 * KeyValueLine)^0), + TableBlock = SpaceBreak^1 * Advance * ensure(TableBlockInner, PopIndent) / mark"table", + + ClassDecl = key"class" * -P":" * (Assignable + Cc(nil)) * (key"extends" * PreventIndent * ensure(Exp, PopIndent) + C"")^-1 * (ClassBlock + Ct("")) / mark"class", + + ClassBlock = SpaceBreak^1 * Advance * + Ct(ClassLine * (SpaceBreak^1 * ClassLine)^0) * PopIndent, + ClassLine = CheckIndent * (( + KeyValueList / mark"props" + + Statement / mark"stm" + + Exp / mark"stm" + ) * sym","^-1), + + Export = key"export" * ( + Cc"class" * ClassDecl + + op"*" + op"^" + + Ct(NameList) * (sym"=" * Ct(ExpListLow))^-1) / mark"export", + + KeyValue = (sym":" * -SomeSpace * Name) / self_assign + Ct((KeyName + sym"[" * Exp * sym"]" + DoubleString + SingleString) * symx":" * (Exp + TableBlock)), + KeyValueList = KeyValue * (sym"," * KeyValue)^0, + KeyValueLine = CheckIndent * KeyValueList * sym","^-1, + + FnArgsDef = sym"(" * Ct(FnArgDefList^-1) * + (key"using" * Ct(NameList + Space * "nil") + Ct"") * + sym")" + Ct"" * Ct"", + + FnArgDefList = FnArgDef * (sym"," * FnArgDef)^0 * (sym"," * Ct(VarArg))^0 + Ct(VarArg), + FnArgDef = Ct((Name + SelfName) * (sym"=" * Exp)^-1), + + FunLit = FnArgsDef * + (sym"->" * Cc"slim" + sym"=>" * Cc"fat") * + (Body + Ct"") / mark"fndef", + + NameList = Name * (sym"," * Name)^0, + NameOrDestructure = Name + TableLit, + AssignableNameList = NameOrDestructure * (sym"," * NameOrDestructure)^0, + + ExpList = Exp * (sym"," * Exp)^0, + ExpListLow = Exp * ((sym"," + sym";") * Exp)^0, + + InvokeArgs = -P"-" * (ExpList * (sym"," * (TableBlock + SpaceBreak * Advance * ArgBlock * TableBlock^-1) + TableBlock)^-1 + TableBlock), + ArgBlock = ArgLine * (sym"," * SpaceBreak * ArgLine)^0 * PopIndent, + ArgLine = CheckIndent * ExpList + } + + return { + _g = White * g * White * -1, + match = function(self, str, ...) + + local pos_to_line = function(pos) + return util.pos_to_line(str, pos) + end + + local get_line = function(num) + return util.get_line(str, num) + end + + local tree + local parse_args = {...} + + local pass, err = xpcall(function() + tree = self._g:match(str, unpack(parse_args)) + end, function(err) + return debug.traceback(err, 2) + end) + + -- regular error, let it bubble up + if type(err) == "string" then + return nil, err + end + + if not tree then + local pos = last_pos + local msg + + if err then + local node + node, msg = unpack(err) + msg = msg and " " .. msg + pos = node[-1] + end + + local line_no = pos_to_line(pos) + local line_str = get_line(line_no) or "" + + return nil, err_msg:format(msg or "", line_no, trim(line_str)) + end + return tree + end + } + end) + return { - version = version, - print_version = function() - return print("MoonScript version " .. tostring(version)) - end + extract_line = extract_line, + + -- parse a string + -- returns tree, or nil and error message + string = function (str) + local g = build_grammar() + return g:match(str) + end } + end -package.preload['moonscript.errors'] = function() - local util = require("moonscript.util") - local lpeg = require("lpeg") - local concat, insert +package.preload['moonscript.transform.destructure'] = function() + local ntype, mtype, build + do + local _obj_0 = require("moonscript.types") + ntype, mtype, build = _obj_0.ntype, _obj_0.mtype, _obj_0.build + end + local NameProxy + do + local _obj_0 = require("moonscript.transform.names") + NameProxy = _obj_0.NameProxy + end + local insert do local _obj_0 = table - concat, insert = _obj_0.concat, _obj_0.insert + insert = _obj_0.insert + end + local unpack + do + local _obj_0 = require("moonscript.util") + unpack = _obj_0.unpack end - local split, pos_to_line - split, pos_to_line = util.split, util.pos_to_line local user_error - user_error = function(...) - return error({ - "user-error", - ... - }) + do + local _obj_0 = require("moonscript.errors") + user_error = _obj_0.user_error end - local lookup_line - lookup_line = function(fname, pos, cache) - if not cache[fname] then - do - local _with_0 = assert(io.open(fname)) - cache[fname] = _with_0:read("*a") - _with_0:close() + local util = require("moonscript.util") + local join + join = function(...) + do + local out = { } + local i = 1 + local _list_0 = { + ... + } + for _index_0 = 1, #_list_0 do + local tbl = _list_0[_index_0] + for _index_1 = 1, #tbl do + local v = tbl[_index_1] + out[i] = v + i = i + 1 + end end + return out end - return pos_to_line(cache[fname], pos) end - local reverse_line_number - reverse_line_number = function(fname, line_table, line_num, cache) - for i = line_num, 0, -1 do - if line_table[i] then - return lookup_line(fname, line_table[i], cache) + local has_destructure + has_destructure = function(names) + for _index_0 = 1, #names do + local n = names[_index_0] + if ntype(n) == "table" then + return true end end - return "unknown" + return false end - local truncate_traceback - truncate_traceback = function(traceback, chunk_func) - if chunk_func == nil then - chunk_func = "moonscript_chunk" + local extract_assign_names + extract_assign_names = function(name, accum, prefix) + if accum == nil then + accum = { } end - traceback = split(traceback, "\n") - local stop = #traceback - while stop > 1 do - if traceback[stop]:match(chunk_func) then - break - end - stop = stop - 1 + if prefix == nil then + prefix = { } end - do - local _accum_0 = { } - local _len_0 = 1 - local _max_0 = stop - for _index_0 = 1, _max_0 < 0 and #traceback + _max_0 or _max_0 do - local t = traceback[_index_0] - _accum_0[_len_0] = t - _len_0 = _len_0 + 1 + local i = 1 + local _list_0 = name[2] + for _index_0 = 1, #_list_0 do + local tuple = _list_0[_index_0] + local value, suffix + if #tuple == 1 then + local s = { + "index", + { + "number", + i + } + } + i = i + 1 + value, suffix = tuple[1], s + else + local key = tuple[1] + local s + if ntype(key) == "key_literal" then + local key_name = key[2] + if ntype(key_name) == "colon_stub" then + s = key_name + else + s = { + "dot", + key_name + } + end + else + s = { + "index", + key + } + end + value, suffix = tuple[2], s + end + suffix = join(prefix, { + suffix + }) + local _exp_0 = ntype(value) + if "value" == _exp_0 or "ref" == _exp_0 or "chain" == _exp_0 or "self" == _exp_0 then + insert(accum, { + value, + suffix + }) + elseif "table" == _exp_0 then + extract_assign_names(value, accum, suffix) + else + user_error("Can't destructure value of type: " .. tostring(ntype(value))) end - traceback = _accum_0 end - local rep = "function '" .. chunk_func .. "'" - traceback[#traceback] = traceback[#traceback]:gsub(rep, "main chunk") - return concat(traceback, "\n") + return accum end - local rewrite_traceback - rewrite_traceback = function(text, err) - local line_tables = require("moonscript.line_tables") - local V, S, Ct, C - V, S, Ct, C = lpeg.V, lpeg.S, lpeg.Ct, lpeg.C - local header_text = "stack traceback:" - local Header, Line = V("Header"), V("Line") - local Break = lpeg.S("\n") - local g = lpeg.P({ - Header, - Header = header_text * Break * Ct(Line ^ 1), - Line = "\t" * C((1 - Break) ^ 0) * (Break + -1) - }) - local cache = { } - local rewrite_single - rewrite_single = function(trace) - local fname, line, msg = trace:match('^(.-):(%d+): (.*)$') - local tbl = line_tables["@" .. tostring(fname)] - if fname and tbl then - return concat({ - fname, - ":", - reverse_line_number(fname, tbl, line, cache), - ": ", - "(", - line, - ") ", - msg + local build_assign + build_assign = function(scope, destruct_literal, receiver) + local extracted_names = extract_assign_names(destruct_literal) + local names = { } + local values = { } + local inner = { + "assign", + names, + values + } + local obj + if scope:is_local(receiver) then + obj = receiver + else + do + obj = NameProxy("obj") + inner = build["do"]({ + build.assign_one(obj, receiver), + { + "assign", + names, + values + } }) + obj = obj + end + end + for _index_0 = 1, #extracted_names do + local tuple = extracted_names[_index_0] + insert(names, tuple[1]) + insert(values, NameProxy.chain(obj, unpack(tuple[2]))) + end + return build.group({ + { + "declare", + names + }, + inner + }) + end + local split_assign + split_assign = function(scope, assign) + local names, values = unpack(assign, 2) + local g = { } + local total_names = #names + local total_values = #values + local start = 1 + for i, n in ipairs(names) do + if ntype(n) == "table" then + if i > start then + local stop = i - 1 + insert(g, { + "assign", + (function() + local _accum_0 = { } + local _len_0 = 1 + for i = start, stop do + _accum_0[_len_0] = names[i] + _len_0 = _len_0 + 1 + end + return _accum_0 + end)(), + (function() + local _accum_0 = { } + local _len_0 = 1 + for i = start, stop do + _accum_0[_len_0] = values[i] + _len_0 = _len_0 + 1 + end + return _accum_0 + end)() + }) + end + insert(g, build_assign(scope, n, values[i])) + start = i + 1 + end + end + if total_names >= start or total_values >= start then + local name_slice + if total_names < start then + name_slice = { + "_" + } else - return trace + do + local _accum_0 = { } + local _len_0 = 1 + for i = start, total_names do + _accum_0[_len_0] = names[i] + _len_0 = _len_0 + 1 + end + name_slice = _accum_0 + end end + local value_slice + if total_values < start then + value_slice = { + "nil" + } + else + do + local _accum_0 = { } + local _len_0 = 1 + for i = start, total_values do + _accum_0[_len_0] = values[i] + _len_0 = _len_0 + 1 + end + value_slice = _accum_0 + end + end + insert(g, { + "assign", + name_slice, + value_slice + }) end - err = rewrite_single(err) - local match = g:match(text) - if not (match) then - return nil - end - for i, trace in ipairs(match) do - match[i] = rewrite_single(trace) - end - return concat({ - "moon: " .. err, - header_text, - "\t" .. concat(match, "\n\t") - }, "\n") + return build.group(g) end return { - rewrite_traceback = rewrite_traceback, - truncate_traceback = truncate_traceback, - user_error = user_error, - reverse_line_number = reverse_line_number + has_destructure = has_destructure, + split_assign = split_assign, + build_assign = build_assign } end -package.preload['moonscript.parse'] = function() - local debug_grammar = false - local lpeg = require("lpeg") - lpeg.setmaxstack(10000) - local err_msg = "Failed to parse:%s\n [%d] >> %s" - local Stack - Stack = require("moonscript.data").Stack - local trim, pos_to_line, get_line +package.preload['moonscript.transform.names'] = function() + local build do - local _obj_0 = require("moonscript.util") - trim, pos_to_line, get_line = _obj_0.trim, _obj_0.pos_to_line, _obj_0.get_line + local _obj_0 = require("moonscript.types") + build = _obj_0.build end local unpack - unpack = require("moonscript.util").unpack - local wrap_env - wrap_env = require("moonscript.parse.env").wrap_env - local R, S, V, P, C, Ct, Cmt, Cg, Cb, Cc - R, S, V, P, C, Ct, Cmt, Cg, Cb, Cc = lpeg.R, lpeg.S, lpeg.V, lpeg.P, lpeg.C, lpeg.Ct, lpeg.Cmt, lpeg.Cg, lpeg.Cb, lpeg.Cc - local White, Break, Stop, Comment, Space, SomeSpace, SpaceBreak, EmptyLine, AlphaNum, Num, Shebang, L, _Name do - local _obj_0 = require("moonscript.parse.literals") - White, Break, Stop, Comment, Space, SomeSpace, SpaceBreak, EmptyLine, AlphaNum, Num, Shebang, L, _Name = _obj_0.White, _obj_0.Break, _obj_0.Stop, _obj_0.Comment, _obj_0.Space, _obj_0.SomeSpace, _obj_0.SpaceBreak, _obj_0.EmptyLine, _obj_0.AlphaNum, _obj_0.Num, _obj_0.Shebang, _obj_0.L, _obj_0.Name + local _obj_0 = require("moonscript.util") + unpack = _obj_0.unpack end - local SpaceName = Space * _Name - Num = Space * (Num / function(v) - return { - "number", - v - } - end) - local Indent, Cut, ensure, extract_line, mark, pos, flatten_or_mark, is_assignable, check_assignable, format_assign, format_single_assign, sym, symx, simple_string, wrap_func_arg, join_chain, wrap_decorator, check_lua_string, self_assign, got + local LocalName do - local _obj_0 = require("moonscript.parse.util") - Indent, Cut, ensure, extract_line, mark, pos, flatten_or_mark, is_assignable, check_assignable, format_assign, format_single_assign, sym, symx, simple_string, wrap_func_arg, join_chain, wrap_decorator, check_lua_string, self_assign, got = _obj_0.Indent, _obj_0.Cut, _obj_0.ensure, _obj_0.extract_line, _obj_0.mark, _obj_0.pos, _obj_0.flatten_or_mark, _obj_0.is_assignable, _obj_0.check_assignable, _obj_0.format_assign, _obj_0.format_single_assign, _obj_0.sym, _obj_0.symx, _obj_0.simple_string, _obj_0.wrap_func_arg, _obj_0.join_chain, _obj_0.wrap_decorator, _obj_0.check_lua_string, _obj_0.self_assign, _obj_0.got - end - local build_grammar = wrap_env(debug_grammar, function(root) - local _indent = Stack(0) - local _do_stack = Stack(0) - local state = { - last_pos = 0 - } - local check_indent - check_indent = function(str, pos, indent) - state.last_pos = pos - return _indent:top() == indent - end - local advance_indent - advance_indent = function(str, pos, indent) - local top = _indent:top() - if top ~= -1 and indent > top then - _indent:push(indent) - return true - end - end - local push_indent - push_indent = function(str, pos, indent) - _indent:push(indent) - return true - end - local pop_indent - pop_indent = function() - assert(_indent:pop(), "unexpected outdent") - return true - end - local check_do - check_do = function(str, pos, do_node) - local top = _do_stack:top() - if top == nil or top then - return true, do_node + local _base_0 = { + get_name = function(self) + return self.name end - return false - end - local disable_do - disable_do = function() - _do_stack:push(false) - return true - end - local pop_do - pop_do = function() - assert(_do_stack:pop() ~= nil, "unexpected do pop") - return true - end - local DisableDo = Cmt("", disable_do) - local PopDo = Cmt("", pop_do) - local keywords = { } - local key - key = function(chars) - keywords[chars] = true - return Space * chars * -AlphaNum - end - local op - op = function(chars) - local patt = Space * C(chars) - if chars:match("^%w*$") then - keywords[chars] = true - patt = patt * -AlphaNum - end - return patt - end - local Name = Cmt(SpaceName, function(str, pos, name) - if keywords[name] then - return false + } + _base_0.__index = _base_0 + local _class_0 = setmetatable({ + __init = function(self, name) + self.name = name + self[1] = "temp_name" + end, + __base = _base_0, + __name = "LocalName" + }, { + __index = _base_0, + __call = function(cls, ...) + local _self_0 = setmetatable({}, _base_0) + cls.__init(_self_0, ...) + return _self_0 end - return true - end) / trim - local SelfName = Space * "@" * ("@" * (_Name / mark("self_class") + Cc("self.__class")) + _Name / mark("self") + Cc("self")) - local KeyName = SelfName + Space * _Name / mark("key_literal") - local VarArg = Space * P("...") / trim - local g = P({ - root or File, - File = Shebang ^ -1 * (Block + Ct("")), - Block = Ct(Line * (Break ^ 1 * Line) ^ 0), - CheckIndent = Cmt(Indent, check_indent), - Line = (CheckIndent * Statement + Space * L(Stop)), - Statement = pos(Import + While + With + For + ForEach + Switch + Return + Local + Export + BreakLoop + Ct(ExpList) * (Update + Assign) ^ -1 / format_assign) * Space * ((key("if") * Exp * (key("else") * Exp) ^ -1 * Space / mark("if") + key("unless") * Exp / mark("unless") + CompInner / mark("comprehension")) * Space) ^ -1 / wrap_decorator, - Body = Space ^ -1 * Break * EmptyLine ^ 0 * InBlock + Ct(Statement), - Advance = L(Cmt(Indent, advance_indent)), - PushIndent = Cmt(Indent, push_indent), - PreventIndent = Cmt(Cc(-1), push_indent), - PopIndent = Cmt("", pop_indent), - InBlock = Advance * Block * PopIndent, - Local = key("local") * ((op("*") + op("^")) / mark("declare_glob") + Ct(NameList) / mark("declare_with_shadows")), - Import = key("import") * Ct(ImportNameList) * SpaceBreak ^ 0 * key("from") * Exp / mark("import"), - ImportName = (sym("\\") * Ct(Cc("colon") * Name) + Name), - ImportNameList = SpaceBreak ^ 0 * ImportName * ((SpaceBreak ^ 1 + sym(",") * SpaceBreak ^ 0) * ImportName) ^ 0, - BreakLoop = Ct(key("break") / trim) + Ct(key("continue") / trim), - Return = key("return") * (ExpListLow / mark("explist") + C("")) / mark("return"), - WithExp = Ct(ExpList) * Assign ^ -1 / format_assign, - With = key("with") * DisableDo * ensure(WithExp, PopDo) * key("do") ^ -1 * Body / mark("with"), - Switch = key("switch") * DisableDo * ensure(Exp, PopDo) * key("do") ^ -1 * Space ^ -1 * Break * SwitchBlock / mark("switch"), - SwitchBlock = EmptyLine ^ 0 * Advance * Ct(SwitchCase * (Break ^ 1 * SwitchCase) ^ 0 * (Break ^ 1 * SwitchElse) ^ -1) * PopIndent, - SwitchCase = key("when") * Ct(ExpList) * key("then") ^ -1 * Body / mark("case"), - SwitchElse = key("else") * Body / mark("else"), - IfCond = Exp * Assign ^ -1 / format_single_assign, - IfElse = (Break * EmptyLine ^ 0 * CheckIndent) ^ -1 * key("else") * Body / mark("else"), - IfElseIf = (Break * EmptyLine ^ 0 * CheckIndent) ^ -1 * key("elseif") * pos(IfCond) * key("then") ^ -1 * Body / mark("elseif"), - If = key("if") * IfCond * key("then") ^ -1 * Body * IfElseIf ^ 0 * IfElse ^ -1 / mark("if"), - Unless = key("unless") * IfCond * key("then") ^ -1 * Body * IfElseIf ^ 0 * IfElse ^ -1 / mark("unless"), - While = key("while") * DisableDo * ensure(Exp, PopDo) * key("do") ^ -1 * Body / mark("while"), - For = key("for") * DisableDo * ensure(Name * sym("=") * Ct(Exp * sym(",") * Exp * (sym(",") * Exp) ^ -1), PopDo) * key("do") ^ -1 * Body / mark("for"), - ForEach = key("for") * Ct(AssignableNameList) * key("in") * DisableDo * ensure(Ct(sym("*") * Exp / mark("unpack") + ExpList), PopDo) * key("do") ^ -1 * Body / mark("foreach"), - Do = key("do") * Body / mark("do"), - Comprehension = sym("[") * Exp * CompInner * sym("]") / mark("comprehension"), - TblComprehension = sym("{") * Ct(Exp * (sym(",") * Exp) ^ -1) * CompInner * sym("}") / mark("tblcomprehension"), - CompInner = Ct((CompForEach + CompFor) * CompClause ^ 0), - CompForEach = key("for") * Ct(AssignableNameList) * key("in") * (sym("*") * Exp / mark("unpack") + Exp) / mark("foreach"), - CompFor = key("for" * Name * sym("=") * Ct(Exp * sym(",") * Exp * (sym(",") * Exp) ^ -1) / mark("for")), - CompClause = CompFor + CompForEach + key("when") * Exp / mark("when"), - Assign = sym("=") * (Ct(With + If + Switch) + Ct(TableBlock + ExpListLow)) / mark("assign"), - Update = ((sym("..=") + sym("+=") + sym("-=") + sym("*=") + sym("/=") + sym("%=") + sym("or=") + sym("and=") + sym("&=") + sym("|=") + sym(">>=") + sym("<<=")) / trim) * Exp / mark("update"), - CharOperators = Space * C(S("+-*/%^><|&")), - WordOperators = op("or") + op("and") + op("<=") + op(">=") + op("~=") + op("!=") + op("==") + op("..") + op("<<") + op(">>") + op("//"), - BinaryOperator = (WordOperators + CharOperators) * SpaceBreak ^ 0, - Assignable = Cmt(Chain, check_assignable) + Name + SelfName, - Exp = Ct(Value * (BinaryOperator * Value) ^ 0) / flatten_or_mark("exp"), - SimpleValue = If + Unless + Switch + With + ClassDecl + ForEach + For + While + Cmt(Do, check_do) + sym("-") * -SomeSpace * Exp / mark("minus") + sym("#") * Exp / mark("length") + sym("~") * Exp / mark("bitnot") + key("not") * Exp / mark("not") + TblComprehension + TableLit + Comprehension + FunLit + Num, - ChainValue = (Chain + Callable) * Ct(InvokeArgs ^ -1) / join_chain, - Value = pos(SimpleValue + Ct(KeyValueList) / mark("table") + ChainValue + String), - SliceValue = Exp, - String = Space * DoubleString + Space * SingleString + LuaString, - SingleString = simple_string("'"), - DoubleString = simple_string('"', true), - LuaString = Cg(LuaStringOpen, "string_open") * Cb("string_open") * Break ^ -1 * C((1 - Cmt(C(LuaStringClose) * Cb("string_open"), check_lua_string)) ^ 0) * LuaStringClose / mark("string"), - LuaStringOpen = sym("[") * P("=") ^ 0 * "[" / trim, - LuaStringClose = "]" * P("=") ^ 0 * "]", - Callable = pos(Name / mark("ref")) + SelfName + VarArg + Parens / mark("parens"), - Parens = sym("(") * SpaceBreak ^ 0 * Exp * SpaceBreak ^ 0 * sym(")"), - FnArgs = symx("(") * SpaceBreak ^ 0 * Ct(FnArgsExpList ^ -1) * SpaceBreak ^ 0 * sym(")") + sym("!") * -P("=") * Ct(""), - FnArgsExpList = Exp * ((Break + sym(",")) * White * Exp) ^ 0, - Chain = (Callable + String + -S(".\\")) * ChainItems / mark("chain") + Space * (DotChainItem * ChainItems ^ -1 + ColonChain) / mark("chain"), - ChainItems = ChainItem ^ 1 * ColonChain ^ -1 + ColonChain, - ChainItem = Invoke + DotChainItem + Slice + symx("[") * Exp / mark("index") * sym("]"), - DotChainItem = symx(".") * _Name / mark("dot"), - ColonChainItem = symx("\\") * _Name / mark("colon"), - ColonChain = ColonChainItem * (Invoke * ChainItems ^ -1) ^ -1, - Slice = symx("[") * (SliceValue + Cc(1)) * sym(",") * (SliceValue + Cc("")) * (sym(",") * SliceValue) ^ -1 * sym("]") / mark("slice"), - Invoke = FnArgs / mark("call") + SingleString / wrap_func_arg + DoubleString / wrap_func_arg + L(P("[")) * LuaString / wrap_func_arg, - TableValue = KeyValue + Ct(Exp), - TableLit = sym("{") * Ct(TableValueList ^ -1 * sym(",") ^ -1 * (SpaceBreak * TableLitLine * (sym(",") ^ -1 * SpaceBreak * TableLitLine) ^ 0 * sym(",") ^ -1) ^ -1) * White * sym("}") / mark("table"), - TableValueList = TableValue * (sym(",") * TableValue) ^ 0, - TableLitLine = PushIndent * ((TableValueList * PopIndent) + (PopIndent * Cut)) + Space, - TableBlockInner = Ct(KeyValueLine * (SpaceBreak ^ 1 * KeyValueLine) ^ 0), - TableBlock = SpaceBreak ^ 1 * Advance * ensure(TableBlockInner, PopIndent) / mark("table"), - ClassDecl = key("class") * -P(":") * (Assignable + Cc(nil)) * (key("extends") * PreventIndent * ensure(Exp, PopIndent) + C("")) ^ -1 * (ClassBlock + Ct("")) / mark("class"), - ClassBlock = SpaceBreak ^ 1 * Advance * Ct(ClassLine * (SpaceBreak ^ 1 * ClassLine) ^ 0) * PopIndent, - ClassLine = CheckIndent * ((KeyValueList / mark("props") + Statement / mark("stm") + Exp / mark("stm")) * sym(",") ^ -1), - Export = key("export") * (Cc("class") * ClassDecl + op("*") + op("^") + Ct(NameList) * (sym("=") * Ct(ExpListLow)) ^ -1) / mark("export"), - KeyValue = (sym(":") * -SomeSpace * Name * lpeg.Cp()) / self_assign + Ct((KeyName + sym("[") * Exp * sym("]") + Space * DoubleString + Space * SingleString) * symx(":") * (Exp + TableBlock + SpaceBreak ^ 1 * Exp)), - KeyValueList = KeyValue * (sym(",") * KeyValue) ^ 0, - KeyValueLine = CheckIndent * KeyValueList * sym(",") ^ -1, - FnArgsDef = sym("(") * White * Ct(FnArgDefList ^ -1) * (key("using") * Ct(NameList + Space * "nil") + Ct("")) * White * sym(")") + Ct("") * Ct(""), - FnArgDefList = FnArgDef * ((sym(",") + Break) * White * FnArgDef) ^ 0 * ((sym(",") + Break) * White * Ct(VarArg)) ^ 0 + Ct(VarArg), - FnArgDef = Ct((Name + SelfName) * (sym("=") * Exp) ^ -1), - FunLit = FnArgsDef * (sym("->") * Cc("slim") + sym("=>") * Cc("fat")) * (Body + Ct("")) / mark("fndef"), - NameList = Name * (sym(",") * Name) ^ 0, - NameOrDestructure = Name + TableLit, - AssignableNameList = NameOrDestructure * (sym(",") * NameOrDestructure) ^ 0, - ExpList = Exp * (sym(",") * Exp) ^ 0, - ExpListLow = Exp * ((sym(",") + sym(";")) * Exp) ^ 0, - InvokeArgs = -P("-") * (ExpList * (sym(",") * (TableBlock + SpaceBreak * Advance * ArgBlock * TableBlock ^ -1) + TableBlock) ^ -1 + TableBlock), - ArgBlock = ArgLine * (sym(",") * SpaceBreak * ArgLine) ^ 0 * PopIndent, - ArgLine = CheckIndent * ExpList }) - return g, state - end) - local file_parser - file_parser = function() - local g, state = build_grammar() - local file_grammar = White * g * White * -1 - return { - match = function(self, str) - local tree - local _, err = xpcall((function() - tree = file_grammar:match(str) - end), function(err) - return debug.traceback(err, 2) - end) - if type(err) == "string" then - return nil, err - end - if not (tree) then - local msg - local err_pos = state.last_pos - if err then - local node - node, msg = unpack(err) - if msg then - msg = " " .. msg - end - err_pos = node[-1] - end - local line_no = pos_to_line(str, err_pos) - local line_str = get_line(str, line_no) or "" - return nil, err_msg:format(msg or "", line_no, trim(line_str)) - end - return tree - end - } - end - return { - extract_line = extract_line, - build_grammar = build_grammar, - string = function(str) - return file_parser():match(str) - end - } - -end -package.preload['moonscript.base'] = function() - local compile = require("moonscript.compile") - local parse = require("moonscript.parse") - local concat, insert, remove - do - local _obj_0 = table - concat, insert, remove = _obj_0.concat, _obj_0.insert, _obj_0.remove + _base_0.__class = _class_0 + LocalName = _class_0 end - local split, dump, get_options, unpack + local NameProxy do - local _obj_0 = require("moonscript.util") - split, dump, get_options, unpack = _obj_0.split, _obj_0.dump, _obj_0.get_options, _obj_0.unpack - end - local lua = { - loadstring = loadstring, - load = load - } - local dirsep, line_tables, create_moonpath, to_lua, loadstring, loadfile, dofile - dirsep = "/" - line_tables = require("moonscript.line_tables") - create_moonpath = function(package_path) - local moonpaths - do - local _accum_0 = { } - local _len_0 = 1 - local _list_0 = split(package_path, ";") - for _index_0 = 1, #_list_0 do - local _continue_0 = false - repeat - local path = _list_0[_index_0] - local prefix = path:match("^(.-)%.lua$") - if not (prefix) then - _continue_0 = true - break + local _base_0 = { + get_name = function(self, scope, dont_put) + if dont_put == nil then + dont_put = true + end + if not self.name then + self.name = scope:free_name(self.prefix, dont_put) + end + return self.name + end, + chain = function(self, ...) + local items = { + base = self, + ... + } + for k, v in ipairs(items) do + if type(v) == "string" then + items[k] = { + "dot", + v + } + else + items[k] = v end - local _value_0 = prefix .. ".moon" - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - _continue_0 = true - until true - if not _continue_0 then - break + end + return build.chain(items) + end, + index = function(self, key) + if type(key) == "string" then + key = { + "ref", + key + } + end + return build.chain({ + base = self, + { + "index", + key + } + }) + end, + __tostring = function(self) + if self.name then + return ("name<%s>"):format(self.name) + else + return ("name"):format(self.prefix) end end - moonpaths = _accum_0 - end - return concat(moonpaths, ";") - end - to_lua = function(text, options) - if options == nil then - options = { } - end - if "string" ~= type(text) then - local t = type(text) - return nil, "expecting string (got " .. t .. ")" - end - local tree, err = parse.string(text) - if not tree then - return nil, err - end - local code, ltable, pos = compile.tree(tree, options) - if not code then - return nil, compile.format_error(ltable, pos, text) - end - return code, ltable - end - loadstring = function(...) - local options, str, chunk_name, mode, env = get_options(...) - chunk_name = chunk_name or "=(moonscript.loadstring)" - local code, ltable_or_err = to_lua(str, options) - if not (code) then - return nil, ltable_or_err - end - if chunk_name then - line_tables[chunk_name] = ltable_or_err - end - return (lua.loadstring or lua.load)(code, chunk_name, unpack({ - mode, - env - })) - end - loadfile = function(fname, ...) - local file, err = io.open(fname) - if not (file) then - return nil, err - end - local text = assert(file:read("*a")) - file:close() - return loadstring(text, "@" .. tostring(fname), ...) - end - dofile = function(...) - local f = assert(loadfile(...)) - return f() + } + _base_0.__index = _base_0 + local _class_0 = setmetatable({ + __init = function(self, prefix) + self.prefix = prefix + self[1] = "temp_name" + end, + __base = _base_0, + __name = "NameProxy" + }, { + __index = _base_0, + __call = function(cls, ...) + local _self_0 = setmetatable({}, _base_0) + cls.__init(_self_0, ...) + return _self_0 + end + }) + _base_0.__class = _class_0 + NameProxy = _class_0 end return { - _NAME = "moonscript", - to_lua = to_lua, - dirsep = dirsep, - dofile = dofile, - loadfile = loadfile, - loadstring = loadstring, - create_moonpath = create_moonpath + NameProxy = NameProxy, + LocalName = LocalName } end -package.preload['moonscript.compile'] = function() +package.preload['moonscript.transform'] = function() + local types = require("moonscript.types") local util = require("moonscript.util") - local dump = require("moonscript.dump") - local transform = require("moonscript.transform") + local data = require("moonscript.data") + local reversed, unpack + reversed, unpack = util.reversed, util.unpack + local ntype, mtype, build, smart_node, is_slice, value_is_singular + ntype, mtype, build, smart_node, is_slice, value_is_singular = types.ntype, types.mtype, types.build, types.smart_node, types.is_slice, types.value_is_singular + local insert + do + local _obj_0 = table + insert = _obj_0.insert + end local NameProxy, LocalName do local _obj_0 = require("moonscript.transform.names") NameProxy, LocalName = _obj_0.NameProxy, _obj_0.LocalName end - local Set - Set = require("moonscript.data").Set - local ntype, value_can_be_statement - do - local _obj_0 = require("moonscript.types") - ntype, value_can_be_statement = _obj_0.ntype, _obj_0.value_can_be_statement - end - local statement_compilers = require("moonscript.compile.statement") - local value_compilers = require("moonscript.compile.value") - local concat, insert - do - local _obj_0 = table - concat, insert = _obj_0.concat, _obj_0.insert - end - local pos_to_line, get_closest_line, trim, unpack - pos_to_line, get_closest_line, trim, unpack = util.pos_to_line, util.get_closest_line, util.trim, util.unpack - local mtype = util.moon.type - local indent_char = " " - local Line, DelayedLine, Lines, Block, RootBlock + local destructure = require("moonscript.transform.destructure") + local NOOP = { + "noop" + } + local Run, apply_to_last, is_singular, extract_declarations, expand_elseif_assign, constructor_name, with_continue_listener, Transformer, construct_comprehension, Statement, Accumulator, default_accumulator, implicitly_return, Value do - local _class_0 local _base_0 = { - mark_pos = function(self, pos, line) - if line == nil then - line = #self - end - if not (self.posmap[line]) then - self.posmap[line] = pos - end + call = function(self, state) + return self.fn(state) + end + } + _base_0.__index = _base_0 + local _class_0 = setmetatable({ + __init = function(self, fn) + self.fn = fn + self[1] = "run" end, - add = function(self, item) - local _exp_0 = mtype(item) - if Line == _exp_0 then - item:render(self) - elseif Block == _exp_0 then - item:render(self) + __base = _base_0, + __name = "Run" + }, { + __index = _base_0, + __call = function(cls, ...) + local _self_0 = setmetatable({}, _base_0) + cls.__init(_self_0, ...) + return _self_0 + end + }) + _base_0.__class = _class_0 + Run = _class_0 + end + apply_to_last = function(stms, fn) + local last_exp_id = 0 + for i = #stms, 1, -1 do + local stm = stms[i] + if stm and mtype(stm) ~= Run then + last_exp_id = i + break + end + end + return (function() + local _accum_0 = { } + local _len_0 = 1 + for i, stm in ipairs(stms) do + if i == last_exp_id then + _accum_0[_len_0] = { + "transform", + stm, + fn + } else - self[#self + 1] = item - end - return self - end, - flatten_posmap = function(self, line_no, out) - if line_no == nil then - line_no = 0 + _accum_0[_len_0] = stm end - if out == nil then - out = { } + _len_0 = _len_0 + 1 + end + return _accum_0 + end)() + end + is_singular = function(body) + if #body ~= 1 then + return false + end + if "group" == ntype(body) then + return is_singular(body[2]) + else + return body[1] + end + end + extract_declarations = function(self, body, start, out) + if body == nil then + body = self.current_stms + end + if start == nil then + start = self.current_stm_i + 1 + end + if out == nil then + out = { } + end + for i = start, #body do + local _continue_0 = false + repeat + local stm = body[i] + if stm == nil then + _continue_0 = true + break end - local posmap = self.posmap - for i, l in ipairs(self) do - local _exp_0 = mtype(l) - if "string" == _exp_0 or DelayedLine == _exp_0 then - line_no = line_no + 1 - out[line_no] = posmap[i] - for _ in l:gmatch("\n") do - line_no = line_no + 1 + stm = self.transform.statement(stm) + body[i] = stm + local _exp_0 = stm[1] + if "assign" == _exp_0 or "declare" == _exp_0 then + local _list_0 = stm[2] + for _index_0 = 1, #_list_0 do + local name = _list_0[_index_0] + if ntype(name) == "ref" then + insert(out, name) + elseif type(name) == "string" then + insert(out, name) end - out[line_no] = posmap[i] - elseif Lines == _exp_0 then - local _ - _, line_no = l:flatten_posmap(line_no, out) - else - error("Unknown item in Lines: " .. tostring(l)) end + elseif "group" == _exp_0 then + extract_declarations(self, stm[2], 1, out) + end + _continue_0 = true + until true + if not _continue_0 then + break + end + end + return out + end + expand_elseif_assign = function(ifstm) + for i = 4, #ifstm do + local case = ifstm[i] + if ntype(case) == "elseif" and ntype(case[2]) == "assign" then + local split = { + unpack(ifstm, 1, i - 1) + } + insert(split, { + "else", + { + { + "if", + case[2], + case[3], + unpack(ifstm, i + 1) + } + } + }) + return split + end + end + return ifstm + end + constructor_name = "new" + with_continue_listener = function(body) + local continue_name = nil + return { + Run(function(self) + return self:listen("continue", function() + if not (continue_name) then + continue_name = NameProxy("continue") + self:put_name(continue_name) + end + return continue_name + end) + end), + build.group(body), + Run(function(self) + if not (continue_name) then + return + end + self:put_name(continue_name, nil) + return self:splice(function(lines) + return { + { + "assign", + { + continue_name + }, + { + "false" + } + }, + { + "repeat", + "true", + { + lines, + { + "assign", + { + continue_name + }, + { + "true" + } + } + } + }, + { + "if", + { + "not", + continue_name + }, + { + { + "break" + } + } + } + } + end) + end) + } + end + do + local _base_0 = { + transform_once = function(self, scope, node, ...) + if self.seen_nodes[node] then + return node end - return out, line_no - end, - flatten = function(self, indent, buffer) - if indent == nil then - indent = nil + self.seen_nodes[node] = true + local transformer = self.transformers[ntype(node)] + if transformer then + return transformer(scope, node, ...) or node + else + return node end - if buffer == nil then - buffer = { } + end, + transform = function(self, scope, node, ...) + if self.seen_nodes[node] then + return node end - for i = 1, #self do - local l = self[i] - local t = mtype(l) - if t == DelayedLine then - l = l:render() - t = "string" - end - local _exp_0 = t - if "string" == _exp_0 then - if indent then - insert(buffer, indent) - end - insert(buffer, l) - if "string" == type(self[i + 1]) then - local lc = l:sub(-1) - if (lc == ")" or lc == "]") and self[i + 1]:sub(1, 1) == "(" then - insert(buffer, ";") - end - end - insert(buffer, "\n") - elseif Lines == _exp_0 then - l:flatten(indent and indent .. indent_char or indent_char, buffer) + self.seen_nodes[node] = true + while true do + local transformer = self.transformers[ntype(node)] + local res + if transformer then + res = transformer(scope, node, ...) or node else - error("Unknown item in Lines: " .. tostring(l)) + res = node + end + if res == node then + return node end + node = res end - return buffer + return node end, - __tostring = function(self) - local strip - strip = function(t) - if "table" == type(t) then - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #t do - local v = t[_index_0] - _accum_0[_len_0] = strip(v) - _len_0 = _len_0 + 1 - end - return _accum_0 - else - return t - end + bind = function(self, scope) + return function(...) + return self:transform(scope, ...) end - return "Lines<" .. tostring(util.dump(strip(self)):sub(1, -2)) .. ">" + end, + __call = function(self, ...) + return self:transform(...) + end, + can_transform = function(self, node) + return self.transformers[ntype(node)] ~= nil end } _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self) - self.posmap = { } + local _class_0 = setmetatable({ + __init = function(self, transformers) + self.transformers = transformers + self.seen_nodes = setmetatable({ }, { + __mode = "k" + }) end, __base = _base_0, - __name = "Lines" + __name = "Transformer" }, { __index = _base_0, __call = function(cls, ...) @@ -5133,1133 +3309,1925 @@ package.preload['moonscript.compile'] = function() end }) _base_0.__class = _class_0 - Lines = _class_0 + Transformer = _class_0 end - do - local _class_0 - local _base_0 = { - pos = nil, - append_list = function(self, items, delim) - for i = 1, #items do - self:append(items[i]) - if i < #items then - insert(self, delim) - end + construct_comprehension = function(inner, clauses) + local current_stms = inner + for _, clause in reversed(clauses) do + local t = clause[1] + local _exp_0 = t + if "for" == _exp_0 then + local name, bounds + _, name, bounds = clause[1], clause[2], clause[3] + current_stms = { + "for", + name, + bounds, + current_stms + } + elseif "foreach" == _exp_0 then + local names, iter + _, names, iter = clause[1], clause[2], clause[3] + current_stms = { + "foreach", + names, + { + iter + }, + current_stms + } + elseif "when" == _exp_0 then + local cond + _, cond = clause[1], clause[2] + current_stms = { + "if", + cond, + current_stms + } + else + current_stms = error("Unknown comprehension clause: " .. t) + end + current_stms = { + current_stms + } + end + return current_stms[1] + end + Statement = Transformer({ + transform = function(self, tuple) + local _, node, fn + _, node, fn = tuple[1], tuple[2], tuple[3] + return fn(node) + end, + root_stms = function(self, body) + return apply_to_last(body, implicitly_return(self)) + end, + ["return"] = function(self, node) + local ret_val = node[2] + local ret_val_type = ntype(ret_val) + if ret_val_type == "explist" and #ret_val == 2 then + ret_val = ret_val[2] + ret_val_type = ntype(ret_val) + end + if types.cascading[ret_val_type] then + return implicitly_return(self)(ret_val) + end + if ret_val_type == "chain" or ret_val_type == "comprehension" or ret_val_type == "tblcomprehension" then + ret_val = Value:transform_once(self, ret_val) + if ntype(ret_val) == "block_exp" then + return build.group(apply_to_last(ret_val[2], function(stm) + return { + "return", + stm + } + end)) end - return nil - end, - append = function(self, first, ...) - if Line == mtype(first) then - if not (self.pos) then - self.pos = first.pos + end + node[2] = ret_val + return node + end, + declare_glob = function(self, node) + local names = extract_declarations(self) + if node[2] == "^" then + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #names do + local _continue_0 = false + repeat + local name = names[_index_0] + if not (name[2]:match("^%u")) then + _continue_0 = true + break + end + local _value_0 = name + _accum_0[_len_0] = _value_0 + _len_0 = _len_0 + 1 + _continue_0 = true + until true + if not _continue_0 then + break + end end - for _index_0 = 1, #first do - local value = first[_index_0] - self:append(value) + names = _accum_0 + end + end + return { + "declare", + names + } + end, + assign = function(self, node) + local names, values = unpack(node, 2) + local num_values = #values + local num_names = #values + if num_names == 1 and num_values == 1 then + local first_value = values[1] + local first_name = names[1] + local first_type = ntype(first_value) + if first_type == "chain" then + first_value = Value:transform_once(self, first_value) + first_type = ntype(first_value) + end + local _exp_0 = ntype(first_value) + if "block_exp" == _exp_0 then + local block_body = first_value[2] + local idx = #block_body + block_body[idx] = build.assign_one(first_name, block_body[idx]) + return build.group({ + { + "declare", + { + first_name + } + }, + { + "do", + block_body + } + }) + elseif "comprehension" == _exp_0 or "tblcomprehension" == _exp_0 or "foreach" == _exp_0 or "for" == _exp_0 or "while" == _exp_0 then + return build.assign_one(first_name, Value:transform_once(self, first_value)) + else + values[1] = first_value + end + end + local transformed + if num_values == 1 then + local value = values[1] + local t = ntype(value) + if t == "decorated" then + value = self.transform.statement(value) + t = ntype(value) + end + if types.cascading[t] then + local ret + ret = function(stm) + if types.is_value(stm) then + return { + "assign", + names, + { + stm + } + } + else + return stm + end end - else - insert(self, first) - end - if ... then - return self:append(...) + transformed = build.group({ + { + "declare", + names + }, + self.transform.statement(value, ret, node) + }) end - end, - render = function(self, buffer) - local current = { } - local add_current - add_current = function() - buffer:add(concat(current)) - return buffer:mark_pos(self.pos) + end + node = transformed or node + if destructure.has_destructure(names) then + return destructure.split_assign(self, node) + end + return node + end, + continue = function(self, node) + local continue_name = self:send("continue") + if not (continue_name) then + error("continue must be inside of a loop") + end + return build.group({ + build.assign_one(continue_name, "true"), + { + "break" + } + }) + end, + export = function(self, node) + if #node > 2 then + if node[2] == "class" then + local cls = smart_node(node[3]) + return build.group({ + { + "export", + { + cls.name + } + }, + cls + }) + else + return build.group({ + { + "export", + node[2] + }, + build.assign({ + names = node[2], + values = node[3] + }) + }) end - for _index_0 = 1, #self do - local chunk = self[_index_0] - local _exp_0 = mtype(chunk) - if Block == _exp_0 then - local _list_0 = chunk:render(Lines()) - for _index_1 = 1, #_list_0 do - local block_chunk = _list_0[_index_1] - if "string" == type(block_chunk) then - insert(current, block_chunk) - else - add_current() - buffer:add(block_chunk) - current = { } - end - end + else + return nil + end + end, + update = function(self, node) + local _, name, op, exp = unpack(node) + local op_final = op:match("^(.+)=$") + if not op_final then + error("Unknown op: " .. op) + end + if not (value_is_singular(exp)) then + exp = { + "parens", + exp + } + end + return build.assign_one(name, { + "exp", + name, + op_final, + exp + }) + end, + import = function(self, node) + local _, names, source = unpack(node) + local table_values + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #names do + local name = names[_index_0] + local dest_val + if ntype(name) == "colon_stub" then + dest_val = name[2] else - insert(current, chunk) + dest_val = name end + local _value_0 = { + { + "key_literal", + name + }, + dest_val + } + _accum_0[_len_0] = _value_0 + _len_0 = _len_0 + 1 end - if current[1] then - add_current() + table_values = _accum_0 + end + local dest = { + "table", + table_values + } + return { + "assign", + { + dest + }, + { + source + }, + [-1] = node[-1] + } + end, + comprehension = function(self, node, action) + local _, exp, clauses = unpack(node) + action = action or function(exp) + return { + exp + } + end + return construct_comprehension(action(exp), clauses) + end, + ["do"] = function(self, node, ret) + if ret then + node[2] = apply_to_last(node[2], ret) + end + return node + end, + decorated = function(self, node) + local stm, dec = unpack(node, 2) + local wrapped + local _exp_0 = dec[1] + if "if" == _exp_0 then + local cond, fail = unpack(dec, 2) + if fail then + fail = { + "else", + { + fail + } + } end - return buffer - end, - __tostring = function(self) - return "Line<" .. tostring(util.dump(self):sub(1, -2)) .. ">" + wrapped = { + "if", + cond, + { + stm + }, + fail + } + elseif "unless" == _exp_0 then + wrapped = { + "unless", + dec[2], + { + stm + } + } + elseif "comprehension" == _exp_0 then + wrapped = { + "comprehension", + stm, + dec[2] + } + else + wrapped = error("Unknown decorator " .. dec[1]) end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function() end, - __base = _base_0, - __name = "Line" - }, { - __index = _base_0, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + if ntype(stm) == "assign" then + wrapped = build.group({ + build.declare({ + names = (function() + local _accum_0 = { } + local _len_0 = 1 + local _list_0 = stm[2] + for _index_0 = 1, #_list_0 do + local name = _list_0[_index_0] + if ntype(name) == "ref" then + _accum_0[_len_0] = name + _len_0 = _len_0 + 1 + end + end + return _accum_0 + end)() + }), + wrapped + }) end - }) - _base_0.__class = _class_0 - Line = _class_0 - end - do - local _class_0 - local _base_0 = { - prepare = function() end, - render = function(self) - self:prepare() - return concat(self) + return wrapped + end, + unless = function(self, node) + return { + "if", + { + "not", + { + "parens", + node[2] + } + }, + unpack(node, 3) + } + end, + ["if"] = function(self, node, ret) + if ntype(node[2]) == "assign" then + local _, assign, body = unpack(node) + if destructure.has_destructure(assign[2]) then + local name = NameProxy("des") + body = { + destructure.build_assign(self, assign[2][1], name), + build.group(node[3]) + } + return build["do"]({ + build.assign_one(name, assign[3][1]), + { + "if", + name, + body, + unpack(node, 4) + } + }) + else + local name = assign[2][1] + return build["do"]({ + assign, + { + "if", + name, + unpack(node, 3) + } + }) + end end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, fn) - self.prepare = fn - end, - __base = _base_0, - __name = "DelayedLine" - }, { - __index = _base_0, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + node = expand_elseif_assign(node) + if ret then + smart_node(node) + node['then'] = apply_to_last(node['then'], ret) + for i = 4, #node do + local case = node[i] + local body_idx = #node[i] + case[body_idx] = apply_to_last(case[body_idx], ret) + end end - }) - _base_0.__class = _class_0 - DelayedLine = _class_0 - end - do - local _class_0 - local _base_0 = { - header = "do", - footer = "end", - export_all = false, - export_proper = false, - value_compilers = value_compilers, - statement_compilers = statement_compilers, - __tostring = function(self) - local h - if "string" == type(self.header) then - h = self.header + return node + end, + with = function(self, node, ret) + local exp, block = unpack(node, 2) + local copy_scope = true + local scope_name, named_assign + if ntype(exp) == "assign" then + local names, values = unpack(exp, 2) + local first_name = names[1] + if ntype(first_name) == "ref" then + scope_name = first_name + named_assign = exp + exp = values[1] + copy_scope = false else - h = unpack(self.header:render({ })) + scope_name = NameProxy("with") + exp = values[1] + values[1] = scope_name + named_assign = { + "assign", + names, + values + } end - return "Block<" .. tostring(h) .. "> <- " .. tostring(self.parent) - end, - set = function(self, name, value) - self._state[name] = value - end, - get = function(self, name) - return self._state[name] - end, - get_current = function(self, name) - return rawget(self._state, name) - end, - listen = function(self, name, fn) - self._listeners[name] = fn - end, - unlisten = function(self, name) - self._listeners[name] = nil - end, - send = function(self, name, ...) - do - local fn = self._listeners[name] - if fn then - return fn(self, ...) + elseif self:is_local(exp) then + scope_name = exp + copy_scope = false + end + scope_name = scope_name or NameProxy("with") + return build["do"]({ + Run(function(self) + return self:set("scope_var", scope_name) + end), + copy_scope and build.assign_one(scope_name, exp) or NOOP, + named_assign or NOOP, + build.group(block), + (function() + if ret then + return ret(scope_name) end - end - end, - extract_assign_name = function(self, node) - local is_local = false - local real_name - local _exp_0 = mtype(node) - if LocalName == _exp_0 then - is_local = true - real_name = node:get_name(self) - elseif NameProxy == _exp_0 then - real_name = node:get_name(self) - elseif "table" == _exp_0 then - real_name = node[1] == "ref" and node[2] - elseif "string" == _exp_0 then - real_name = node - end - return real_name, is_local - end, - declare = function(self, names) - local undeclared - do - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #names do - local _continue_0 = false - repeat - local name = names[_index_0] - local real_name, is_local = self:extract_assign_name(name) - if not (is_local or real_name and not self:has_name(real_name, true)) then - _continue_0 = true - break - end - self:put_name(real_name) - if self:name_exported(real_name) then - _continue_0 = true - break - end - local _value_0 = real_name - _accum_0[_len_0] = _value_0 - _len_0 = _len_0 + 1 - _continue_0 = true - until true - if not _continue_0 then - break + end)() + }) + end, + foreach = function(self, node, _) + smart_node(node) + local source = unpack(node.iter) + local destructures = { } + do + local _accum_0 = { } + local _len_0 = 1 + for i, name in ipairs(node.names) do + if ntype(name) == "table" then + do + local proxy = NameProxy("des") + insert(destructures, destructure.build_assign(self, name, proxy)) + _accum_0[_len_0] = proxy end + else + _accum_0[_len_0] = name end - undeclared = _accum_0 - end - return undeclared - end, - whitelist_names = function(self, names) - self._name_whitelist = Set(names) - end, - name_exported = function(self, name) - if self.export_all then - return true - end - if self.export_proper and name:match("^%u") then - return true - end - end, - put_name = function(self, name, ...) - local value = ... - if select("#", ...) == 0 then - value = true - end - if NameProxy == mtype(name) then - name = name:get_name(self) + _len_0 = _len_0 + 1 end - self._names[name] = value - end, - has_name = function(self, name, skip_exports) - if not skip_exports and self:name_exported(name) then - return true + node.names = _accum_0 + end + if next(destructures) then + insert(destructures, build.group(node.body)) + node.body = destructures + end + if ntype(source) == "unpack" then + local list = source[2] + local index_name = NameProxy("index") + local list_name = self:is_local(list) and list or NameProxy("list") + local slice_var = nil + local bounds + if is_slice(list) then + local slice = list[#list] + table.remove(list) + table.remove(slice, 1) + if self:is_local(list) then + list_name = list + end + if slice[2] and slice[2] ~= "" then + local max_tmp_name = NameProxy("max") + slice_var = build.assign_one(max_tmp_name, slice[2]) + slice[2] = { + "exp", + max_tmp_name, + "<", + 0, + "and", + { + "length", + list_name + }, + "+", + max_tmp_name, + "or", + max_tmp_name + } + else + slice[2] = { + "length", + list_name + } + end + bounds = slice + else + bounds = { + 1, + { + "length", + list_name + } + } end - local yes = self._names[name] - if yes == nil and self.parent then - if not self._name_whitelist or self._name_whitelist[name] then - return self.parent:has_name(name, true) + return build.group({ + list_name ~= list and build.assign_one(list_name, list) or NOOP, + slice_var or NOOP, + build["for"]({ + name = index_name, + bounds = bounds, + body = { + { + "assign", + node.names, + { + NameProxy.index(list_name, index_name) + } + }, + build.group(node.body) + } + }) + }) + end + node.body = with_continue_listener(node.body) + end, + ["while"] = function(self, node) + smart_node(node) + node.body = with_continue_listener(node.body) + end, + ["for"] = function(self, node) + smart_node(node) + node.body = with_continue_listener(node.body) + end, + switch = function(self, node, ret) + local _, exp, conds = unpack(node) + local exp_name = NameProxy("exp") + local convert_cond + convert_cond = function(cond) + local t, case_exps, body = unpack(cond) + local out = { } + insert(out, t == "case" and "elseif" or "else") + if t ~= "else" then + local cond_exp = { } + for i, case in ipairs(case_exps) do + if i == 1 then + insert(cond_exp, "exp") + else + insert(cond_exp, "or") + end + if not (value_is_singular(case)) then + case = { + "parens", + case + } + end + insert(cond_exp, { + "exp", + case, + "==", + exp_name + }) end + insert(out, cond_exp) else - return yes + body = case_exps end - end, - is_local = function(self, node) - local t = mtype(node) - if t == "string" then - return self:has_name(node, false) + if ret then + body = apply_to_last(body, ret) end - if t == NameProxy or t == LocalName then - return true + insert(out, body) + return out + end + local first = true + local if_stm = { + "if" + } + for _index_0 = 1, #conds do + local cond = conds[_index_0] + local if_cond = convert_cond(cond) + if first then + first = false + insert(if_stm, if_cond[2]) + insert(if_stm, if_cond[3]) + else + insert(if_stm, if_cond) end - if t == "table" then - if node[1] == "ref" or (node[1] == "chain" and #node == 2) then - return self:is_local(node[2]) + end + return build.group({ + build.assign_one(exp_name, exp), + if_stm + }) + end, + class = function(self, node, ret, parent_assign) + local _, name, parent_val, body = unpack(node) + if parent_val == "" then + parent_val = nil + end + local statements = { } + local properties = { } + for _index_0 = 1, #body do + local item = body[_index_0] + local _exp_0 = item[1] + if "stm" == _exp_0 then + insert(statements, item[2]) + elseif "props" == _exp_0 then + for _index_1 = 2, #item do + local tuple = item[_index_1] + if ntype(tuple[1]) == "self" then + insert(statements, build.assign_one(unpack(tuple))) + else + insert(properties, tuple) + end end end - return false - end, - free_name = function(self, prefix, dont_put) - prefix = prefix or "moon" - local searching = true - local name, i = nil, 0 - while searching do - name = concat({ - "", - prefix, - i - }, "_") - i = i + 1 - searching = self:has_name(name, true) - end - if not dont_put then - self:put_name(name) - end - return name - end, - init_free_var = function(self, prefix, value) - local name = self:free_name(prefix, true) - self:stm({ - "assign", - { - name - }, - { - value - } - }) - return name - end, - add = function(self, item, pos) - do - local _with_0 = self._lines - _with_0:add(item) - if pos then - _with_0:mark_pos(pos) + end + local constructor + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #properties do + local _continue_0 = false + repeat + local tuple = properties[_index_0] + local key = tuple[1] + local _value_0 + if key[1] == "key_literal" and key[2] == constructor_name then + constructor = tuple[2] + _continue_0 = true + break + else + _value_0 = tuple + end + _accum_0[_len_0] = _value_0 + _len_0 = _len_0 + 1 + _continue_0 = true + until true + if not _continue_0 then + break end end - return item - end, - render = function(self, buffer) - buffer:add(self.header) - buffer:mark_pos(self.pos) - if self.next then - buffer:add(self._lines) - self.next:render(buffer) + properties = _accum_0 + end + local parent_cls_name = NameProxy("parent") + local base_name = NameProxy("base") + local self_name = NameProxy("self") + local cls_name = NameProxy("class") + if not (constructor) then + if parent_val then + constructor = build.fndef({ + args = { + { + "..." + } + }, + arrow = "fat", + body = { + build.chain({ + base = "super", + { + "call", + { + "..." + } + } + }) + } + }) else - if #self._lines == 0 and "string" == type(buffer[#buffer]) then - buffer[#buffer] = buffer[#buffer] .. (" " .. (unpack(Lines():add(self.footer)))) - else - buffer:add(self._lines) - buffer:add(self.footer) - buffer:mark_pos(self.pos) - end - end - return buffer - end, - block = function(self, header, footer) - return Block(self, header, footer) - end, - line = function(self, ...) - do - local _with_0 = Line() - _with_0:append(...) - return _with_0 + constructor = build.fndef() end - end, - is_stm = function(self, node) - return self.statement_compilers[ntype(node)] ~= nil - end, - is_value = function(self, node) - local t = ntype(node) - return self.value_compilers[t] ~= nil or t == "value" - end, - name = function(self, node, ...) - if type(node) == "string" then - return node + end + local real_name = name or parent_assign and parent_assign[2][1] + local _exp_0 = ntype(real_name) + if "chain" == _exp_0 then + local last = real_name[#real_name] + local _exp_1 = ntype(last) + if "dot" == _exp_1 then + real_name = { + "string", + '"', + last[2] + } + elseif "index" == _exp_1 then + real_name = last[2] else - return self:value(node, ...) + real_name = "nil" end - end, - value = function(self, node, ...) - node = self.transform.value(node) - local action - if type(node) ~= "table" then - action = "raw_value" + elseif "nil" == _exp_0 then + real_name = "nil" + else + local name_t = type(real_name) + local flattened_name + if name_t == "string" then + flattened_name = real_name + elseif name_t == "table" and real_name[1] == "ref" then + flattened_name = real_name[2] else - action = node[1] + flattened_name = error("don't know how to extract name from " .. tostring(name_t)) end - local fn = self.value_compilers[action] - if not (fn) then - error({ - "compile-error", - "Failed to find value compiler for: " .. dump.value(node), - node[-1] + real_name = { + "string", + '"', + flattened_name + } + end + local cls = build.table({ + { + "__init", + constructor + }, + { + "__base", + base_name + }, + { + "__name", + real_name + }, + parent_val and { + "__parent", + parent_cls_name + } or nil + }) + local class_index + if parent_val then + local class_lookup = build["if"]({ + cond = { + "exp", + { + "ref", + "val" + }, + "==", + "nil" + }, + ["then"] = { + parent_cls_name:index("name") + } + }) + insert(class_lookup, { + "else", + { + "val" + } + }) + class_index = build.fndef({ + args = { + { + "cls" + }, + { + "name" + } + }, + body = { + build.assign_one(LocalName("val"), build.chain({ + base = "rawget", + { + "call", + { + base_name, + { + "ref", + "name" + } + } + } + })), + class_lookup + } + }) + else + class_index = base_name + end + local cls_mt = build.table({ + { + "__index", + class_index + }, + { + "__call", + build.fndef({ + args = { + { + "cls" + }, + { + "..." + } + }, + body = { + build.assign_one(self_name, build.chain({ + base = "setmetatable", + { + "call", + { + "{}", + base_name + } + } + })), + build.chain({ + base = "cls.__init", + { + "call", + { + self_name, + "..." + } + } + }), + self_name + } }) - end - local out = fn(self, node, ...) - if type(node) == "table" and node[-1] then - if type(out) == "string" then - do - local _with_0 = Line() - _with_0:append(out) - out = _with_0 + } + }) + cls = build.chain({ + base = "setmetatable", + { + "call", + { + cls, + cls_mt + } + } + }) + local value = nil + do + local out_body = { + Run(function(self) + if name then + self:put_name(name) end - end - out.pos = node[-1] - end - return out - end, - values = function(self, values, delim) - delim = delim or ', ' - do - local _with_0 = Line() - _with_0:append_list((function() - local _accum_0 = { } - local _len_0 = 1 - for _index_0 = 1, #values do - local v = values[_index_0] - _accum_0[_len_0] = self:value(v) - _len_0 = _len_0 + 1 + return self:set("super", function(block, chain) + if chain then + local slice + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 3, #chain do + local item = chain[_index_0] + _accum_0[_len_0] = item + _len_0 = _len_0 + 1 + end + slice = _accum_0 + end + local new_chain = { + "chain", + parent_cls_name + } + local head = slice[1] + if head == nil then + return parent_cls_name + end + local _exp_1 = head[1] + if "call" == _exp_1 then + local calling_name = block:get("current_block") + slice[1] = { + "call", + { + "self", + unpack(head[2]) + } + } + if ntype(calling_name) == "key_literal" then + insert(new_chain, { + "dot", + calling_name[2] + }) + else + insert(new_chain, { + "index", + calling_name + }) + end + elseif "colon" == _exp_1 then + local call = head[3] + insert(new_chain, { + "dot", + head[2] + }) + slice[1] = { + "call", + { + "self", + unpack(call[2]) + } + } + end + for _index_0 = 1, #slice do + local item = slice[_index_0] + insert(new_chain, item) + end + return new_chain + else + return parent_cls_name + end + end) + end), + { + "declare_glob", + "*" + }, + parent_val and build.assign_one(parent_cls_name, parent_val) or NOOP, + build.assign_one(base_name, { + "table", + properties + }), + build.assign_one(base_name:chain("__index"), base_name), + parent_val and build.chain({ + base = "setmetatable", + { + "call", + { + base_name, + build.chain({ + base = parent_cls_name, + { + "dot", + "__base" + } + }) + } + } + }) or NOOP, + build.assign_one(cls_name, cls), + build.assign_one(base_name:chain("__class"), cls_name), + build.group((function() + if #statements > 0 then + return { + build.assign_one(LocalName("self"), cls_name), + build.group(statements) + } end - return _accum_0 - end)(), delim) - return _with_0 - end + end)()), + parent_val and build["if"]({ + cond = { + "exp", + parent_cls_name:chain("__inherited") + }, + ["then"] = { + parent_cls_name:chain("__inherited", { + "call", + { + parent_cls_name, + cls_name + } + }) + } + }) or NOOP, + build.group((function() + if name then + return { + build.assign_one(name, cls_name) + } + end + end)()), + (function() + if ret then + return ret(cls_name) + end + end)() + } + value = build.group({ + build.group((function() + if ntype(name) == "value" then + return { + build.declare({ + names = { + name + } + }) + } + end + end)()), + build["do"](out_body) + }) + end + return value + end + }) + do + local _base_0 = { + body_idx = { + ["for"] = 4, + ["while"] = 3, + foreach = 4 + }, + convert = function(self, node) + local index = self.body_idx[ntype(node)] + node[index] = self:mutate_body(node[index]) + return self:wrap(node) end, - stm = function(self, node, ...) - if not node then - return + wrap = function(self, node, group_type) + if group_type == nil then + group_type = "block_exp" end - node = self.transform.statement(node) - local result - do - local fn = self.statement_compilers[ntype(node)] - if fn then - result = fn(self, node, ...) - else - if value_can_be_statement(node) then - result = self:value(node) + return build[group_type]({ + build.assign_one(self.accum_name, build.table()), + build.assign_one(self.len_name, 1), + node, + group_type == "block_exp" and self.accum_name or NOOP + }) + end, + mutate_body = function(self, body) + local single_stm = is_singular(body) + local val + if single_stm and types.is_value(single_stm) then + body = { } + val = single_stm + else + body = apply_to_last(body, function(n) + if types.is_value(n) then + return build.assign_one(self.value_name, n) else - result = self:stm({ - "assign", + return build.group({ { - "_" + "declare", + { + self.value_name + } }, - { - node - } + n }) end - end - end - if result then - if type(node) == "table" and type(result) == "table" and node[-1] then - result.pos = node[-1] - end - self:add(result) + end) + val = self.value_name end - return nil + local update = { + build.assign_one(NameProxy.index(self.accum_name, self.len_name), val), + { + "update", + self.len_name, + "+=", + 1 + } + } + insert(body, build.group(update)) + return body + end + } + _base_0.__index = _base_0 + local _class_0 = setmetatable({ + __init = function(self, accum_name) + self.accum_name = NameProxy("accum") + self.value_name = NameProxy("value") + self.len_name = NameProxy("len") end, - stms = function(self, stms, ret) - if ret then - error("deprecated stms call, use transformer") + __base = _base_0, + __name = "Accumulator" + }, { + __index = _base_0, + __call = function(cls, ...) + local _self_0 = setmetatable({}, _base_0) + cls.__init(_self_0, ...) + return _self_0 + end + }) + _base_0.__class = _class_0 + Accumulator = _class_0 + end + default_accumulator = function(self, node) + return Accumulator():convert(node) + end + implicitly_return = function(scope) + local is_top = true + local fn + fn = function(stm) + local t = ntype(stm) + if t == "decorated" then + stm = scope.transform.statement(stm) + t = ntype(stm) + end + if types.cascading[t] then + is_top = false + return scope.transform.statement(stm, fn) + elseif types.manual_return[t] or not types.is_value(stm) then + if is_top and t == "return" and stm[2] == "" then + return NOOP + else + return stm + end + else + if t == "comprehension" and not types.comprehension_has_value(stm) then + return stm + else + return { + "return", + stm + } + end + end + end + return fn + end + Value = Transformer({ + ["for"] = default_accumulator, + ["while"] = default_accumulator, + foreach = default_accumulator, + ["do"] = function(self, node) + return build.block_exp(node[2]) + end, + decorated = function(self, node) + return self.transform.statement(node) + end, + class = function(self, node) + return build.block_exp({ + node + }) + end, + string = function(self, node) + local delim = node[2] + local convert_part + convert_part = function(part) + if type(part) == "string" or part == nil then + return { + "string", + delim, + part or "" + } + else + return build.chain({ + base = "tostring", + { + "call", + { + part[2] + } + } + }) end - local current_stms, current_stm_i - current_stms, current_stm_i = self.current_stms, self.current_stm_i - self.current_stms = stms - for i = 1, #stms do - self.current_stm_i = i - self:stm(stms[i]) + end + if #node <= 3 then + if type(node[3]) == "string" then + return node + else + return convert_part(node[3]) end - self.current_stms = current_stms - self.current_stm_i = current_stm_i - return nil - end, - splice = function(self, fn) - local lines = { - "lines", - self._lines + end + local e = { + "exp", + convert_part(node[3]) + } + for i = 4, #node do + insert(e, "..") + insert(e, convert_part(node[i])) + end + return e + end, + comprehension = function(self, node) + local a = Accumulator() + node = self.transform.statement(node, function(exp) + return a:mutate_body({ + exp + }) + end) + return a:wrap(node) + end, + tblcomprehension = function(self, node) + local _, explist, clauses = unpack(node) + local key_exp, value_exp = unpack(explist) + local accum = NameProxy("tbl") + local inner + if value_exp then + local dest = build.chain({ + base = accum, + { + "index", + key_exp + } + }) + inner = { + build.assign_one(dest, value_exp) + } + else + local key_name, val_name = NameProxy("key"), NameProxy("val") + local dest = build.chain({ + base = accum, + { + "index", + key_name + } + }) + inner = { + build.assign({ + names = { + key_name, + val_name + }, + values = { + key_exp + } + }), + build.assign_one(dest, val_name) } - self._lines = Lines() - return self:stms(fn(lines)) end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, parent, header, footer) - self.parent, self.header, self.footer = parent, header, footer - self._lines = Lines() - self._names = { } - self._state = { } - self._listeners = { } - do - self.transform = { - value = transform.Value:bind(self), - statement = transform.Statement:bind(self) + return build.block_exp({ + build.assign_one(accum, build.table()), + construct_comprehension(inner, clauses), + accum + }) + end, + fndef = function(self, node) + smart_node(node) + node.body = apply_to_last(node.body, implicitly_return(self)) + node.body = { + Run(function(self) + return self:listen("varargs", function() end) + end), + unpack(node.body) + } + return node + end, + ["if"] = function(self, node) + return build.block_exp({ + node + }) + end, + unless = function(self, node) + return build.block_exp({ + node + }) + end, + with = function(self, node) + return build.block_exp({ + node + }) + end, + switch = function(self, node) + return build.block_exp({ + node + }) + end, + chain = function(self, node) + local stub = node[#node] + for i = 3, #node do + local part = node[i] + if ntype(part) == "dot" and data.lua_keywords[part[2]] then + node[i] = { + "index", + { + "string", + '"', + part[2] + } } end - if self.parent then - self.root = self.parent.root - self.indent = self.parent.indent + 1 - setmetatable(self._state, { - __index = self.parent._state - }) - return setmetatable(self._listeners, { - __index = self.parent._listeners + end + if ntype(node[2]) == "string" then + node[2] = { + "parens", + node[2] + } + elseif type(stub) == "table" and stub[1] == "colon_stub" then + table.remove(node, #node) + local base_name = NameProxy("base") + local fn_name = NameProxy("fn") + local is_super = ntype(node[2]) == "ref" and node[2][2] == "super" + return build.block_exp({ + build.assign({ + names = { + base_name + }, + values = { + node + } + }), + build.assign({ + names = { + fn_name + }, + values = { + build.chain({ + base = base_name, + { + "dot", + stub[2] + } + }) + } + }), + build.fndef({ + args = { + { + "..." + } + }, + body = { + build.chain({ + base = fn_name, + { + "call", + { + is_super and "self" or base_name, + "..." + } + } + }) + } }) - else - self.indent = 0 - end - end, - __base = _base_0, - __name = "Block" - }, { - __index = _base_0, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + }) end - }) - _base_0.__class = _class_0 - Block = _class_0 + end, + block_exp = function(self, node) + local _, body = unpack(node) + local fn = nil + local arg_list = { } + fn = smart_node(build.fndef({ + body = { + Run(function(self) + return self:listen("varargs", function() + insert(arg_list, "...") + insert(fn.args, { + "..." + }) + return self:unlisten("varargs") + end) + end), + unpack(body) + } + })) + return build.chain({ + base = { + "parens", + fn + }, + { + "call", + arg_list + } + }) + end + }) + return { + Statement = Statement, + Value = Value, + Run = Run + } + +end +package.preload['moonscript.types'] = function() + local util = require("moonscript.util") + local Set + do + local _obj_0 = require("moonscript.data") + Set = _obj_0.Set + end + local insert + do + local _obj_0 = table + insert = _obj_0.insert + end + local unpack + unpack = util.unpack + local manual_return = Set({ + "foreach", + "for", + "while", + "return" + }) + local cascading = Set({ + "if", + "unless", + "with", + "switch", + "class", + "do" + }) + local ntype + ntype = function(node) + local _exp_0 = type(node) + if "nil" == _exp_0 then + return "nil" + elseif "table" == _exp_0 then + return node[1] + else + return "value" + end end + local mtype do - local _class_0 - local _parent_0 = Block - local _base_0 = { - __tostring = function(self) - return "RootBlock<>" - end, - root_stms = function(self, stms) - if not (self.options.implicitly_return_root == false) then - stms = transform.Statement.transformers.root_stms(self, stms) - end - return self:stms(stms) - end, - render = function(self) - local buffer = self._lines:flatten() - if buffer[#buffer] == "\n" then - buffer[#buffer] = nil - end - return table.concat(buffer) - end - } - _base_0.__index = _base_0 - setmetatable(_base_0, _parent_0.__base) - _class_0 = setmetatable({ - __init = function(self, options) - self.options = options - self.root = self - return _class_0.__parent.__init(self) - end, - __base = _base_0, - __name = "RootBlock", - __parent = _parent_0 - }, { - __index = function(cls, name) - local val = rawget(_base_0, name) - if val == nil then - local parent = rawget(cls, "__parent") - if parent then - return parent[name] - end - else - return val - end - end, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 + local moon_type = util.moon.type + mtype = function(val) + local mt = getmetatable(val) + if mt and mt.smart_node then + return "table" end - }) - _base_0.__class = _class_0 - if _parent_0.__inherited then - _parent_0.__inherited(_parent_0, _class_0) + return moon_type(val) end - RootBlock = _class_0 end - local format_error - format_error = function(msg, pos, file_str) - local line_message - if pos then - local line = pos_to_line(file_str, pos) - local line_str - line_str, line = get_closest_line(file_str, line) - line_str = line_str or "" - line_message = (" [%d] >> %s"):format(line, trim(line_str)) + local has_value + has_value = function(node) + if ntype(node) == "chain" then + local ctype = ntype(node[#node]) + return ctype ~= "call" and ctype ~= "colon" + else + return true end - return concat({ - "Compile error: " .. msg, - line_message - }, "\n") end - local value - value = function(value) - local out = nil - do - local _with_0 = RootBlock() - _with_0:add(_with_0:value(value)) - out = _with_0:render() + local is_value + is_value = function(stm) + local compile = require("moonscript.compile") + local transform = require("moonscript.transform") + return compile.Block:is_value(stm) or transform.Value:can_transform(stm) + end + local comprehension_has_value + comprehension_has_value = function(comp) + return is_value(comp[2]) + end + local value_is_singular + value_is_singular = function(node) + return type(node) ~= "table" or node[1] ~= "exp" or #node == 2 + end + local is_slice + is_slice = function(node) + return ntype(node) == "chain" and ntype(node[#node]) == "slice" + end + local t = { } + local node_types = { + class = { + { + "name", + "Tmp" + }, + { + "body", + t + } + }, + fndef = { + { + "args", + t + }, + { + "whitelist", + t + }, + { + "arrow", + "slim" + }, + { + "body", + t + } + }, + foreach = { + { + "names", + t + }, + { + "iter" + }, + { + "body", + t + } + }, + ["for"] = { + { + "name" + }, + { + "bounds", + t + }, + { + "body", + t + } + }, + ["while"] = { + { + "cond", + t + }, + { + "body", + t + } + }, + assign = { + { + "names", + t + }, + { + "values", + t + } + }, + declare = { + { + "names", + t + } + }, + ["if"] = { + { + "cond", + t + }, + { + "then", + t + } + } + } + local build_table + build_table = function() + local key_table = { } + for node_name, args in pairs(node_types) do + local index = { } + for i, tuple in ipairs(args) do + local prop_name = tuple[1] + index[prop_name] = i + 1 + end + key_table[node_name] = index end - return out + return key_table end - local tree - tree = function(tree, options) - if options == nil then - options = { } + local key_table = build_table() + local make_builder + make_builder = function(name) + local spec = node_types[name] + if not spec then + error("don't know how to build node: " .. name) end - assert(tree, "missing tree") - local scope = (options.scope or RootBlock)(options) - local runner = coroutine.create(function() - return scope:root_stms(tree) - end) - local success, err = coroutine.resume(runner) - if not (success) then - local error_msg, error_pos - if type(err) == "table" then - local _exp_0 = err[1] - if "user-error" == _exp_0 or "compile-error" == _exp_0 then - error_msg, error_pos = unpack(err, 2) + return function(props) + if props == nil then + props = { } + end + local node = { + name + } + for i, arg in ipairs(spec) do + local key, default_value = unpack(arg) + local val + if props[key] then + val = props[key] else - error_msg, error_pos = error("Unknown error thrown", util.dump(error_msg)) + val = default_value + end + if val == t then + val = { } end - else - error_msg, error_pos = concat({ - err, - debug.traceback(runner) - }, "\n") + node[i + 1] = val end - return nil, error_msg, error_pos or scope.last_pos + return node end - local lua_code = scope:render() - local posmap = scope._lines:flatten_posmap() - return lua_code, posmap end - do - local data = require("moonscript.data") - for name, cls in pairs({ - Line = Line, - Lines = Lines, - DelayedLine = DelayedLine - }) do - data[name] = cls + local build = nil + build = setmetatable({ + group = function(body) + if body == nil then + body = { } + end + return { + "group", + body + } + end, + ["do"] = function(body) + return { + "do", + body + } + end, + assign_one = function(name, value) + return build.assign({ + names = { + name + }, + values = { + value + } + }) + end, + table = function(tbl) + if tbl == nil then + tbl = { } + end + for _index_0 = 1, #tbl do + local tuple = tbl[_index_0] + if type(tuple[1]) == "string" then + tuple[1] = { + "key_literal", + tuple[1] + } + end + end + return { + "table", + tbl + } + end, + block_exp = function(body) + return { + "block_exp", + body + } + end, + chain = function(parts) + local base = parts.base or error("expecting base property for chain") + if type(base) == "string" then + base = { + "ref", + base + } + end + local node = { + "chain", + base + } + for _index_0 = 1, #parts do + local part = parts[_index_0] + insert(node, part) + end + return node + end + }, { + __index = function(self, name) + self[name] = make_builder(name) + return rawget(self, name) + end + }) + local smart_node_mt = setmetatable({ }, { + __index = function(self, node_type) + local index = key_table[node_type] + local mt = { + smart_node = true, + __index = function(node, key) + if index[key] then + return rawget(node, index[key]) + elseif type(key) == "string" then + return error("unknown key: `" .. key .. "` on node type: `" .. ntype(node) .. "`") + end + end, + __newindex = function(node, key, value) + if index[key] then + key = index[key] + end + return rawset(node, key, value) + end + } + self[node_type] = mt + return mt end + }) + local smart_node + smart_node = function(node) + return setmetatable(node, smart_node_mt[ntype(node)]) end return { - tree = tree, - value = value, - format_error = format_error, - Block = Block, - RootBlock = RootBlock + ntype = ntype, + smart_node = smart_node, + build = build, + is_value = is_value, + is_slice = is_slice, + manual_return = manual_return, + cascading = cascading, + value_is_singular = value_is_singular, + comprehension_has_value = comprehension_has_value, + has_value = has_value, + mtype = mtype } end -package.preload['moonscript.parse.util'] = function() - local unpack - unpack = require("moonscript.util").unpack - local P, C, S, Cp, Cmt, V +package.preload['moonscript.util'] = function() + local concat do - local _obj_0 = require("lpeg") - P, C, S, Cp, Cmt, V = _obj_0.P, _obj_0.C, _obj_0.S, _obj_0.Cp, _obj_0.Cmt, _obj_0.V + local _obj_0 = table + concat = _obj_0.concat end - local ntype - ntype = require("moonscript.types").ntype - local Space - Space = require("moonscript.parse.literals").Space - local Indent = C(S("\t ") ^ 0) / function(str) - do - local sum = 0 - for v in str:gmatch("[\t ]") do - local _exp_0 = v - if " " == _exp_0 then - sum = sum + 1 - elseif "\t" == _exp_0 then - sum = sum + 4 + local unpack = unpack or table.unpack + local type = type + local moon = { + is_object = function(value) + return type(value) == "table" and value.__class + end, + is_a = function(thing, t) + if not (type(thing) == "table") then + return false + end + local cls = thing.__class + while cls do + if cls == t then + return true + end + cls = cls.__parent + end + return false + end, + type = function(value) + local base_type = type(value) + if base_type == "table" then + local cls = value.__class + if cls then + return cls end end - return sum + return base_type + end + } + local pos_to_line + pos_to_line = function(str, pos) + local line = 1 + for _ in str:sub(1, pos):gmatch("\n") do + line = line + 1 end + return line end - local Cut = P(function() - return false - end) - local ensure - ensure = function(patt, finally) - return patt * finally + finally * Cut + local trim + trim = function(str) + return str:match("^%s*(.-)%s*$") end - local extract_line - extract_line = function(str, start_pos) - str = str:sub(start_pos) - do - local m = str:match("^(.-)\n") - if m then - return m + local get_line + get_line = function(str, line_num) + for line in str:gmatch("([^\n]*)\n?") do + if line_num == 1 then + return line end + line_num = line_num - 1 end - return str:match("^.-$") end - local show_line_position - show_line_position = function(str, pos, context) - if context == nil then - context = true + local get_closest_line + get_closest_line = function(str, line_num) + local line = get_line(str, line_num) + if (not line or trim(line) == "") and line_num > 1 then + return get_closest_line(str, line_num - 1) + else + return line, line_num end - local lines = { - { } - } - for c in str:gmatch(".") do - lines[#lines] = lines[#lines] or { } - table.insert(lines[#lines], c) - if c == "\n" then - lines[#lines + 1] = { } + end + local reversed + reversed = function(seq) + return coroutine.wrap(function() + for i = #seq, 1, -1 do + coroutine.yield(i, seq[i]) end + end) + end + local split + split = function(str, delim) + if str == "" then + return { } end - for i, line in ipairs(lines) do - lines[i] = table.concat(line) + str = str .. delim + local _accum_0 = { } + local _len_0 = 1 + for m in str:gmatch("(.-)" .. delim) do + _accum_0[_len_0] = m + _len_0 = _len_0 + 1 end - local out - local remaining = pos - 1 - for k, line in ipairs(lines) do - if remaining < #line then - local left = line:sub(1, remaining) - local right = line:sub(remaining + 1) - out = { - tostring(left) .. "◉" .. tostring(right) - } - if context then - do - local before = lines[k - 1] - if before then - table.insert(out, 1, before) - end - end - do - local after = lines[k + 1] - if after then - table.insert(out, after) - end + return _accum_0 + end + local dump + dump = function(what) + local seen = { } + local _dump + _dump = function(what, depth) + if depth == nil then + depth = 0 + end + local t = type(what) + if t == "string" then + return '"' .. what .. '"\n' + elseif t == "table" then + if seen[what] then + return "recursion(" .. tostring(what) .. ")...\n" + end + seen[what] = true + depth = depth + 1 + local lines + do + local _accum_0 = { } + local _len_0 = 1 + for k, v in pairs(what) do + _accum_0[_len_0] = (" "):rep(depth * 4) .. "[" .. tostring(k) .. "] = " .. _dump(v, depth) + _len_0 = _len_0 + 1 end + lines = _accum_0 end - break + seen[what] = false + return "{\n" .. concat(lines) .. (" "):rep((depth - 1) * 4) .. "}\n" else - remaining = remaining - #line - end - end - if not (out) then - return "-" - end - out = table.concat(out) - return (out:gsub("\n*$", "")) - end - local mark - mark = function(name) - return function(...) - return { - name, - ... - } - end - end - local pos - pos = function(patt) - return (Cp() * patt) / function(pos, value) - if type(value) == "table" then - value[-1] = pos + return tostring(what) .. "\n" end - return value end + return _dump(what) end - local got - got = function(what, context) - if context == nil then - context = true + local debug_posmap + debug_posmap = function(posmap, moon_code, lua_code) + local tuples + do + local _accum_0 = { } + local _len_0 = 1 + for k, v in pairs(posmap) do + _accum_0[_len_0] = { + k, + v + } + _len_0 = _len_0 + 1 + end + tuples = _accum_0 end - return Cmt("", function(str, pos) - print("++ got " .. tostring(what), "[" .. tostring(show_line_position(str, pos, context)) .. "]") - return true + table.sort(tuples, function(a, b) + return a[1] < b[1] end) - end - local flatten_or_mark - flatten_or_mark = function(name) - return function(tbl) - if #tbl == 1 then - return tbl[1] - end - table.insert(tbl, 1, name) - return tbl + local lines + do + local _accum_0 = { } + local _len_0 = 1 + for _index_0 = 1, #tuples do + local pair = tuples[_index_0] + local lua_line, pos = unpack(pair) + local moon_line = pos_to_line(moon_code, pos) + local lua_text = get_line(lua_code, lua_line) + local moon_text = get_closest_line(moon_code, moon_line) + local _value_0 = tostring(pos) .. "\t " .. tostring(lua_line) .. ":[ " .. tostring(trim(lua_text)) .. " ] >> " .. tostring(moon_line) .. ":[ " .. tostring(trim(moon_text)) .. " ]" + _accum_0[_len_0] = _value_0 + _len_0 = _len_0 + 1 + end + lines = _accum_0 end + return concat(lines, "\n") end - local is_assignable - do - local chain_assignable = { - index = true, - dot = true, - slice = true - } - is_assignable = function(node) - if node == "..." then - return false - end - local _exp_0 = ntype(node) - if "ref" == _exp_0 or "self" == _exp_0 or "value" == _exp_0 or "self_class" == _exp_0 or "table" == _exp_0 then - return true - elseif "chain" == _exp_0 then - return chain_assignable[ntype(node[#node])] - else - return false + local setfenv = setfenv or function(fn, env) + local name + local i = 1 + while true do + name = debug.getupvalue(fn, i) + if not name or name == "_ENV" then + break end + i = i + 1 end - end - local check_assignable - check_assignable = function(str, pos, value) - if is_assignable(value) then - return true, value - else - return false + if name then + debug.upvaluejoin(fn, i, (function() + return env + end), 1) end + return fn end - local format_assign - do - local flatten_explist = flatten_or_mark("explist") - format_assign = function(lhs_exps, assign) - if not (assign) then - return flatten_explist(lhs_exps) - end - for _index_0 = 1, #lhs_exps do - local assign_exp = lhs_exps[_index_0] - if not (is_assignable(assign_exp)) then - error({ - assign_exp, - "left hand expression is not assignable" - }) - end + local getfenv = getfenv or function(fn) + local i = 1 + while true do + local name, val = debug.getupvalue(fn, i) + if not (name) then + break end - local t = ntype(assign) - local _exp_0 = t - if "assign" == _exp_0 then - return { - "assign", - lhs_exps, - unpack(assign, 2) - } - elseif "update" == _exp_0 then - return { - "update", - lhs_exps[1], - unpack(assign, 2) - } - else - return error("unknown assign expression: " .. tostring(t)) + if name == "_ENV" then + return val end + i = i + 1 end + return nil end - local format_single_assign - format_single_assign = function(lhs, assign) - if assign then - return format_assign({ - lhs - }, assign) - else - return lhs - end - end - local sym - sym = function(chars) - return Space * chars - end - local symx - symx = function(chars) - return chars - end - local simple_string - simple_string = function(delim, allow_interpolation) - local inner = P("\\" .. tostring(delim)) + "\\\\" + (1 - P(delim)) - if allow_interpolation then - local interp = symx('#{') * V("Exp") * sym('}') - inner = (C((inner - interp) ^ 1) + interp / mark("interpolate")) ^ 0 + local get_options + get_options = function(...) + local count = select("#", ...) + local opts = select(count, ...) + if type(opts) == "table" then + return opts, unpack({ + ... + }, nil, count - 1) else - inner = C(inner ^ 0) - end - return C(symx(delim)) * inner * sym(delim) / mark("string") - end - local wrap_func_arg - wrap_func_arg = function(value) - return { - "call", - { - value - } - } - end - local join_chain - join_chain = function(callee, args) - if #args == 0 then - return callee - end - args = { - "call", - args - } - if ntype(callee) == "chain" then - table.insert(callee, args) - return callee - end - return { - "chain", - callee, - args - } - end - local wrap_decorator - wrap_decorator = function(stm, dec) - if not (dec) then - return stm + return { }, ... end - return { - "decorated", - stm, - dec - } - end - local check_lua_string - check_lua_string = function(str, pos, right, left) - return #left == #right - end - local self_assign - self_assign = function(name, pos) - return { - { - "key_literal", - name - }, - { - "ref", - name, - [-1] = pos - } - } end return { - Indent = Indent, - Cut = Cut, - ensure = ensure, - extract_line = extract_line, - mark = mark, - pos = pos, - flatten_or_mark = flatten_or_mark, - is_assignable = is_assignable, - check_assignable = check_assignable, - format_assign = format_assign, - format_single_assign = format_single_assign, - sym = sym, - symx = symx, - simple_string = simple_string, - wrap_func_arg = wrap_func_arg, - join_chain = join_chain, - wrap_decorator = wrap_decorator, - check_lua_string = check_lua_string, - self_assign = self_assign, - got = got, - show_line_position = show_line_position + moon = moon, + pos_to_line = pos_to_line, + get_closest_line = get_closest_line, + get_line = get_line, + reversed = reversed, + trim = trim, + split = split, + dump = dump, + debug_posmap = debug_posmap, + getfenv = getfenv, + setfenv = setfenv, + get_options = get_options, + unpack = unpack } end -package.preload['moonscript.parse.literals'] = function() - local safe_module - safe_module = require("moonscript.util").safe_module - local S, P, R, C - do - local _obj_0 = require("lpeg") - S, P, R, C = _obj_0.S, _obj_0.P, _obj_0.R, _obj_0.C - end - local lpeg = require("lpeg") - local L = lpeg.luversion and lpeg.L or function(v) - return #v - end - local White = S(" \t\r\n") ^ 0 - local plain_space = S(" \t") ^ 0 - local Break = P("\r") ^ -1 * P("\n") - local Stop = Break + -1 - local Comment = P("--") * (1 - S("\r\n")) ^ 0 * L(Stop) - local Space = plain_space * Comment ^ -1 - local SomeSpace = S(" \t") ^ 1 * Comment ^ -1 - local SpaceBreak = Space * Break - local EmptyLine = SpaceBreak - local AlphaNum = R("az", "AZ", "09", "__") - local Name = C(R("az", "AZ", "__") * AlphaNum ^ 0) - local Num = P("0x") * R("09", "af", "AF") ^ 1 * (S("uU") ^ -1 * S("lL") ^ 2) ^ -1 + R("09") ^ 1 * (S("uU") ^ -1 * S("lL") ^ 2) + (R("09") ^ 1 * (P(".") * R("09") ^ 1) ^ -1 + P(".") * R("09") ^ 1) * (S("eE") * P("-") ^ -1 * R("09") ^ 1) ^ -1 - local Shebang = P("#!") * P(1 - Stop) ^ 0 - return safe_module("moonscript.parse.literals", { - L = L, - White = White, - Break = Break, - Stop = Stop, - Comment = Comment, - Space = Space, - SomeSpace = SomeSpace, - SpaceBreak = SpaceBreak, - EmptyLine = EmptyLine, - AlphaNum = AlphaNum, - Name = Name, - Num = Num, - Shebang = Shebang - }) - -end -package.preload['moonscript.parse.env'] = function() - local getfenv, setfenv - do - local _obj_0 = require("moonscript.util") - getfenv, setfenv = _obj_0.getfenv, _obj_0.setfenv - end - local wrap_env - wrap_env = function(debug, fn) - local V, Cmt - do - local _obj_0 = require("lpeg") - V, Cmt = _obj_0.V, _obj_0.Cmt - end - local env = getfenv(fn) - local wrap_name = V - if debug then - local indent = 0 - local indent_char = " " - local iprint - iprint = function(...) - local args = table.concat((function(...) - local _accum_0 = { } - local _len_0 = 1 - local _list_0 = { - ... - } - for _index_0 = 1, #_list_0 do - local a = _list_0[_index_0] - _accum_0[_len_0] = tostring(a) - _len_0 = _len_0 + 1 - end - return _accum_0 - end)(...), ", ") - return io.stderr:write(tostring(indent_char:rep(indent)) .. tostring(args) .. "\n") - end - wrap_name = function(name) - local v = V(name) - v = Cmt("", function(str, pos) - local rest = str:sub(pos, -1):match("^([^\n]*)") - iprint("* " .. tostring(name) .. " (" .. tostring(rest) .. ")") - indent = indent + 1 - return true - end) * Cmt(v, function(str, pos, ...) - iprint(name, true) - indent = indent - 1 - return true, ... - end) + Cmt("", function() - iprint(name, false) - indent = indent - 1 - return false - end) - return v - end - end - return setfenv(fn, setmetatable({ }, { - __index = function(self, name) - local value = env[name] - if value ~= nil then - return value - end - if name:match("^[A-Z][A-Za-z0-9]*$") then - local v = wrap_name(name) - return v - end - return error("unknown variable referenced: " .. tostring(name)) - end - })) - end - return { - wrap_env = wrap_env - } +package.preload['moonscript.version'] = function() -end -package.preload['moonscript'] = function() - do - local _with_0 = require("moonscript.base") - return _with_0 - end + module("moonscript.version", package.seeall) -end -package.preload['moonscript.data'] = function() - local concat, remove, insert - do - local _obj_0 = table - concat, remove, insert = _obj_0.concat, _obj_0.remove, _obj_0.insert - end - local Set - Set = function(items) - local _tbl_0 = { } - for _index_0 = 1, #items do - local k = items[_index_0] - _tbl_0[k] = true - end - return _tbl_0 - end - local Stack - do - local _class_0 - local _base_0 = { - __tostring = function(self) - return "" - end, - pop = function(self) - return remove(self) - end, - push = function(self, value, ...) - insert(self, value) - if ... then - return self:push(...) - else - return value - end - end, - top = function(self) - return self[#self] - end - } - _base_0.__index = _base_0 - _class_0 = setmetatable({ - __init = function(self, ...) - self:push(...) - return nil - end, - __base = _base_0, - __name = "Stack" - }, { - __index = _base_0, - __call = function(cls, ...) - local _self_0 = setmetatable({}, _base_0) - cls.__init(_self_0, ...) - return _self_0 - end - }) - _base_0.__class = _class_0 - Stack = _class_0 + version = "0.2.5" + function print_version() + print("MoonScript version "..version) end - local lua_keywords = Set({ - 'and', - 'break', - 'do', - 'else', - 'elseif', - 'end', - 'false', - 'for', - 'function', - 'if', - 'in', - 'local', - 'nil', - 'not', - 'or', - 'repeat', - 'return', - 'then', - 'true', - 'until', - 'while' - }) - return { - Set = Set, - Stack = Stack, - lua_keywords = lua_keywords - } - -end -package.preload['moonscript.transform'] = function() - return { - Statement = require("moonscript.transform.statement"), - Value = require("moonscript.transform.value") - } end return package.preload["moonscript"]() diff --git a/automation/include/re.lua b/automation/include/re.lua index b5e8fc1581..64dbbd6455 100644 --- a/automation/include/re.lua +++ b/automation/include/re.lua @@ -1,4 +1,4 @@ --- Copyright (c) 2013, Thomas Goyne +-- Copyright (c) 2013, Thomas Goyne -- -- Permission to use, copy, modify, and distribute this software for any -- purpose with or without fee is hereby granted, provided that the above diff --git a/automation/include/unicode.lua b/automation/include/unicode.lua index a9e09debc1..caed5bbc08 100644 --- a/automation/include/unicode.lua +++ b/automation/include/unicode.lua @@ -1,4 +1,4 @@ --- Copyright (c) 2013, Thomas Goyne +-- Copyright (c) 2013, Thomas Goyne -- -- Permission to use, copy, modify, and distribute this software for any -- purpose with or without fee is hereby granted, provided that the above diff --git a/automation/include/utils-auto4.lua b/automation/include/utils-auto4.lua index 08e2e45cca..c28f645269 100644 --- a/automation/include/utils-auto4.lua +++ b/automation/include/utils-auto4.lua @@ -1,4 +1,4 @@ ---[[ +--[[ Copyright (c) 2005-2010, Niels Martin Hansen, Rodrigo Braz Monteiro All rights reserved. diff --git a/automation/include/utils.lua b/automation/include/utils.lua index f9b70157a0..053a5c0b25 100644 --- a/automation/include/utils.lua +++ b/automation/include/utils.lua @@ -1,4 +1,4 @@ ---[[ +--[[ Copyright (c) 2005-2006, Niels Martin Hansen, Rodrigo Braz Monteiro All rights reserved. diff --git a/src/audio_display.cpp b/src/audio_display.cpp index cdc942692e..fd7db878db 100644 --- a/src/audio_display.cpp +++ b/src/audio_display.cpp @@ -767,6 +767,15 @@ void AudioDisplay::ReloadRenderingSettings() spectrum_width[spectrum_quality], spectrum_distance[spectrum_quality]); + // Frequency curve + int64_t spectrum_freq_curve = OPT_GET("Audio/Renderer/Spectrum/FreqCurve")->GetInt(); + spectrum_freq_curve = mid(0, spectrum_freq_curve, 4); + const float spectrum_fref_pos[] = { 0.001f, 0.125f, 0.333f, 0.425f, 0.999f }; + + audio_spectrum_renderer->set_reference_frequency_position( + spectrum_fref_pos[spectrum_freq_curve] + ); + audio_renderer_provider = std::move(audio_spectrum_renderer); } else @@ -1236,6 +1245,7 @@ void AudioDisplay::OnAudioOpen(agi::AudioProvider *provider) OPT_SUB("Colour/Audio Display/Spectrum", &AudioDisplay::ReloadRenderingSettings, this), OPT_SUB("Colour/Audio Display/Waveform", &AudioDisplay::ReloadRenderingSettings, this), OPT_SUB("Audio/Renderer/Spectrum/Quality", &AudioDisplay::ReloadRenderingSettings, this), + OPT_SUB("Audio/Renderer/Spectrum/FreqCurve", &AudioDisplay::ReloadRenderingSettings, this), }); OnTimingController(); } diff --git a/src/audio_renderer_spectrum.cpp b/src/audio_renderer_spectrum.cpp index 563ffadc3d..b78ec139f3 100644 --- a/src/audio_renderer_spectrum.cpp +++ b/src/audio_renderer_spectrum.cpp @@ -100,6 +100,8 @@ AudioSpectrumRenderer::~AudioSpectrumRenderer() void AudioSpectrumRenderer::RecreateCache() { + update_derivation_values(); + #ifdef WITH_FFTW3 if (dft_plan) { @@ -143,20 +145,29 @@ void AudioSpectrumRenderer::OnSetProvider() void AudioSpectrumRenderer::SetResolution(size_t _derivation_size, size_t _derivation_dist) { - if (derivation_dist != _derivation_dist) + if (derivation_dist_user != _derivation_dist) { - derivation_dist = _derivation_dist; - if (cache) - cache->Age(0); + derivation_dist_user = _derivation_dist; + update_derivation_values(); + AgeCache (0); } - if (derivation_size != _derivation_size) + if (derivation_size_user != _derivation_size) { - derivation_size = _derivation_size; + derivation_size_user = _derivation_size; RecreateCache(); } } +void AudioSpectrumRenderer::set_reference_frequency_position(float pos_fref_) +{ + assert(pos_fref_ > 0.f); + assert(pos_fref_ < 1.f); + + pos_fref = pos_fref_; +} + + template void AudioSpectrumRenderer::ConvertToFloat(size_t count, T *dest) { for (size_t si = 0; si < count; ++si) @@ -165,6 +176,32 @@ void AudioSpectrumRenderer::ConvertToFloat(size_t count, T *dest) { } } +void AudioSpectrumRenderer::update_derivation_values() +{ + // Below this sampling rate (Hz), the derivation values are identical to + // the user-provided ones. Otherwise, they are scaled according to the + // ratio between the sampling rates. + // The threshold is set at 50 kHz so with standard rates like 48 kHz, + // the values are kept identical, and scaled with higher standard rates + // like 88.2 or 96 kHz. + constexpr float sample_rate_ref = 50000.f; + + derivation_dist = derivation_dist_user; + derivation_size = derivation_size_user; + + if (provider != nullptr) + { + const int sample_rate = provider->GetSampleRate(); + float mult = float(sample_rate) / sample_rate_ref; + while (mult > 1) + { + ++derivation_dist; + ++derivation_size; + mult *= 0.5f; + } + } +} + void AudioSpectrumRenderer::FillBlock(size_t block_index, float *block) { assert(cache); @@ -173,12 +210,19 @@ void AudioSpectrumRenderer::FillBlock(size_t block_index, float *block) int64_t first_sample = (((int64_t)block_index) << derivation_dist) - ((int64_t)1 << derivation_size); provider->GetInt16MonoAudio(audio_scratch.data(), first_sample, 2 << derivation_size); + // Because the FFTs used here are unnormalized DFTs, we have to compensate + // the possible length difference between derivation_size used in the + // calculations and its user-provided counterpart. Thus, the display is + // kept independent of the sampling rate. + const float scale_fix = + 1.f / sqrtf (float (1 << (derivation_size - derivation_size_user))); + #ifdef WITH_FFTW3 ConvertToFloat(2 << derivation_size, dft_input); fftw_execute(dft_plan); - double scale_factor = 9 / sqrt(2 << (derivation_size + 1)); + double scale_factor = scale_fix * 9 / sqrt(2 << (derivation_size + 1)); fftw_complex *o = dft_output; for (size_t si = (size_t)1< 0; --si) @@ -196,7 +240,7 @@ void AudioSpectrumRenderer::FillBlock(size_t block_index, float *block) FFT fft; fft.Transform(2< 0; --si) { @@ -211,6 +255,10 @@ void AudioSpectrumRenderer::FillBlock(size_t block_index, float *block) void AudioSpectrumRenderer::Render(wxBitmap &bmp, int start, AudioRenderingStyle style) { + // Misc. utility functions + auto floor_int = [](float val) { return int(floorf(val)); }; + auto round_int = [](float val) { return int(floorf(val + 0.5f)); }; + if (!cache) return; @@ -229,9 +277,34 @@ void AudioSpectrumRenderer::Render(wxBitmap &bmp, int start, AudioRenderingStyle const AudioColorScheme *pal = &colors[style]; - /// @todo Make minband and maxband configurable - int minband = 0; - int maxband = 1 << derivation_size; + // Sampling rate, in Hz. + const float sample_rate = float(provider->GetSampleRate()); + + // Number of FFT bins, excluding the "Nyquist" one + const int nbr_bins = 1 << derivation_size; + + // minband and maxband define an half-open range. + int minband = 1; // Starts at 1, we don't care about showing the DC. + int maxband = std::min( + round_int(nbr_bins * max_freq / (sample_rate * 0.5f)), + nbr_bins + ); + assert(minband < maxband); + + // Precomputes this once, this will be useful for the log curve. + const float scale_log = logf(maxband / minband); + + // Turns the user-specified 1 kHz position into a ratio between the linear + // and logarithmic curves that we can directly use in the following + // calculations. + assert(pos_fref > 0); + assert(pos_fref < 1); + float b_fref = nbr_bins * freq_ref / (sample_rate * 0.5f); + b_fref = mid(1.f, b_fref, float(maxband - 1)); + const float clin = minband + (maxband - minband) * pos_fref; + const float clog = minband * expf(pos_fref * scale_log); + float log_ratio_calc = (b_fref - clin) / (clog - clin); + log_ratio_calc = mid(0.f, log_ratio_calc, 1.f); // ax = absolute x, absolute to the virtual spectrum bitmap for (int ax = start; ax < end; ++ax) @@ -243,36 +316,51 @@ void AudioSpectrumRenderer::Render(wxBitmap &bmp, int start, AudioRenderingStyle // Prepare bitmap writing unsigned char *px = imgdata + (imgheight-1) * stride + (ax - start) * 3; - // Scale up or down vertically? - if (imgheight > 1<= imgdata); - assert(px < imgdata + imgheight*stride); - auto ideal = (double)(y+1.)/imgheight * (maxband-minband) + minband; - float sample1 = power[(int)floor(ideal)+minband]; - float sample2 = power[(int)ceil(ideal)+minband]; - float frac = ideal - floor(ideal); - float val = (1-frac)*sample1 + frac*sample2; - pal->map(val*amplitude_scale, px); - px -= stride; + // Bin index is an interpolation between the linear and log curves. + const float pos_rel = float(y + 1) / float(imgheight); + const float b_lin = minband + pos_rel * (maxband - minband); + const float b_log = minband * expf(pos_rel * scale_log); + bin_nxt = b_lin + log_ratio_calc * (b_log - b_lin); } - } - else - { - // Pick greatest - for (int y = 0; y < imgheight; ++y) + + float val = 0; + + // Interpolate between consecutive bins + if (bin_nxt - bin_prv < 2) + { + const int bin_0 = floor_int(bin_cur); + const int bin_1 = std::min(bin_0 + 1, nbr_bins - 1); + const float frac = bin_cur - float(bin_0); + const float v0 = power[bin_0]; + const float v1 = power[bin_1]; + val = v0 + frac * (v1 - v0); + } + + // Pick the greatest bin on the interval + else { - assert(px >= imgdata); - assert(px < imgdata + imgheight*stride); - int sample1 = std::max(0, maxband * y/imgheight + minband); - int sample2 = std::min((1<map(maxval*amplitude_scale, px); - px -= stride; + int bin_inf = floor_int((bin_prv + bin_cur) * 0.5f); + int bin_sup = floor_int((bin_cur + bin_nxt) * 0.5f); + bin_inf = std::min(bin_inf, nbr_bins - 2); + bin_sup = std::min(bin_sup, nbr_bins - 1); + assert(bin_inf < bin_sup); + val = *std::max_element(&power[bin_inf], &power[bin_sup]); } + + pal->map(val * amplitude_scale, px); + + px -= stride; + bin_prv = bin_cur; + bin_cur = bin_nxt; } } diff --git a/src/audio_renderer_spectrum.h b/src/audio_renderer_spectrum.h index d4641f37de..3bc9847770 100644 --- a/src/audio_renderer_spectrum.h +++ b/src/audio_renderer_spectrum.h @@ -61,10 +61,34 @@ class AudioSpectrumRenderer final : public AudioRendererBitmapProvider { /// Colour tables used for rendering std::vector colors; + /// User-provided value for derivation_size + size_t derivation_size_user = 0; + + /// User-provided value for derivation_dist + size_t derivation_dist_user = 0; + + /// Maximum audible, displayed frequency. Avoids wasting the display space + /// with ultrasonic content at sampling rates > 40 kHz. + float max_freq = 20000.f; + + /// Relative vertical position of the 1 kHz frequency, in (0 ; 1) open range + /// 0 = bottom of the display zone, 1 = top + /// The actual position, as displayed, is limited by the available mapping + /// curves (linear and log). + /// Values close to 0 will give a linear curve, and close to 1 a log curve. + float pos_fref = 1.0f / 3; + + /// Reference frequency which vertical position is constant, Hz. + const float freq_ref = 1000.0f; + /// Binary logarithm of number of samples to use in deriving frequency-power data + /// This could differ from the user-provided value because the actual value + /// used in computations may be scaled, depending on the sampling rate. size_t derivation_size = 0; /// Binary logarithm of number of samples between the start of derivations + /// This could differ from the user-provided value because the actual value + /// used in computations may be scaled, depending on the sampling rate. size_t derivation_dist = 0; /// @brief Reset in response to changing audio provider @@ -90,6 +114,9 @@ class AudioSpectrumRenderer final : public AudioRendererBitmapProvider { template void ConvertToFloat(size_t count, T *dest); + /// @brief Updates the derivation_* after a derivation_*_user change. + void update_derivation_values(); + #ifdef WITH_FFTW3 /// FFTW plan data fftw_plan dft_plan = nullptr; @@ -133,6 +160,12 @@ class AudioSpectrumRenderer final : public AudioRendererBitmapProvider { /// is specified too large, it will be clamped to the size. void SetResolution(size_t derivation_size, size_t derivation_dist); + /// @brief Set the vertical relative position of the reference frequency (1 kHz) + /// @param fref_pos_ Vertical position of the 1 kHz frequency. Between 0 and 1, boundaries excluded. + /// + /// A value close to 0 gives a linear display, and close to 1 a logarithmic display. + void set_reference_frequency_position(float pos_fref_); + /// @brief Cleans up the cache /// @param max_size Maximum size in bytes for the cache void AgeCache(size_t max_size) override; diff --git a/src/libresrc/default_config.json b/src/libresrc/default_config.json index 8a648fa47c..3ea90c8de6 100644 --- a/src/libresrc/default_config.json +++ b/src/libresrc/default_config.json @@ -71,7 +71,8 @@ "Spectrum" : { "Cutoff" : 0, "Memory Max" : 128, - "Quality" : 1 + "Quality" : 1, + "FreqCurve" : 0 } }, "Snap" : { diff --git a/src/libresrc/osx/default_config.json b/src/libresrc/osx/default_config.json index 82223cf484..4d55817884 100644 --- a/src/libresrc/osx/default_config.json +++ b/src/libresrc/osx/default_config.json @@ -71,7 +71,8 @@ "Spectrum" : { "Cutoff" : 0, "Memory Max" : 128, - "Quality" : 1 + "Quality" : 1, + "FreqCurve" : 0 } }, "Snap" : { diff --git a/src/preferences.cpp b/src/preferences.cpp index 1a315ed50a..79b5b2b932 100644 --- a/src/preferences.cpp +++ b/src/preferences.cpp @@ -394,6 +394,10 @@ void Advanced_Audio(wxTreebook *book, Preferences *parent) { wxArrayString sq_choice(4, sq_arr); p->OptionChoice(spectrum, _("Quality"), sq_choice, "Audio/Renderer/Spectrum/Quality"); + const wxString sc_arr[5] = { _("Linear"), _("Extended"), _("Medium"), _("Compressed"), _("Logarithmic") }; + wxArrayString sc_choice(5, sc_arr); + p->OptionChoice(spectrum, _("Frequency mapping"), sc_choice, "Audio/Renderer/Spectrum/FreqCurve"); + p->OptionAdd(spectrum, _("Cache memory max (MB)"), "Audio/Renderer/Spectrum/Memory Max", 2, 1024); #ifdef WITH_AVISYNTH diff --git a/src/video_provider_avs.cpp b/src/video_provider_avs.cpp index 7c779e800e..77583f39ef 100644 --- a/src/video_provider_avs.cpp +++ b/src/video_provider_avs.cpp @@ -224,6 +224,19 @@ AVSValue AvisynthVideoProvider::Open(agi::fs::path const& filename) { return env->Invoke("Import", videoFilename); } + // Try L-SMASH-Works + // Load LSMASHSource.dll from app dir if it exists + auto lsmashworks_path(config::path->Decode("?data/LSMASHSource.dll")); + if (agi::fs::FileExists(lsmashworks_path)) + env->Invoke("LoadPlugin", env->SaveString(agi::fs::ShortName(lsmashworks_path).c_str())); + + // Then try using L-SMASH-Works + if (env->FunctionExists("LWLibavVideoSource")) { + decoder_name = "Avisynth/LWLibavVideoSource"; + LOG_I("avisynth/video") << "Opening file with LWLibavVideoSource"; + return env->Invoke("LWLibavVideoSource", videoFilename); + } + // Open avi file with AviSource if (agi::fs::HasExtension(filename, "avi")) { LOG_I("avisynth/video") << "Opening .avi file with AviSource";