This is a little project I was asked to do for an interview. The assignment was to create a web service that would run arbitray Javascript. I also had to make web page to drive the service.
I'm assuming a UNIX enviornment for these instructions. Replace folder separaters appropriately for your platform.
You'll need to have nodejs and git installed.
- Clone the repository using git.
- Using a terminal, navigate to the top level folder of the project.
- Run
npm install
to get the required node modules. - Run
./node_modules/less/bin/lessc ./public/css/content.less > ./public/css/content.css
to generate the CSS. - Navigate into the
server
directory. - Run
node server.js
You should now be able to navigate to localhost:8080 and play around with the application.
The client is made up of a single HTML page, index.html. This page contains the controls to interact with the application. In the main tab, Applications, you have the ablitity to write code snippets in javascript in the upper left hand textarea and run them on the server by using the 'Run' button. The output of the execution will be shown in the lower textrea.
You can also save off code snippets by using the 'Save' button. If you choose not to name the snippet, it will be entered in the snippet list, on the right, as No Name.
You can also upload Javascript or text files from your computer with the 'Upload' button and have the code therein placed in the input box.
The information in the other tabs is just stubbed. There were no further requirements for those areas.
The server was built on nodejs. It accepts three types of requests:
- GET requests for static files
- POST requests for javascript execution
- POST requests for file upload
To manage the static file requests I'm using a node module called static. I pass the request into its serve method and it handles the work of locating the files based on a webroot and passing them into the response.
For Javascript execution, I take in the request and evaluate it using Javascript's eval
method. I store the output in a variable and attempt to write it into the response. If no return value was given for the execution, the server writes <No output> into the response.
To handle file uploads I'm using another node module called formidable. I pass any multipart form request through formidable's parse method. This reads in any file and saves it locally to the server. I then use the node module fs, part of the standard installation of node, to read back that file and write its contents into the response. If the file is not one of the types accepted by the server (either Javascript or text at this point), the server responds with a 415, unsupported media type, error code.
When any other request method is attempted, the server responds with a 405, method not allowed, error code.
First of all, there are no naming requirements for the snippets. An improvement here would be to have some sort of uniqueness checking on the names. The auto-naming would also have to be improved to account for this.
Snippets also don't save during a user's session. An improvement here could be using HTML5 local storage to save the snippets on the user's machine. An actual user framework could also be hooked up so that many users could save off snippets and have them stored on the server. Then, even after local storage has expired, the snippets would be grabbed from the server and shown.
The system doesn't care right now what type of file a user wants to send to the server and has no size constraints. In the future a MIME-Type check should be implemented to make sure the user isn't trying to upload a bad file to be parsed by the server. For performance purposes the upload size should be limited to a standard size, say 100kb.
In doing this project I came across the situation where calling console.log()
wouldn't print anything out to the client. This was expected, given that console.log()
doesn't return anything, but I questioned whether or not console output should be written to the response. After doing some investigation with other arbitrary Javascript executors I've found that that isn't being done anywhere else. While it might be nice to implement, it isn't necessary in order to have feature parity with other products.
The biggest problem the server faces right now is that the arbitray code coming from the client could be malicious. The code being executed has access to the full node stack. Knowledge of the server implementation could allow a malicious user to attack responses, spawn new servers, or any other number of things. The best way to combat this would be to make an execution environment for the uploaded Javascript, where the code doesn't have access to anything node related.