目录
版本更新
最近 gscript
更新了 v0.0.11
版本,重点更新了:
docker
运行环境- 新增了 byte 原始类型
- 新增了一些字符串标准库
strings/stringbuilder
- 数组切片语法:
int[] b = a[1: len(a)];
引言
前段时间发布了 gscript
的在线 playground
,
这是一个可以在线运行 gscript
脚本的网站,其本质原理是接收用户的输入源码从而在服务器上运行的服务;这简直就是后门大开的 xss
攻击,为保住服务器我设置了运行 api
的后端服务的用户权限,这样可以避免执行一些恶意的请求。
但也避免不了一些用户执行了一些耗时操作,比如一个死循环、或者是我提供 demo
里的打印杨辉三角。
这本质上是一个递归函数,当打印的三角层数过高时便会非常耗时,同时也非常消耗 cpu。
有几次我去检查服务器时发现了几个 cpu 过高的进程,基本上都是这样的耗时操作,不可避免的会影响到服务器的性能。
使用 docker
为了解决这类问题,很自然的就能想到可以使用 docker
,所有的资源都和宿主机是隔离开的,无论怎么瞎折腾也不会影响到宿主机。
说干就干,最后修改了 api 执行脚本的地方:
string filename = d.unix("asia/shanghai") "temp.gs" ; s.writefile(filename, body, 438); string pwd = s.getwd(); // string res = s.command("gscript", filename); string res = s.command("docker","run","--rm","-v", pwd ":/usr/src/gscript","-w","/usr/src/gscript", "crossoverjie/gscript","gscript", filename); s.remove(filename); r.body = res; r.ast = dumpast(body); r.symbol=dumpsymbol(body); ctx.json(200, r);
主要修改的就是将直接执行的 gscript
命令修改为了调用 docker
执行。
但其实也还有改进空间,后续新增协程之后可以便可监控运行时间,超时后便会自动 kill 进程。
我也将该 docker
上传到了 dockerhub
,现在大家想在本地体验 gscript
的 repl
时也只需要运行docker
就能使用。
docker pull crossoverjie/gscript docker run --rm -it crossoverjie/gscript:latest gscript
当然也可以执行用 docker
执行 gscript
脚本:
docker run --rm -v $pwd:/usr/src/gscript -w /usr/src/gscript crossoverjie/gscript gscript {yourpath}/temp.gs
编写 gscript 标准库
接下来重点聊聊 gscript
标准库的事情,其实编写标准库是一个费时费力的事情。
现在编译器已经提供了一些可用的内置函数,借由这些内置函数写一些常见的工具类是完全没有问题的。
对写 gscript
标准库感谢的朋友可以当做一个参考,这里我打了一个样,先看下运行效果:
// 字符串工具类 stringbuilder b = stringbuilder(); b.writestring("10"); b.writestring("20"); int l = b.writestring("30"); string s = b.string(); printf("s:%s, len=%d ",s,l); assertequal(s,"102030"); byte[] b2 = tobytearray("40"); b.writebytes(b2); s = b.string(); assertequal(s,"10203040"); println(s); // strings 工具类 strings s = strings(); string[] elems = {"name=xxx","age=xx"}; string ret = s.join(elems, "&"); println(ret); assertequal(ret, "name=xxx&age=xx"); bool b = s.hasprefix("http://www.xx.com", "http"); println(b); assertequal(b,true); b = s.hasprefix("http://www.xx.com", "https"); println(b); assertequal(b,false);
其中的实现源码基本上是借鉴了 go 的标准库,先来看看 stringbuilder
的源码:
class stringbuilder{ byte[] buf = [0]{}; // append contents to buf, it returns the length of s int writestring(string s){ byte[] temp = tobytearray(s); append(buf, temp); return len(temp); } // append b to buf, it returns the length of b. int writebytes(byte[] b){ append(buf, b); return len(b); } // copies the buffer to a new. grow(int n){ if (n > 0) { // when there is not enough space left. if (cap(buf) - len(buf) < n) { byte[] newbuf = [len(buf), 2*cap(buf) n]{}; copy(newbuf, buf); buf = newbuf; } } } string string(){ return tostring(buf); } }
主要就是借助了原始的数组类型以及 tobytearray/tostring
字节数组和字符串的转换函数实现的。
class strings{ // concatenates the elements of its first argument to create a single string. the separator // string sep is placed between elements in the resulting string. string join(string[] elems, string sep){ if (len(elems) == 0) { return ""; } if (len(elems) == 1) { return elems[0]; } byte[] bs = tobytearray(sep); int n = len(bs) * (len(elems) -1); for (int i=0; i < len(elems); i ) { string s = elems[i]; byte[] bs = tobytearray(s); n = n len(bs); } stringbuilder sb = stringbuilder(); sb.grow(n); string first = elems[0]; sb.writestring(first); string[] remain = elems[1:len(elems)]; for(int i=0; i < len(remain); i ){ sb.writestring(sep); string r = remain[i]; sb.writestring(r); } return sb.string(); } // tests whether the string s begins with prefix. bool hasprefix(string s, string prefix){ byte[] bs = tobytearray(s); byte[] bp = tobytearray(prefix); return len(bs) >= len(bp) && tostring(bs[0:len(bp)]) == prefix; } }
strings
工具类也是类似的,都是一些内置函数的组合运用;
在写标准库的过程中还会有额外收获,可以再次阅读一遍 go 标准库的实现流程,换了一种语法实现出来,会加深对 go 标准库的理解。
所以欢迎感兴趣的朋友向 gscript
贡献标准库,由于我个人精力有限,实现过程中可能会发现缺少某些内置函数或数据结构,这也没关系,反馈 issue
后我会尽快处理。
由于目前 gscript
还不支持包管理,所以新增的函数可以创建 class
来实现,后续支持包或者是 namespace
之后直接将该 class
迁移过去即可。
本文相关资源链接
- gscript 源码:
- playground 源码:
- gscript docker地址:
以上就是gscript 编写标准库示例详解的详细内容,更多关于gscript 编写标准库的资料请关注其它相关文章!