mirror of
https://github.com/Open-Argon/argon-v3.git
synced 2025-12-06 00:46:07 +00:00
Compare commits
34 Commits
master
...
master-fai
| Author | SHA1 | Date | |
|---|---|---|---|
| 2543822418 | |||
| afbf868703 | |||
|
|
1c048579a5 | ||
| 199b2320dd | |||
| 857eb442c2 | |||
|
|
5ceff0ddd2 | ||
| 2d3c7c42ce | |||
| 90b506d22f | |||
| f9777611dd | |||
| bc3e50697f | |||
| 6226cc52ac | |||
| c881d06cb5 | |||
| 584841030c | |||
| 58c57f154b | |||
| c698b878da | |||
| 280635d46e | |||
| e05267c886 | |||
| 4c412a763a | |||
| de95c3e8b5 | |||
| d0d73db6cf | |||
| aea3b73e71 | |||
|
|
e06d28ca02 | ||
| bf145db749 | |||
| 0514783230 | |||
| 7ef9384d76 | |||
| 957029ebd4 | |||
| afc5ae2086 | |||
| a6c14b49a5 | |||
| 1e625a589a | |||
| 5e65ff01ac | |||
| 5e48a70c77 | |||
|
|
69faa23ea9 | ||
|
|
126d287611 | ||
| 97ec428aca |
2
build
2
build
@@ -1 +1 @@
|
||||
go build -o bin/argon ./src
|
||||
GOOS=js GOARCH=wasm go build -o wasm/bin/argon.wasm ./src
|
||||
@@ -1,2 +1,4 @@
|
||||
@echo off
|
||||
go build -o bin/argon.exe ./src
|
||||
set GOOS=js
|
||||
set GOARCH=wasm
|
||||
go build -o wasm/bin/argon.wasm ./src
|
||||
2
merge
Executable file
2
merge
Executable file
@@ -0,0 +1,2 @@
|
||||
git fetch upstream
|
||||
git merge upstream/master master
|
||||
2
merge.bat
Normal file
2
merge.bat
Normal file
@@ -0,0 +1,2 @@
|
||||
git fetch upstream
|
||||
git merge upstream/master master
|
||||
30
readme.md
30
readme.md
@@ -2,23 +2,37 @@
|
||||
<p>
|
||||
<img width="150" src="logos/ArLogo.png">
|
||||
</p>
|
||||
<h1>Argon v3</h1>
|
||||
<h1>Argon v3.wasm</h1>
|
||||
</div>
|
||||
|
||||
### HEADS UP! 🙂
|
||||
|
||||
Argon v3.wasm is a fork of Argon v3 which has had parts rewritten for support in WebAssembly (WASM).
|
||||
|
||||
It includes all the same features and functionality as Argon v3, but has been compiled to run efficiently on WebAssembly.
|
||||
|
||||
Please note that Argon v3.wasm can only be compiled for use with WebAssembly.
|
||||
|
||||
|
||||
---
|
||||
|
||||
ARGON 3 is a math-driven programming language designed to make code easy to read and write. It's not meant to be fast, as it's interpreted. This specification should be used as a guideline, and is subject to change for later versions. Later updates for Argon 3 should be backwards compatible (where possible) with code designed for older versions of the interpreter.
|
||||
|
||||
## 📚 Features
|
||||
- Easy to read and write: Argon 3 is designed with clarity of code in mind, making it easier for you and others to read and write code.
|
||||
- All numbers are stored as rational numbers, preventing precision errors.
|
||||
- Math-driven: Designed for mathematical computations, Argon 3 uses techniques and rules set in maths. It's designed to be easy for mathematicians to write and understand algorithms in.
|
||||
- Interpreted: Argon 3 is an interpreted language, so you don't need to compile your code before running it.
|
||||
- Cross-platform: Argon 3 can be run on any platform that has an interpreter for it.
|
||||
- Lightweight: The Argon 3 interpreter is small and doesn't require a lot of system resources to run.
|
||||
|
||||
- Easy to read and write: Argon 3 is designed with clarity of code in mind, making it easier for you and others to read and write code.
|
||||
- All numbers are stored as rational numbers, preventing precision errors.
|
||||
- Math-driven: Designed for mathematical computations, Argon 3 uses techniques and rules set in maths. It's designed to be easy for mathematicians to write and understand algorithms in.
|
||||
- Interpreted: Argon 3 is an interpreted language, so you don't need to compile your code before running it.
|
||||
- Cross-platform: Argon 3 can be run on any platform that has an interpreter for it.
|
||||
- Lightweight: The Argon 3 interpreter is small and doesn't require a lot of system resources to run.
|
||||
|
||||
## 💻 Installation
|
||||
|
||||
As of now, Argon 3 does not have an installer. Feel free to clone this repo and run the `build` file for your plateform. the build will be found in `bin/argon(.exe)`.
|
||||
|
||||
## 📖 Usage
|
||||
|
||||
To use Argon 3, you can create a file with the .ar extension and write your code in it. Then, you can run your code using the interpreter. For example, if you have a file called example.ar, you can run it using the following command:
|
||||
|
||||
```
|
||||
@@ -38,7 +52,7 @@ f(x) = x^2 + 2*x + 1
|
||||
term.log('f(10) =', f(10))
|
||||
```
|
||||
|
||||
This code defines a function f(x) that calculates x^2 + 2*x + 1. It then calls the function with an argument of 10 and logs the result to the console.
|
||||
This code defines a function f(x) that calculates x^2 + 2\*x + 1. It then calls the function with an argument of 10 and logs the result to the console.
|
||||
|
||||
Please note that this example is subject to change as the specification is in beta and may be updated frequently.
|
||||
|
||||
|
||||
2
run
2
run
@@ -1,2 +0,0 @@
|
||||
# run the go run command passing the path to the main.go file, with the working directory set to the bin folder. pass in the arguments
|
||||
go run ./src "$@"
|
||||
5
run.bat
5
run.bat
@@ -1,5 +0,0 @@
|
||||
@echo off
|
||||
|
||||
:: run the go run command passing the path to the main.go file, with the working directory set to the bin folder. pass in the arguments
|
||||
|
||||
go run ./src %*
|
||||
@@ -533,7 +533,7 @@ func ArArray(arr []any) ArObject {
|
||||
},
|
||||
}
|
||||
val.obj["__Equal__"] = builtinFunc{
|
||||
"__Equal__",
|
||||
"__LessThanEqual__",
|
||||
func(args ...any) (any, ArErr) {
|
||||
if len(args) != 1 {
|
||||
return nil, ArErr{
|
||||
|
||||
@@ -5,10 +5,14 @@ import (
|
||||
"os"
|
||||
)
|
||||
|
||||
func makeGlobal() ArObject {
|
||||
func makeGlobal(allowDocument bool) ArObject {
|
||||
var vars = anymap{}
|
||||
vars["global"] = vars
|
||||
vars["term"] = ArTerm
|
||||
if allowDocument {
|
||||
vars["document"] = ArDocument
|
||||
}
|
||||
vars["js"] = ArJS
|
||||
vars["number"] = builtinFunc{"number", ArgonNumber}
|
||||
vars["string"] = builtinFunc{"string", ArgonString}
|
||||
vars["infinity"] = infinity
|
||||
|
||||
@@ -4,6 +4,6 @@ var websiteLang = "https://argon.wbell.dev/"
|
||||
var docs = "https://argon.wbell.dev/docs/"
|
||||
var mainrepo = "https://github.com/Open-Argon/argon-v3"
|
||||
var mainissuesPage = "https://github.com/Open-Argon/argon-v3/issues"
|
||||
var fork = false
|
||||
var forkrepo = ""
|
||||
var forkissuesPage = ""
|
||||
var fork = true
|
||||
var forkrepo = "https://github.com/Open-Argon/argon-v3.wasm"
|
||||
var forkissuesPage = "https://github.com/Open-Argon/argon-v3.wasm/issues"
|
||||
|
||||
213
src/document.go
Normal file
213
src/document.go
Normal file
@@ -0,0 +1,213 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"syscall/js"
|
||||
)
|
||||
|
||||
func windowElement(element js.Value) ArObject {
|
||||
return ArObject{
|
||||
obj: anymap{
|
||||
"innerHTML": builtinFunc{"innerHTML", func(args ...any) (any, ArErr) {
|
||||
if len(args) > 0 {
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "innerHTML only accepts strings", 0, "", "", true}
|
||||
}
|
||||
element.Set("innerHTML", args[0].(string))
|
||||
}
|
||||
return element.Get("innerHTML").String(), ArErr{}
|
||||
}},
|
||||
"innerText": builtinFunc{"innerText", func(args ...any) (any, ArErr) {
|
||||
if len(args) > 0 {
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "innerText only accepts strings", 0, "", "", true}
|
||||
}
|
||||
element.Set("innerText", args[0].(string))
|
||||
}
|
||||
return element.Get("innerText").String(), ArErr{}
|
||||
}},
|
||||
"addEventListener": builtinFunc{"addEventListener", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 2 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for addEventListener", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "addEventListener's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
event := args[0].(string)
|
||||
if typeof(args[1]) != "function" {
|
||||
return nil, ArErr{"Argument Error", "addEventListener's second argument must be a function", 0, "", "", true}
|
||||
}
|
||||
callable := args[1]
|
||||
element.Call("addEventListener", event, js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
runCall(call{
|
||||
callable: callable,
|
||||
args: []any{},
|
||||
}, stack{}, 0)
|
||||
return nil
|
||||
}))
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"removeEventListener": builtinFunc{"removeEventListener", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 2 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for removeEventListener", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "removeEventListener's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
event := args[0].(string)
|
||||
if typeof(args[1]) != "function" {
|
||||
return nil, ArErr{"Argument Error", "removeEventListener's second argument must be a function", 0, "", "", true}
|
||||
}
|
||||
callable := args[1]
|
||||
element.Call("removeEventListener", event, js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
runCall(call{
|
||||
callable: callable,
|
||||
args: []any{},
|
||||
}, stack{}, 0)
|
||||
return nil
|
||||
}))
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"appendChild": builtinFunc{"appendChild", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 1 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for appendChild", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "map" {
|
||||
return nil, ArErr{"Argument Error", "appendChild's first argument must be a map", 0, "", "", true}
|
||||
}
|
||||
child := args[0].(anymap)
|
||||
if child["__TYPE__"] != "windowElement" {
|
||||
return nil, ArErr{"Argument Error", "appendChild's first argument must be an element", 0, "", "", true}
|
||||
}
|
||||
element.Call("appendChild", child["__element__"])
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"removeChild": builtinFunc{"removeChild", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 1 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for removeChild", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "map" {
|
||||
return nil, ArErr{"Argument Error", "removeChild's first argument must be a map", 0, "", "", true}
|
||||
}
|
||||
child := args[0].(anymap)
|
||||
if child["__TYPE__"] != "windowElement" {
|
||||
return nil, ArErr{"Argument Error", "removeChild's first argument must be an element", 0, "", "", true}
|
||||
}
|
||||
element.Call("removeChild", child["__element__"])
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"setAttribute": builtinFunc{"setAttribute", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 2 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for setAttribute", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "setAttribute's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
element.Call("setAttribute", args[0].(string), anyToArgon(args[1], false, false, 3, 0, false, 0))
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"__element__": element,
|
||||
"__TYPE__": "windowElement",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
var ArDocument = Map(
|
||||
anymap{
|
||||
"body": builtinFunc{"getElementById", func(args ...any) (any, ArErr) {
|
||||
return windowElement(js.Global().Get("document").Get("body")), ArErr{}
|
||||
}},
|
||||
"head": builtinFunc{"getElementById", func(args ...any) (any, ArErr) {
|
||||
return windowElement(js.Global().Get("document").Get("head")), ArErr{}
|
||||
}},
|
||||
"getElementById": builtinFunc{"getElementById", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 1 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for getElementById", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "getElementById's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
id := args[0].(string)
|
||||
result := js.Global().Get("document").Call("getElementById", id)
|
||||
if js.Null().Equal(result) {
|
||||
return nil, ArErr{}
|
||||
}
|
||||
return windowElement(result), ArErr{}
|
||||
}},
|
||||
"createElement": builtinFunc{"createElement", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 1 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for createElement", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "createElement's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
tag := args[0].(string)
|
||||
return windowElement(js.Global().Get("document").Call("createElement", tag)), ArErr{}
|
||||
}},
|
||||
"createTextNode": builtinFunc{"createTextNode", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 1 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for createTextNode", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "createTextNode's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
text := args[0].(string)
|
||||
return windowElement(js.Global().Get("document").Call("createTextNode", text)), ArErr{}
|
||||
}},
|
||||
"createComment": builtinFunc{"createComment", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 1 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for createComment", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "createComment's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
text := args[0].(string)
|
||||
return windowElement(js.Global().Get("document").Call("createComment", text)), ArErr{}
|
||||
}},
|
||||
"createDocumentFragment": builtinFunc{"createDocumentFragment", func(args ...any) (any, ArErr) {
|
||||
return windowElement(js.Global().Get("document").Call("createDocumentFragment")), ArErr{}
|
||||
}},
|
||||
"addEventListener": builtinFunc{"addEventListener", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 2 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for addEventListener", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "addEventListener's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
event := args[0].(string)
|
||||
if typeof(args[1]) != "function" {
|
||||
return nil, ArErr{"Argument Error", "addEventListener's second argument must be a function", 0, "", "", true}
|
||||
}
|
||||
callable := args[1]
|
||||
js.Global().Get("document").Call("addEventListener", event, js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
runCall(call{
|
||||
callable: callable,
|
||||
args: []any{},
|
||||
}, stack{}, 0)
|
||||
return nil
|
||||
}))
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"removeEventListener": builtinFunc{"removeEventListener", func(args ...any) (any, ArErr) {
|
||||
if len(args) < 2 {
|
||||
return nil, ArErr{"Argument Error", "Not enough arguments for removeEventListener", 0, "", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "string" {
|
||||
return nil, ArErr{"Argument Error", "removeEventListener's first argument must be a string", 0, "", "", true}
|
||||
}
|
||||
event := args[0].(string)
|
||||
if typeof(args[1]) != "function" {
|
||||
return nil, ArErr{"Argument Error", "removeEventListener's second argument must be a function", 0, "", "", true}
|
||||
}
|
||||
callable := args[1]
|
||||
js.Global().Get("document").Call("removeEventListener", event, js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
runCall(call{
|
||||
callable: callable,
|
||||
args: []any{},
|
||||
}, stack{}, 0)
|
||||
return nil
|
||||
}))
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"__TYPE__": "document",
|
||||
},
|
||||
)
|
||||
@@ -2,6 +2,8 @@ package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
|
||||
"github.com/jwalton/go-supportscolor"
|
||||
)
|
||||
|
||||
type ArErr struct {
|
||||
@@ -19,5 +21,9 @@ func panicErr(err ArErr) {
|
||||
fmt.Println(" " + err.code)
|
||||
fmt.Println()
|
||||
}
|
||||
fmt.Printf("\x1b[%dm%s\x1b[0m", 91, fmt.Sprint(err.TYPE, ": ", err.message, "\n"))
|
||||
if supportscolor.Stdout().SupportsColor {
|
||||
fmt.Printf("\x1b[%dm%s\x1b[0m", 91, fmt.Sprint(err.TYPE, ": ", err.message, "\n"))
|
||||
} else {
|
||||
fmt.Println(fmt.Sprint(err.TYPE, ": ", err.message))
|
||||
}
|
||||
}
|
||||
|
||||
188
src/import.go
188
src/import.go
@@ -5,7 +5,6 @@ import (
|
||||
"errors"
|
||||
"log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
var imported = make(map[string]ArObject)
|
||||
@@ -48,101 +47,108 @@ func readFile(path string) []UNPARSEcode {
|
||||
}
|
||||
|
||||
func importMod(realpath string, origin string, main bool, global ArObject) (ArObject, ArErr) {
|
||||
extention := filepath.Ext(realpath)
|
||||
path := realpath
|
||||
if extention == "" {
|
||||
path += ".ar"
|
||||
return ArObject{}, ArErr{
|
||||
TYPE: "Import Error",
|
||||
message: "importing in WASM is currently not supported",
|
||||
EXISTS: true,
|
||||
}
|
||||
ex, err := os.Getwd()
|
||||
if err != nil {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "Could not get working directory", EXISTS: true}
|
||||
}
|
||||
exc, err := os.Executable()
|
||||
if err != nil {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "Could not get executable", EXISTS: true}
|
||||
}
|
||||
executable := filepath.Dir(exc)
|
||||
isABS := filepath.IsAbs(path)
|
||||
var pathsToTest []string
|
||||
if isABS {
|
||||
pathsToTest = []string{
|
||||
filepath.Join(path),
|
||||
filepath.Join(realpath, "init.ar"),
|
||||
/*
|
||||
extention := filepath.Ext(realpath)
|
||||
path := realpath
|
||||
if extention == "" {
|
||||
path += ".ar"
|
||||
}
|
||||
} else {
|
||||
pathsToTest = []string{
|
||||
filepath.Join(origin, realpath, "init.ar"),
|
||||
filepath.Join(origin, path),
|
||||
filepath.Join(origin, "modules", path),
|
||||
filepath.Join(origin, "modules", realpath, "init.ar"),
|
||||
filepath.Join(ex, path),
|
||||
filepath.Join(ex, "modules", realpath, "init.ar"),
|
||||
filepath.Join(ex, "modules", path),
|
||||
filepath.Join(executable, "modules", realpath, "init.ar"),
|
||||
filepath.Join(executable, "modules", path),
|
||||
ex, err := os.Getwd()
|
||||
if err != nil {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "Could not get working directory", EXISTS: true}
|
||||
}
|
||||
}
|
||||
|
||||
var p string
|
||||
var found bool
|
||||
for _, p = range pathsToTest {
|
||||
if FileExists(p) {
|
||||
found = true
|
||||
break
|
||||
exc, err := os.Executable()
|
||||
if err != nil {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "Could not get executable", EXISTS: true}
|
||||
}
|
||||
executable := filepath.Dir(exc)
|
||||
isABS := filepath.IsAbs(path)
|
||||
var pathsToTest []string
|
||||
if isABS {
|
||||
pathsToTest = []string{
|
||||
filepath.Join(path),
|
||||
filepath.Join(realpath, "init.ar"),
|
||||
}
|
||||
} else {
|
||||
pathsToTest = []string{
|
||||
filepath.Join(origin, realpath, "init.ar"),
|
||||
filepath.Join(origin, path),
|
||||
filepath.Join(origin, "modules", path),
|
||||
filepath.Join(origin, "modules", realpath, "init.ar"),
|
||||
filepath.Join(ex, path),
|
||||
filepath.Join(ex, "modules", realpath, "init.ar"),
|
||||
filepath.Join(ex, "modules", path),
|
||||
filepath.Join(executable, "modules", realpath, "init.ar"),
|
||||
filepath.Join(executable, "modules", path),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if !found {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "File does not exist: " + realpath, EXISTS: true}
|
||||
} else if importing[p] {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "Circular import: " + realpath, EXISTS: true}
|
||||
} else if _, ok := imported[p]; ok {
|
||||
return imported[p], ArErr{}
|
||||
}
|
||||
importing[p] = true
|
||||
codelines := readFile(p)
|
||||
var p string
|
||||
var found bool
|
||||
for _, p = range pathsToTest {
|
||||
if FileExists(p) {
|
||||
found = true
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
translated, translationerr := translate(codelines)
|
||||
if translationerr.EXISTS {
|
||||
return ArObject{}, translationerr
|
||||
}
|
||||
ArgsArArray := []any{}
|
||||
withoutarfile := []string{}
|
||||
if len(Args) > 1 {
|
||||
withoutarfile = Args[1:]
|
||||
}
|
||||
for _, arg := range withoutarfile {
|
||||
ArgsArArray = append(ArgsArArray, arg)
|
||||
}
|
||||
local := newscope()
|
||||
localvars := Map(anymap{
|
||||
"program": Map(anymap{
|
||||
"args": ArArray(ArgsArArray),
|
||||
"origin": origin,
|
||||
"import": builtinFunc{"import", func(args ...any) (any, ArErr) {
|
||||
if len(args) != 1 {
|
||||
return nil, ArErr{"Import Error", "Invalid number of arguments", 0, realpath, "", true}
|
||||
}
|
||||
if _, ok := args[0].(string); !ok {
|
||||
return nil, ArErr{"Import Error", "Invalid argument type", 0, realpath, "", true}
|
||||
}
|
||||
return importMod(args[0].(string), filepath.Dir(p), false, global)
|
||||
}},
|
||||
"cwd": ex,
|
||||
"exc": exc,
|
||||
"file": Map(anymap{
|
||||
"name": filepath.Base(p),
|
||||
"path": p,
|
||||
if !found {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "File does not exist: " + realpath, EXISTS: true}
|
||||
} else if importing[p] {
|
||||
return ArObject{}, ArErr{TYPE: "Import Error", message: "Circular import: " + realpath, EXISTS: true}
|
||||
} else if _, ok := imported[p]; ok {
|
||||
return imported[p], ArErr{}
|
||||
}
|
||||
importing[p] = true
|
||||
codelines := readFile(p)
|
||||
|
||||
translated, translationerr := translate(codelines)
|
||||
if translationerr.EXISTS {
|
||||
return ArObject{}, translationerr
|
||||
}
|
||||
ArgsArArray := []any{}
|
||||
withoutarfile := []string{}
|
||||
if len(Args) > 1 {
|
||||
withoutarfile = Args[1:]
|
||||
}
|
||||
for _, arg := range withoutarfile {
|
||||
ArgsArArray = append(ArgsArArray, arg)
|
||||
}
|
||||
global := newscope()
|
||||
localvars := Map(anymap{
|
||||
"program": Map(anymap{
|
||||
"args": ArArray(ArgsArArray),
|
||||
"origin": origin,
|
||||
"import": builtinFunc{"import", func(args ...any) (any, ArErr) {
|
||||
if len(args) != 1 {
|
||||
return nil, ArErr{"Import Error", "Invalid number of arguments", 0, realpath, "", true}
|
||||
}
|
||||
if _, ok := args[0].(string); !ok {
|
||||
return nil, ArErr{"Import Error", "Invalid argument type", 0, realpath, "", true}
|
||||
}
|
||||
return importMod(args[0].(string), filepath.Dir(p), false)
|
||||
}},
|
||||
"cwd": ex,
|
||||
"exc": exc,
|
||||
"file": Map(anymap{
|
||||
"name": filepath.Base(p),
|
||||
"path": p,
|
||||
}),
|
||||
"main": main,
|
||||
"scope": global,
|
||||
}),
|
||||
"main": main,
|
||||
"scope": global,
|
||||
}),
|
||||
})
|
||||
_, runimeErr := ThrowOnNonLoop(run(translated, stack{global, localvars, local}))
|
||||
importing[p] = false
|
||||
if runimeErr.EXISTS {
|
||||
return ArObject{}, runimeErr
|
||||
}
|
||||
imported[p] = local
|
||||
return local, ArErr{}
|
||||
})
|
||||
_, runimeErr := ThrowOnNonLoop(run(translated, stack{vars, localvars, global}))
|
||||
importing[p] = false
|
||||
if runimeErr.EXISTS {
|
||||
return ArObject{}, runimeErr
|
||||
}
|
||||
imported[p] = global
|
||||
return global, ArErr{}
|
||||
*/
|
||||
}
|
||||
|
||||
15
src/input.go
15
src/input.go
@@ -1,9 +1,9 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"fmt"
|
||||
"os"
|
||||
"syscall/js"
|
||||
|
||||
"golang.org/x/term"
|
||||
)
|
||||
@@ -12,12 +12,15 @@ func input(args ...any) string {
|
||||
output := []any{}
|
||||
for i := 0; i < len(args); i++ {
|
||||
output = append(output, anyToArgon(args[i], false, true, 3, 0, true, 0))
|
||||
if i != len(args)-1 {
|
||||
output = append(output, " ")
|
||||
}
|
||||
}
|
||||
fmt.Print(output...)
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
scanner.Scan()
|
||||
input := scanner.Text()
|
||||
return input
|
||||
text := fmt.Sprint(output...)
|
||||
fmt.Print(text)
|
||||
inputData := js.Global().Get("getInput").Invoke(fmt.Sprint(output...)).String()
|
||||
fmt.Println(inputData)
|
||||
return (inputData)
|
||||
}
|
||||
|
||||
func getPassword(args ...any) (string, error) {
|
||||
|
||||
36
src/main.go
36
src/main.go
@@ -3,6 +3,7 @@ package main
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"syscall/js"
|
||||
)
|
||||
|
||||
// args without the program path
|
||||
@@ -15,6 +16,7 @@ func newscope() ArObject {
|
||||
}
|
||||
|
||||
func main() {
|
||||
c := make(chan ArObject)
|
||||
defer func() {
|
||||
if r := recover(); r != nil {
|
||||
fmt.Println("There was a fundamental error in argon v3 that caused it to crash.")
|
||||
@@ -37,18 +39,24 @@ func main() {
|
||||
}()
|
||||
initRandom()
|
||||
garbageCollect()
|
||||
global := makeGlobal()
|
||||
if len(Args) == 0 {
|
||||
shell(global)
|
||||
os.Exit(0)
|
||||
}
|
||||
ex, e := os.Getwd()
|
||||
if e != nil {
|
||||
panic(e)
|
||||
}
|
||||
_, err := importMod(Args[0], ex, true, global)
|
||||
if err.EXISTS {
|
||||
panicErr(err)
|
||||
os.Exit(1)
|
||||
}
|
||||
obj := js.Global().Get("Object").New()
|
||||
obj.Set("eval", js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
code := ""
|
||||
allowDocument := false
|
||||
if len(args) >= 1 {
|
||||
code = args[0].String()
|
||||
}
|
||||
if len(args) >= 2 {
|
||||
allowDocument = args[1].Bool()
|
||||
}
|
||||
val, err := wasmRun(code, allowDocument)
|
||||
if err.EXISTS {
|
||||
panicErr(err)
|
||||
return js.Null()
|
||||
}
|
||||
|
||||
return js.ValueOf(argonToJsValid(val))
|
||||
}))
|
||||
js.Global().Set("Ar", obj)
|
||||
<-c
|
||||
}
|
||||
|
||||
@@ -44,6 +44,7 @@ func parseGenericImport(code UNPARSEcode, index int, codeline []UNPARSEcode) (Ar
|
||||
}
|
||||
|
||||
func runImport(importOBJ ArImport, stack stack, stacklevel int) (any, ArErr) {
|
||||
return nil, ArErr{"Import Error", "importing in WASM is currently not supported", importOBJ.line, importOBJ.path, importOBJ.code, true}
|
||||
val, err := runVal(importOBJ.filePath, stack, stacklevel+1)
|
||||
val = ArValidToAny(val)
|
||||
if err.EXISTS {
|
||||
|
||||
@@ -46,7 +46,6 @@ func translateVal(code UNPARSEcode, index int, codelines []UNPARSEcode, isLine i
|
||||
return parseTryCatch(code, index, codelines)
|
||||
}
|
||||
}
|
||||
|
||||
if isLine >= 1 {
|
||||
if isDoWrap(code) {
|
||||
return parseDoWrap(code, index, codelines)
|
||||
|
||||
221
src/wasm.go
Normal file
221
src/wasm.go
Normal file
@@ -0,0 +1,221 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
"syscall/js"
|
||||
)
|
||||
|
||||
func argonToJsValid(argon any) any {
|
||||
switch x := argon.(type) {
|
||||
case number:
|
||||
f, _ := x.Float64()
|
||||
return f
|
||||
case ArObject:
|
||||
if typeof(x) == "array" {
|
||||
arr := js.Global().Get("Array").New()
|
||||
for i, v := range x.obj["__value__"].([]any) {
|
||||
arr.SetIndex(i, argonToJsValid(v))
|
||||
}
|
||||
return arr
|
||||
} else if typeof(x) == "string" {
|
||||
return x.obj["__value__"].(string)
|
||||
}
|
||||
|
||||
obj := js.Global().Get("Object").New()
|
||||
for k, v := range x.obj {
|
||||
obj.Set(anyToArgon(k, false, false, 3, 0, false, 0), argonToJsValid(v))
|
||||
}
|
||||
return obj
|
||||
case bool, string:
|
||||
return x
|
||||
default:
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
func wasmRun(code string, allowDocument bool) (any, ArErr) {
|
||||
JSclearTimers()
|
||||
initRandom()
|
||||
global := makeGlobal(allowDocument)
|
||||
lines := strings.Split(code, "\n")
|
||||
codelines := []UNPARSEcode{}
|
||||
for i := 0; i < len(lines); i++ {
|
||||
codelines = append(codelines, UNPARSEcode{
|
||||
lines[i],
|
||||
lines[i],
|
||||
i + 1,
|
||||
"<wasm>",
|
||||
})
|
||||
}
|
||||
|
||||
translated, translationerr := translate(codelines)
|
||||
if translationerr.EXISTS {
|
||||
return nil, translationerr
|
||||
}
|
||||
local := newscope()
|
||||
localvars := Map(anymap{
|
||||
"program": Map(anymap{
|
||||
"args": []any{},
|
||||
"origin": "",
|
||||
"import": builtinFunc{"import", func(args ...any) (any, ArErr) {
|
||||
return nil, ArErr{"Import Error", "Cannot Import in WASM", 0, "<wasm>", "", true}
|
||||
}},
|
||||
"cwd": "",
|
||||
"exc": "",
|
||||
"file": Map(anymap{
|
||||
"name": "<wasm>",
|
||||
"path": "",
|
||||
}),
|
||||
"main": true,
|
||||
"scope": global,
|
||||
}),
|
||||
})
|
||||
return ThrowOnNonLoop(run(translated, stack{global, localvars, local}))
|
||||
}
|
||||
|
||||
func await(awaitable js.Value) ([]js.Value, []js.Value) {
|
||||
then := make(chan []js.Value)
|
||||
defer close(then)
|
||||
thenFunc := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
then <- args
|
||||
return nil
|
||||
})
|
||||
defer thenFunc.Release()
|
||||
|
||||
catch := make(chan []js.Value)
|
||||
defer close(catch)
|
||||
catchFunc := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
|
||||
catch <- args
|
||||
return nil
|
||||
})
|
||||
defer catchFunc.Release()
|
||||
|
||||
awaitable.Call("then", thenFunc).Call("catch", catchFunc)
|
||||
|
||||
select {
|
||||
case result := <-then:
|
||||
return result, nil
|
||||
case err := <-catch:
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
var IntervalList = []int{}
|
||||
var TimeoutList = []int{}
|
||||
|
||||
func JSclearTimers() {
|
||||
for _, v := range IntervalList {
|
||||
js.Global().Call("clearInterval", v)
|
||||
}
|
||||
for _, v := range TimeoutList {
|
||||
js.Global().Call("clearTimeout", v)
|
||||
}
|
||||
}
|
||||
|
||||
var ArJS = Map(anymap{
|
||||
"setTimeout": builtinFunc{"setTimeout", func(args ...any) (any, ArErr) {
|
||||
if len(args) > 2 || len(args) < 1 {
|
||||
return nil, ArErr{"TypeError", "Expected 1 or 2 argument, got " + fmt.Sprint(len(args)), 0, "<wasm>", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "function" {
|
||||
return nil, ArErr{"TypeError", "Expected function, got " + typeof(args[0]), 0, "<wasm>", "", true}
|
||||
}
|
||||
var ms int64 = 0
|
||||
if len(args) == 2 {
|
||||
if typeof(args[1]) != "number" {
|
||||
return nil, ArErr{"TypeError", "Expected number, got " + typeof(args[1]), 0, "<wasm>", "", true}
|
||||
}
|
||||
if !args[1].(number).IsInt() {
|
||||
return nil, ArErr{"TypeError", "Expected integer, got float", 0, "<wasm>", "", true}
|
||||
}
|
||||
ms = args[1].(number).Num().Int64()
|
||||
}
|
||||
f := js.FuncOf(func(this js.Value, a []js.Value) interface{} {
|
||||
runCall(
|
||||
call{
|
||||
callable: args[0],
|
||||
args: []any{},
|
||||
},
|
||||
stack{},
|
||||
0,
|
||||
)
|
||||
return nil
|
||||
})
|
||||
n := js.Global().Call("setTimeout", f, ms).Int()
|
||||
TimeoutList = append(TimeoutList, n)
|
||||
return newNumber().SetInt64(int64(n)), ArErr{}
|
||||
}},
|
||||
"setInterval": builtinFunc{"setInterval", func(args ...any) (any, ArErr) {
|
||||
if len(args) > 2 || len(args) < 1 {
|
||||
return nil, ArErr{"TypeError", "Expected 1 or 2 argument, got " + fmt.Sprint(len(args)), 0, "<wasm>", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "function" {
|
||||
return nil, ArErr{"TypeError", "Expected function, got " + typeof(args[0]), 0, "<wasm>", "", true}
|
||||
}
|
||||
var ms int64 = 0
|
||||
if len(args) == 2 {
|
||||
if typeof(args[1]) != "number" {
|
||||
return nil, ArErr{"TypeError", "Expected number, got " + typeof(args[1]), 0, "<wasm>", "", true}
|
||||
}
|
||||
if !args[1].(number).IsInt() {
|
||||
return nil, ArErr{"TypeError", "Expected integer, got float", 0, "<wasm>", "", true}
|
||||
}
|
||||
ms = args[1].(number).Num().Int64()
|
||||
}
|
||||
f := js.FuncOf(func(this js.Value, a []js.Value) interface{} {
|
||||
runCall(
|
||||
call{
|
||||
callable: args[0],
|
||||
args: []any{},
|
||||
},
|
||||
stack{},
|
||||
0,
|
||||
)
|
||||
return nil
|
||||
})
|
||||
n := js.Global().Call("setInterval", f, ms).Int()
|
||||
IntervalList = append(IntervalList, n)
|
||||
return newNumber().SetInt64(int64(n)), ArErr{}
|
||||
}},
|
||||
"clearTimeout": builtinFunc{"clearTimeout", func(args ...any) (any, ArErr) {
|
||||
if len(args) != 1 {
|
||||
return nil, ArErr{"TypeError", "Expected 1 argument, got " + fmt.Sprint(len(args)), 0, "<wasm>", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "number" {
|
||||
return nil, ArErr{"TypeError", "Expected number, got " + typeof(args[0]), 0, "<wasm>", "", true}
|
||||
}
|
||||
if !args[0].(number).IsInt() {
|
||||
return nil, ArErr{"TypeError", "Expected integer, got float", 0, "<wasm>", "", true}
|
||||
}
|
||||
n := args[0].(number).Num().Int64()
|
||||
for i, v := range TimeoutList {
|
||||
if v == int(n) {
|
||||
TimeoutList = append(TimeoutList[:i], TimeoutList[i+1:]...)
|
||||
break
|
||||
}
|
||||
}
|
||||
js.Global().Call("clearTimeout", n)
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
"clearInterval": builtinFunc{"clearInterval", func(args ...any) (any, ArErr) {
|
||||
if len(args) != 1 {
|
||||
return nil, ArErr{"TypeError", "Expected 1 argument, got " + fmt.Sprint(len(args)), 0, "<wasm>", "", true}
|
||||
}
|
||||
if typeof(args[0]) != "number" {
|
||||
return nil, ArErr{"TypeError", "Expected number, got " + typeof(args[0]), 0, "<wasm>", "", true}
|
||||
}
|
||||
if !args[0].(number).IsInt() {
|
||||
return nil, ArErr{"TypeError", "Expected integer, got float", 0, "<wasm>", "", true}
|
||||
}
|
||||
n := args[0].(number).Num().Int64()
|
||||
for i, v := range IntervalList {
|
||||
if v == int(n) {
|
||||
IntervalList = append(IntervalList[:i], IntervalList[i+1:]...)
|
||||
break
|
||||
}
|
||||
}
|
||||
js.Global().Call("clearInterval", n)
|
||||
return nil, ArErr{}
|
||||
}},
|
||||
})
|
||||
Reference in New Issue
Block a user