版本更新

最近 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 执行脚本的地方:

  1. string fileName = d.unix("Asia/Shanghai") + "temp.gs" ;
  2. s.writeFile(fileName, body, 438);
  3. string pwd = s.getwd();
  4. // string res = s.command("gscript", fileName);
  5. string res = s.command("docker","run","--rm","-v", pwd+":/usr/src/gscript","-w","/usr/src/gscript", "crossoverjie/gscript","gscript", fileName);
  6. s.remove(fileName);
  7. r.body = res;
  8. r.ast = dumpAST(body);
  9. r.symbol=dumpSymbol(body);
  10. ctx.JSON(200, r);

主要修改的就是将直接执行的 GScript 命令修改为了调用 docker 执行。

但其实也还有改进空间,后续新增协程之后可以便可监控运行时间,超时后便会自动 kill 进程。

我也将该 Docker 上传到了 DockerHub,现在大家想在本地体验 GScriptREPL 时也只需要运行Docker 就能使用。

  1. docker pull crossoverjie/gscript
  2. docker run --rm -it crossoverjie/gscript:latest gscript

当然也可以执行用 Docker 执行 GScript 脚本:

  1. docker run --rm -v $PWD:/usr/src/gscript -w /usr/src/gscript crossoverjie/gscript gscript {yourpath}/temp.gs

编写 GScript 标准库

接下来重点聊聊 GScript 标准库的事情,其实编写标准库是一个费时费力的事情。



现在编译器已经提供了一些可用的内置函数,借由这些内置函数写一些常见的工具类是完全没有问题的。

对写 GScript 标准库感谢的朋友可以当做一个参考,这里我打了一个样,先看下运行效果:

  1. // 字符串工具类
  2. StringBuilder b = StringBuilder();
  3. b.writeString("10");
  4. b.writeString("20");
  5. int l = b.writeString("30");
  6. string s = b.String();
  7. printf("s:%s, len=%d ",s,l);
  8. assertEqual(s,"102030");
  9. byte[] b2 = toByteArray("40");
  10. b.WriteBytes(b2);
  11. s = b.String();
  12. assertEqual(s,"10203040");
  13. println(s);
  14. // Strings 工具类
  15. Strings s = Strings();
  16. string[] elems = {"name=xxx","age=xx"};
  17. string ret = s.join(elems, "&");
  18. println(ret);
  19. assertEqual(ret, "name=xxx&age=xx");
  20. bool b = s.hasPrefix("http://www.xx.com", "http");
  21. println(b);
  22. assertEqual(b,true);
  23. b = s.hasPrefix("http://www.xx.com", "https");
  24. println(b);
  25. assertEqual(b,false);

其中的实现源码基本上是借鉴了 Go 的标准库,先来看看 StringBuilder 的源码:

  1. class StringBuilder{
  2. byte[] buf = [0]{};
  3. // append contents to buf, it returns the length of s
  4. int writeString(string s){
  5. byte[] temp = toByteArray(s);
  6. append(buf, temp);
  7. return len(temp);
  8. }
  9. // append b to buf, it returns the length of b.
  10. int WriteBytes(byte[] b){
  11. append(buf, b);
  12. return len(b);
  13. }
  14. // copies the buffer to a new.
  15. grow(int n){
  16. if (n > 0) {
  17. // when there is not enough space left.
  18. if (cap(buf) - len(buf) < n) {
  19. byte[] newBuf = [len(buf), 2*cap(buf)+n]{};
  20. copy(newBuf, buf);
  21. buf = newBuf;
  22. }
  23. }
  24. }
  25. string String(){
  26. return toString(buf);
  27. }
  28. }

主要就是借助了原始的数组类型以及 toByteArray/toString 字节数组和字符串的转换函数实现的。

  1. class Strings{
  2. // concatenates the elements of its first argument to create a single string. The separator
  3. // string sep is placed between elements in the resulting string.
  4. string join(string[] elems, string sep){
  5. if (len(elems) == 0) {
  6. return "";
  7. }
  8. if (len(elems) == 1) {
  9. return elems[0];
  10. }
  11. byte[] bs = toByteArray(sep);
  12. int n = len(bs) * (len(elems) -1);
  13. for (int i=0; i < len(elems); i++) {
  14. string s = elems[i];
  15. byte[] bs = toByteArray(s);
  16. n = n + len(bs);
  17. }
  18. StringBuilder sb = StringBuilder();
  19. sb.grow(n);
  20. string first = elems[0];
  21. sb.writeString(first);
  22. string[] remain = elems[1:len(elems)];
  23. for(int i=0; i < len(remain); i++){
  24. sb.writeString(sep);
  25. string r = remain[i];
  26. sb.writeString(r);
  27. }
  28. return sb.String();
  29. }
  30. // tests whether the string s begins with prefix.
  31. bool hasPrefix(string s, string prefix){
  32. byte[] bs = toByteArray(s);
  33. byte[] bp = toByteArray(prefix);
  34. return len(bs) >= len(bp) && toString(bs[0:len(bp)]) == prefix;
  35. }
  36. }

Strings 工具类也是类似的,都是一些内置函数的组合运用;

在写标准库的过程中还会有额外收获,可以再次阅读一遍 Go 标准库的实现流程,换了一种语法实现出来,会加深对 Go 标准库的理解。

所以欢迎感兴趣的朋友向 GScript 贡献标准库,由于我个人精力有限,实现过程中可能会发现缺少某些内置函数或数据结构,这也没关系,反馈 issue 后我会尽快处理。

由于目前 GScript 还不支持包管理,所以新增的函数可以创建 Class 来实现,后续支持包或者是 namespace 之后直接将该 Class 迁移过去即可。


本文相关资源链接

手写编程语言-如何为 GScript 编写标准库的更多相关文章

  1. 手写Javaweb服务器

    简单web服务器 回忆socket 创建客服端(在httpClient_1包下) public class Client {    public static void main(String[] a ...

  2. 如何手写一个js工具库?同时发布到npm上

    自从工作以来,写项目的时候经常需要手写一些方法和引入一些js库 JS基础又十分重要,于是就萌生出自己创建一个JS工具库并发布到npm上的想法 于是就创建了一个名为learnjts的项目,在空余时间也写 ...

  3. C++命名空间、标准库(std,全局命名空间)

    背景 别人遇到的问题: C++ 全局变量不明确与 using namespace std 冲突 我遇到的问题与他相似,函数调用冲突 using namespace std; class compare ...

  4. 利用c++编写bp神经网络实现手写数字识别详解

    利用c++编写bp神经网络实现手写数字识别 写在前面 从大一入学开始,本菜菜就一直想学习一下神经网络算法,但由于时间和资源所限,一直未展开比较透彻的学习.大二下人工智能课的修习,给了我一个学习的契机. ...

  5. 利用神经网络算法的C#手写数字识别

    欢迎大家前往云+社区,获取更多腾讯海量技术实践干货哦~ 下载Demo - 2.77 MB (原始地址):handwritten_character_recognition.zip 下载源码 - 70. ...

  6. Atitit s2018.2 s2 doc list on home ntpc.docx  \Atiitt uke制度体系 法律 法规 规章 条例 国王诏书.docx \Atiitt 手写文字识别 讯飞科大 语音云.docx \Atitit 代码托管与虚拟主机.docx \Atitit 企业文化 每日心灵 鸡汤 值班 发布.docx \Atitit 几大研发体系对比 Stage-Gat

    Atitit s2018.2 s2 doc list on home ntpc.docx \Atiitt uke制度体系  法律 法规 规章 条例 国王诏书.docx \Atiitt 手写文字识别   ...

  7. 使用java语言基于SMTP协议手写邮件客户端

    使用java语言基于SMTP协议手写邮件客户端 1. 说明 电子邮件是互联网上常见的应用,他是互联网早期的产品,直至今日依然受到广大用户的喜爱(在中国可能因为文化背景不同,电子邮件只在办公的时候常用) ...

  8. 利用神经网络算法的C#手写数字识别(一)

    利用神经网络算法的C#手写数字识别 转发来自云加社区,用于学习机器学习与神经网络 欢迎大家前往云+社区,获取更多腾讯海量技术实践干货哦~ 下载Demo - 2.77 MB (原始地址):handwri ...

  9. JDK动态代理深入理解分析并手写简易JDK动态代理(下)

    原文同步发表至个人博客[夜月归途] 原文链接:http://www.guitu18.com/se/java/2019-01-05/27.html 作者:夜月归途 出处:http://www.guitu ...

随机推荐

  1. javascript 原生class操作

    <script type="text/javascript"> function hasClass(elements, cName) { return elements ...

  2. goalng-sync/atomic原子操作

    目录 1.go已经提供了锁,为什么还需要atomic原子操作? 2.atomic原子操作为什么比mutex快? 3.CAS 4.互斥锁与原子操作区别 5.原子操作方法 5.1 atomic.AddIn ...

  3. [NCTF2019]True XML cookbook-1|XXE漏洞

    1.打开题目之后和做的上一道:https://www.cnblogs.com/upfine/p/16534940.html题目界面一样,查看源代码等未发现有用信息,界面如下: 2.那就先按原来那道题的 ...

  4. ACM模式细节

    牛客网的ACM模式需要自己写输入输出,在这里简单记录一下: 基本答题框架: import java.util.*; public class Main{ public static void main ...

  5. 微软Azure配置中心 App Configuration (一):轻松集成到Asp.Net Core

    写在前面 在日常开发中,我这边比较熟悉的配置中心有,携程Apollo,阿里Nacos(配置中心,服务治理一体) 之前文章: Asp.Net Core与携程阿波罗(Apollo)的第一次亲密接触 总体来 ...

  6. 详解MySQL隔离级别

    一个事务具有ACID特性,也就是(Atomicity.Consistency.Isolation.Durability,即原子性.一致性.隔离性.持久性),本文主要讲解一下其中的Isolation,也 ...

  7. Docker0网络及原理探究

    个人观点:Docker网络通信在容器编排.集群部署中具有举足轻重的地位,(玩docker不懂docker0那就......玩不透哇)本篇分析Docker网络,并通过启动几个容器来探究Docker网络及 ...

  8. 窗口部件-基础窗口部件 QWidget

    1 基础窗口部件 QWidget QWidget 类是所有用户界面对象的基类,被称为基础窗口部件.不多废话直接看代码 main.cpp 如下 #include<QtWidgets> int ...

  9. 学习ASP.NET Core Blazor编程系列二——第一个Blazor应用程序(中)

    学习ASP.NET Core Blazor编程系列一--综述 学习ASP.NET Core Blazor编程系列二--第一个Blazor应用程序(上) 四.创建一个Blazor应用程序 1. 第一种创 ...

  10. K8S之YAML配置文件

    通过 YAML 配置文件 部署 Deployment 使用命令(类似 docker-compose) // 部署 kubectl create -f xxx.yml // 删除 kubectl del ...