Skip to content

Commit 324eae6

Browse files
committed
feat: Initial commit
0 parents  commit 324eae6

18 files changed

+1218
-0
lines changed

.github/workflows/push.yml

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
name: CI
2+
3+
on: push
4+
5+
jobs:
6+
release:
7+
runs-on: ubuntu-latest
8+
steps:
9+
- uses: actions/checkout@v3
10+
- uses: actions/setup-go@v3
11+
- uses: actions/setup-node@v3
12+
- run: yarn install
13+
- run: make build
14+
- run: yarn test
15+
- name: release
16+
if: github.ref == 'refs/heads/main'
17+
run: npx --yes semantic-release --branches main
18+
env:
19+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
20+
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
main.wasm
2+
wasm_exec.js
3+
node_modules/
4+
types/

.vscode/settings.json

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"gopls": {
3+
"build.env": {
4+
"GOOS": "js",
5+
"GOARCH": "wasm"
6+
}
7+
}
8+
}

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2022 XING Developers
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.

Makefile

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
build: wasm_exec.js main.wasm types/node.d.mts
2+
3+
wasm_exec.js:
4+
cp $$(go env GOROOT)/misc/wasm/wasm_exec.js wasm_exec.tmp.js
5+
echo "// @ts-nocheck" > wasm_exec.js
6+
cat wasm_exec.tmp.js >> wasm_exec.js
7+
rm wasm_exec.tmp.js
8+
9+
main.wasm: main.go go.mod
10+
GOOS=js GOARCH=wasm go build -o main.wasm
11+
12+
types/node.d.mts: *.cjs *.mjs *.d.ts *.json yarn.lock
13+
$$(yarn bin)/tsc -p .
14+
15+
clean:
16+
rm main.wasm
17+
rm wasm_exec.js
18+
rm -rf types
19+
20+
.PHONY: build clean

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# actionlint
2+
Actionlint as wasm

actionlint.cjs

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
require("./wasm_exec.js");
2+
3+
/**
4+
* @typedef {(go: Go) => Promise<WebAssembly.WebAssemblyInstantiatedSource>} WasmLoader
5+
* @typedef {(source: string, path: string) => Promise<LintResult[]>} RunActionlint
6+
*
7+
* @typedef {Object} LintResult
8+
* @property {string} Message
9+
* @property {string} Filepath
10+
* @property {number} Line
11+
* @property {number} Column
12+
* @property {string} Kind
13+
*/
14+
15+
/**
16+
* @param {WasmLoader} loader
17+
* @returns {RunActionlint}
18+
*/
19+
module.exports.createActionlint = function createActionlint(loader) {
20+
const go = new Go();
21+
22+
/** @type {(() => void)[] | undefined} */
23+
let queued = undefined;
24+
25+
// This function gets called from go once the wasm module is ready and it
26+
// executes the linter for all queued calls.
27+
globalThis.actionlintInitialized = () => {
28+
queued?.forEach((f) => f());
29+
queued = globalThis.actionlintInitialized = undefined;
30+
};
31+
32+
loader(go).then((wasm) => {
33+
// Do not await this promise, because it only resolves once the go main()
34+
// function has exited. But we need the main function to stay alive to be
35+
// able to call the `runActionlint` function.
36+
go.run(wasm.instance);
37+
});
38+
39+
/**
40+
* @param {string} src
41+
* @param {string} path
42+
* @returns {Promise<LintResult[]>}
43+
*/
44+
return async function runLint(src, path) {
45+
// Return a promise, because we need to queue calls to `runLint()` while the
46+
// wasm module is still loading and execute them once the wasm module is
47+
//ready.
48+
return new Promise((resolve, reject) => {
49+
if (typeof runActionlint === "function") {
50+
const [result, err] = runActionlint(src, path);
51+
return err ? reject(err) : resolve(result);
52+
}
53+
54+
if (!queued) {
55+
queued = [];
56+
}
57+
58+
queued.push(() => {
59+
const [result, err] = runActionlint?.(src, path) ?? [
60+
[],
61+
new Error('"runActionlint" is not defined'),
62+
];
63+
return err ? reject(err) : resolve(result);
64+
});
65+
});
66+
};
67+
};

browser.mjs

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { createActionlint } from "./actionlint.cjs";
2+
3+
/**
4+
* @typedef {import("./actionlint.cjs").LintResult} LintResult
5+
* @typedef {import("./actionlint.cjs").WasmLoader} WasmLoader
6+
* @typedef {import("./actionlint.cjs").RunActionlint} RunActionlint
7+
*/
8+
9+
/** @type {RunActionlint | undefined} */
10+
let runLint = undefined;
11+
12+
/**
13+
* @param {URL} url
14+
* @returns {RunActionlint}
15+
*/
16+
export function createLinter(url = new URL("./main.wasm", import.meta.url)) {
17+
if (runLint) {
18+
return runLint;
19+
}
20+
21+
return (runLint = createActionlint(
22+
/** @type {WasmLoader} */ async (go) => {
23+
return WebAssembly.instantiateStreaming(
24+
fetch(url.toString()),
25+
go.importObject
26+
);
27+
}
28+
));
29+
}

globals.d.ts

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
export declare global {
2+
var runActionlint:
3+
| ((src: string, path: string) => [LintResult[], Error | null])
4+
| undefined;
5+
var actionlintInitialized: (() => void) | undefined;
6+
}

go.mod

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
module source.xing.com/fea/act-app
2+
3+
go 1.17
4+
5+
require (
6+
github.com/fatih/color v1.13.0 // indirect
7+
github.com/mattn/go-colorable v0.1.11 // indirect
8+
github.com/mattn/go-isatty v0.0.14 // indirect
9+
github.com/mattn/go-runewidth v0.0.13 // indirect
10+
github.com/rhysd/actionlint v1.6.9 // indirect
11+
github.com/rivo/uniseg v0.2.0 // indirect
12+
github.com/robfig/cron v1.2.0 // indirect
13+
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c // indirect
14+
golang.org/x/sys v0.0.0-20211113001501-0c823b97ae02 // indirect
15+
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect
16+
)

go.sum

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
github.com/fatih/color v1.13.0 h1:8LOYc1KYPPmyKMuN8QV2DNRWNbLo6LZ0iLs8+mlH53w=
2+
github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk=
3+
github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
4+
github.com/mattn/go-colorable v0.1.11 h1:nQ+aFkoE2TMGc0b68U2OKSexC+eq46+XwZzWXHRmPYs=
5+
github.com/mattn/go-colorable v0.1.11/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4=
6+
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
7+
github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y=
8+
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
9+
github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU=
10+
github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
11+
github.com/rhysd/actionlint v1.6.9 h1:8rQQ76o88zctUCzukt0A5O/FO003wTGbkLQuwQkMf9c=
12+
github.com/rhysd/actionlint v1.6.9/go.mod h1:0AA4pvZ2nrZHT6D86eUhieH2NFmLqhxrNex0NEa2A2g=
13+
github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY=
14+
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
15+
github.com/robfig/cron v1.2.0 h1:ZjScXvvxeQ63Dbyxy76Fj3AT3Ut0aKsyd2/tl3DTMuQ=
16+
github.com/robfig/cron v1.2.0/go.mod h1:JGuDeoQd7Z6yL4zQhZ3OPEVHB7fL6Ka6skscFHfmt2k=
17+
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ=
18+
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
19+
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
20+
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
21+
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
22+
golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
23+
golang.org/x/sys v0.0.0-20211113001501-0c823b97ae02 h1:7NCfEGl0sfUojmX78nK9pBJuUlSZWEJA/TwASvfiPLo=
24+
golang.org/x/sys v0.0.0-20211113001501-0c823b97ae02/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
25+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
26+
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo=
27+
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=

main.go

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
package main
2+
3+
import (
4+
"io/ioutil"
5+
"reflect"
6+
"syscall/js"
7+
8+
"github.com/rhysd/actionlint"
9+
)
10+
11+
func toMap(input interface{}) map[string]interface{} {
12+
out := make(map[string]interface{})
13+
value := reflect.ValueOf(input)
14+
if value.Kind() == reflect.Ptr {
15+
value = value.Elem()
16+
}
17+
18+
for i := 0; i < value.NumField(); i++ {
19+
out[value.Type().Field(i).Name] = value.Field(i).Interface()
20+
}
21+
22+
return out
23+
}
24+
25+
func runActionlint(source string, filePath string) (interface{}, error) {
26+
opts := actionlint.LinterOptions{}
27+
linter, err := actionlint.NewLinter(ioutil.Discard, &opts)
28+
if err != nil {
29+
return nil, err
30+
}
31+
32+
errs, err := linter.Lint(filePath, []byte(source), nil)
33+
if err != nil {
34+
return nil, err
35+
}
36+
37+
ret := make([]interface{}, 0, len(errs))
38+
for _, err := range errs {
39+
ret = append(ret, toMap(*err))
40+
}
41+
42+
return ret, nil
43+
}
44+
45+
func main() {
46+
js.Global().Set("runActionlint", js.FuncOf(func(this js.Value, args []js.Value) interface{} {
47+
result, err := runActionlint(args[0].String(), args[1].String())
48+
return js.Global().Get("Array").New(result, err)
49+
}))
50+
51+
js.Global().Call("actionlintInitialized")
52+
53+
select {}
54+
}

node.cjs

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
const { join } = require("path");
2+
const { pathToFileURL } = require("url");
3+
const { readFile } = require("node:fs/promises");
4+
const { createActionlint } = require("./actionlint.cjs");
5+
6+
/**
7+
* @typedef {import("./actionlint.cjs").LintResult} LintResult
8+
* @typedef {import("./actionlint.cjs").WasmLoader} WasmLoader
9+
* @typedef {import("./actionlint.cjs").RunActionlint} RunActionlint
10+
*/
11+
12+
/** @type {RunActionlint | undefined} */
13+
let runLint = undefined;
14+
15+
/**
16+
* @param {URL} url
17+
* @returns {RunActionlint}
18+
*/
19+
module.exports.createLinter = function createLinter(
20+
url = pathToFileURL(join(__dirname, "main.wasm"))
21+
) {
22+
if (runLint) {
23+
return runLint;
24+
}
25+
26+
return (runLint = createActionlint(
27+
/** @type {WasmLoader} */ async (go) => {
28+
return WebAssembly.instantiate(await readFile(url), go.importObject);
29+
}
30+
));
31+
};

node.mjs

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
import { readFile } from "node:fs/promises";
2+
import { createActionlint } from "./actionlint.cjs";
3+
4+
/**
5+
* @typedef {import("./actionlint.cjs").LintResult} LintResult
6+
* @typedef {import("./actionlint.cjs").WasmLoader} WasmLoader
7+
* @typedef {import("./actionlint.cjs").RunActionlint} RunActionlint
8+
*/
9+
10+
/** @type {RunActionlint | undefined} */
11+
let runLint = undefined;
12+
13+
/**
14+
* @param {URL} url
15+
* @returns {RunActionlint}
16+
*/
17+
export function createLinter(url = new URL("./main.wasm", import.meta.url)) {
18+
if (runLint) {
19+
return runLint;
20+
}
21+
22+
return (runLint = createActionlint(
23+
/** @type {WasmLoader} */ async (go) => {
24+
return WebAssembly.instantiate(await readFile(url), go.importObject);
25+
}
26+
));
27+
}

package.json

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
{
2+
"name": "actionlint",
3+
"version": "0.0.0",
4+
"license": "MIT",
5+
"sideEffects": true,
6+
"scripts": {
7+
"test": "tape test.mjs | tap-spec"
8+
},
9+
"main": "./node.cjs",
10+
"exports": {
11+
"types": "./types/node.d.mts",
12+
"node": {
13+
"import": "./node.mjs",
14+
"require": "./node.cjs"
15+
},
16+
"browser": "./browser.mjs"
17+
},
18+
"devDependencies": {
19+
"@types/golang-wasm-exec": "^1.15.0",
20+
"@types/node": "^17.0.21",
21+
"@types/tape": "^4.13.2",
22+
"tap-spec": "^5.0.0",
23+
"tape": "^5.5.2",
24+
"typescript": "^4.6.2"
25+
}
26+
}

0 commit comments

Comments
 (0)