Implementing Local HTTP Server
The following guide shows how to conduct communication between nanohttpd, an http server for Java, and Mathematica. The result is a server that, if you go to its address in a web browser, displays the result of SessionTime[]
, i.e. the time since the Mathematica kernel associated to the server started.
I'm going to write as if the reader was using OS X with Maven installed because that is the operating system I am using, but this solution works on all operating systems with the proper, obvious, modifications. Directories and so on. On OS X Maven can be installed with Brew using
brew -install maven
Getting up and running with nanohttpd:
Download the latest version of nanohttpd from Github.
Follow the steps listed under "quickstart" on nanohttpd.org
Add this to the top of the sample app among the other imports:
import com.wolfram.jlink.*;
Locate JLink.jar on your harddrive. On OS X it is located at
/Applications/Mathematica.app/SystemFiles/Links/JLink
Navigate to the app's directory and run the following command to include JLink.jar in the Maven project (with the appropriate modifications):
mvn install:install-file -Dfile=/Applications/Mathematica.app/Contents/SystemFiles/Links/JLink/JLink.jar -DgroupId=com.wolfram.jlink -DartifactId=JLink -Dversion=1.0 -Dpackaging=jar
And modify the app's pom.xml by adding the file as a dependency:
<dependency>
<groupId>com.wolfram.jlink</groupId>
<artifactId>JLink</artifactId>
<version>1.0</version>
</dependency>
Check that you can still compile the application and that it still works. Now if that's true, replace the code in App.java with this (see the sample program here):
import java.io.IOException;
import java.util.Map;
import com.wolfram.jlink.*;
import fi.iki.elonen.NanoHTTPD;
public class App extends NanoHTTPD {
KernelLink ml;
public App() throws IOException {
super(8888);
start(NanoHTTPD.SOCKET_READ_TIMEOUT, false);
try {
String jLinkDir = "/Applications/Mathematica.app/SystemFiles/Links/JLink";
System.setProperty("com.wolfram.jlink.libdir", jLinkDir); // http://forums.wolfram.com/mathgroup/archive/2008/Aug/msg00664.html
ml = MathLinkFactory.createKernelLink("-linkmode launch -linkname '\"/Applications/Mathematica.app/Contents/MacOS/MathKernel\" -mathlink'");
// Get rid of the initial InputNamePacket the kernel will send
// when it is launched.
ml.discardAnswer();
} catch (MathLinkException e) {
throw new IOException("Fatal error opening link: " + e.getMessage());
}
System.out.println("\nRunning! Point your browers to http://localhost:8888/ \n");
}
public static void main(String[] args) {
try {
new App();
} catch (IOException ioe) {
System.err.println("Couldn't start server:\n" + ioe);
}
}
@Override
public Response serve(IHTTPSession session) {
String msg = "<html><body><p>";
try {
ml.evaluate("SessionTime[]");
ml.waitForAnswer();
double result = ml.getDouble();
msg = msg + Double.toString(result);
} catch (MathLinkException e) {
msg = msg + "MathLinkException occurred: " + e.getMessage();
}
msg = msg + "</p></body></html>";
return newFixedLengthResponse(msg);
}
}
Look up the line with String jLinkDir =
and confirm that the directory is right. If you are using another operating system than OS X you also have to configure the line with MathLinkFactory
in it. Information about that is available here.
Compile the code and run it by (as you did before to run the sample app), navigating to the project's directory and executing the following commands:
mvcompile
mvn exec:java -Dexec.mainClass="com.stackexchange.mathematica.App"
where you have edited mainClass appropriately. You now have an HTTP server on the address http://localhost:8888/ that calls on a Mathematica kernel and uses its response to answer requests.
The following is a sample implementation of a simple HTTP server in Wolfram Language code only:
https://github.com/arnoudbuzing/wolfram-server
You send it a POST request where the body data of the HTTP request contains the Wolfram Language code you wish to evaluate.
The (running) wolframserver.wls script processes the request by evaluating the code string and returning the result as ExpressionJSON which should be generic enough to parse and process in most programming languages (including javascript for web browsers).
It's a new and evolving project for me, so please give it a star if this is useful to you because that will tell me how much interest there is in this (and how much time to spend on it for making improvements).
Starting in Mathematica 12, there is a built-in function SocketListen
that can start a web server and respond to HTTP requests.
SocketListen
is also available in Mathematica 11.2, but only on an experimental basis.
Further reading: Network Programming Guide.