This document has a brief explanation of all the Anura JS APIs and how to use them.
This API is used to define system settings in Anura, it is a key value store of JS objects.
This api allows you to get a value in the key value store.
Usage:
anura.settings.get("applist"); // Get pinned apps in anura's taskbar
This allows you to set a value in the key value store.
Usage:
anura.settings.set("launcher-keybind", false); // Disables the launcher keybind.
This API is used to import libraries. These libraries are similar to apps and can be installed from the Marketplace or sideloaded through the File Manager.
Usage:
const browser = await anura.import("anura.libbrowser");
browser.openTab("https://google.com/");
AnuraOS provides some preinstalled libraries to help streamline the development experience. This includes the browser library as shown above, along with the anura persistence library and the file picker.
You can find the documentation for the preinstalled libraries here.
This API provides access to Anura's x86 backend; Which is used to create PTYs, write directly to serial terminals (not recommended) or access v86 itself.
This is the v86 emulator object, you can find more documentation on it here.
This is a element containing a canvas with the emulated v86 screen.
This allows you to open a PTY and run commands inside of it. It returns the number of the PTY and is used in other interactions.
Usage:
const pty = await anura.x86.openpty(
"/bin/bash",
screenSize.width,
screenSize.height,
(data) => {
// callback gets called every time the PTY returns data
},
);
This allows you to send data to a PTY. This data should be a string or converted to one.
Usage:
const pty = await anura.x86.openpty(
"/bin/bash",
screenSize.width,
screenSize.height,
(data) => {
console.log(data);
},
);
anura.x86.writepty(pty, "Hello World!");
This allows you to resize a PTY.
Usage:
const pty = await anura.x86.openpty(
"TERM=xterm DISPLAY=:0 bash",
screenSize.width,
screenSize.height,
(data) => {
console.log(data);
},
);
anura.x86.resizepty(pty, screenSize.height, screenSize.width);
This api allows you to interact with the v86 virtual hard disk.
This is the size of the v86 hard disk in bytes.
This allows you to load a image into the x86 hard disk.
Usage:
// single file
const rootfs = await fetch(anura.config.x86[x86image].rootfs);
const blob = await rootfs.blob();
await anura.x86hdd.loadfile(blob);
// split into multiple files
const files = [];
let file_1 = await fetch(anura.config.x86[x86image].rootfs[0]);
files[0] = await file_1.blob();
let file_2 = await fetch(anura.config.x86[x86image].rootfs[1]);
files[1] = await file_2.blob();
await anura.x86hdd.loadfile(new Blob(files));
Deletes the x86 hard disk and refreshes the page. This is a destructive action!
Usage:
console.log("deleting hard disk");
await anura.x86hdd.delete();
Resizes the x86 hard disk by adding empty bytes to the image. The
Usage:
console.log("rezising hard disk")
anura.x86?.emulator.stop();
clearInterval(
anura.x86?.saveinterval,
);
await anura.x86.resize(4294967296) // 4 GB
// make the os able to see the empty bytes
const emulator = new V86Starter(
{
wasm_path:
"/lib/v86.wasm",
memory_size:
512 * 1024 * 1024,
vga_memory_size:
8 * 1024 * 1024,
screen_container:
anura.x86!
.screen_container,
initrd: {
url: "/x86images/resizefs.img",
},
bzimage: {
url: "/x86images/bzResize",
async: false,
},
hda: {
buffer: anura.x86hdd,
async: true,
},
cmdline:
"random.trust_cpu=on 8250.nr_uarts=10 spectre_v2=off pti=off",
bios: {
url: "/bios/seabios.bin",
},
vga_bios: {
url: "/bios/vgabios.bin",
},
autostart: true,
uart1: true,
uart2: true,
},
);
let s0data = "";
emulator.add_listener(
"serial0-output-byte",
async (byte: number) => {
const char =
String.fromCharCode(
byte,
);
if (char === "\r") {
anura.logger.debug(
s0data,
);
if (
s0data.includes(
"Finished Disk",
)
) {
await anura.x86hdd.save(
emulator,
);
this.state.resizing =
false;
if (
document.getElementById(
"resize-disk-btn",
)
) {
document.getElementById(
"resize-disk-btn",
)!.innerText =
"Resize Disk";
}
confirm(
"Resized disk! Would you like to reload the page?",
)
? window.location.reload()
: null;
}
s0data = "";
return;
}
s0data += char;
},
);
This allows you to save the v86 hard disk and sends a notification to the user.
Usage:
console.log("saving hard disk");
await anura.x86hdd.save();
This is an array of WeakRefs that contain WMWindows that are in the anura wm.
This api allows you to create a window that will be displayed in the DE.
Usage:
let win = anura.wm.create(instance, {
title: "Example Window",
width: "1280px",
height: "720px",
});
// do things with the window that gets returned
This is is the same as the anura.wm.create
api but creates a window under the Generic App instance.
Usage:
let win = anura.wm.createGeneric({
title: "Example Window",
width: "1280px",
height: "720px",
});
// another use case
let win = anura.wm.createGeneric("Example Window");
// do stuff with the window that gets returned
This API provides access to Anura's networking backend, for routing your requests through a Wisp compatible backend using libcurl.js.\
This part of the api gives you full access to the libcurl.js APIs directly. You can learn more on how to use them here.
This has the same functionality as the DOM fetch function. It returns a Response
and takes in a URL with options or a Request object.
Usage:
let response = await anura.net.fetch("https://anura.pro/MILESTONE");
console.log(await response.text());
This has the same functionality as the built in DOM function and works identically as the regular WebSocket
constructor.
Usage:
let ws = new anura.net.WebSocket("wss://echo.websocket.in/");
ws.addEventListener("open", () => {
console.log("ws connected!");
ws.send("hello".repeat(128));
});
ws.addEventListener("message", (event) => {
console.log(event.data);
});
This API provides access the Anura's internal filesystem, loosely following the node filesystem spec(slightly out of date).
The best documentation on the usage of this API can probably be found Here.
This function allows for the registration of virtual filesystems. These must extend the AFSProvider class and implement all of the filesystem methods. Here is an example for registering an instance of the built in LocalFS provider.
Usage:
await anura.fs.promises.mkdir("/local-mnt");
const dirHandle = await window.showDirectoryPicker();
dirHandle.requestPermission({ mode: "readwrite" });
anura.fs.installProvider(new LocalFS(dirHandle, anuraPath));
This API provides access to Anura's file service, it is useful for handling the opening of files and setting file handlers.
This method takes a file path and then opens this file using the file handler for the file, falling back to the default if it doesnt have a handler for it.
Usage:
anura.files.open("/config_cached.json"); // uses file handler to open json
This method takes a path and returns an icon based on the file extension using a file handler.
Usage:
anura.files.getIcon("/config_cached.json"); // returns icon for json
This method takes a path and returns a human readable file type based on the file extension using a file handler.
Usage:
anura.files.getFileType("/config_cached.json"); // returns icon for json
This method takes an anura library that has an openFile
function that takes a path
.
Usage:
anura.files.setModule("anura.fileviewer", "png"); // set anura.fileviewer library as default handler for png
This API provides access to Anura's URI handler. It is useful for handling the opening of URIs and setting URI handlers.
This method takes a URI and then opens this URI using the handlers that have been registered for it.
Usage:
anura.uri.handle("https://google.com"); // opens google.com in the default browser
This method takes a protocol and a URIHandlerOptions interface and sets the handler for the protocol.
The URIHandlerOptions interface is defined in URIHandler.ts
Usage:
anura.uri.set("https", {
handler: {
// Specifies that the handler is a library
tag: "lib",
// The package name of the library
pkg: "anura.browser,
// The (optional) version of the library
version: "1.0.0",
// The function to call in the library
import: "openTab",
},
// The (optional) prefix to be prepended to the URI
prefix: "https:",
});
This method takes a protocol and removes the handler for the protocol.
Usage:
anura.uri.remove("https");
This method takes a protocol and returns a boolean indicating if the protocol has a handler.
Usage:
anura.uri.has("https"); // Should always return true because the browser registers itself as the handler for https automatically
This API provides access to Anura's notification service, useful if you need to display an alert to the user.
This api allows you to add a notification to the notification service and have a callback that executes along with it.
Usage:
anura.notifications.add({
title: "Test Notification",
description: `This is a test notification`,
callback: function () {
console.log("hi");
},
timeout: 2000,
}); // Show a notification to the user, on click, it says hi in console, it lasts for 2 seconds.
This API allows you to manage processes running in anura.
Returns a list of all the processes running in anura, as a dreamland stateful array of WeakRefs to the processes.
Note: You should never directly mutate this array, use the provided APIs instead.
This API allows you to remove a process from the process list. This is usually ran as the last function of a processes kill function.
Usage:
function kill() {
anura.processes.remove(this.pid);
}
This API allows you to register a process with the process list. This is usually ran by the constructor of a process.
Usage:
// SpecialProcess extends Process
const process = new SpecialProcess();
anura.processes.register(process);
This API allows you to create a process from the given script and type.
Usage:
anura.processes.create("print('Hello, ' + await readln())", "module");
This API allows you to execute a process from the given script file path
Usage:
await anura.processes.execute("/path/to/script.ajs");
This API provides a special process that wraps the anura service worker. This process claims PID 0 and when it is killed, it will unregister the service worker and reload the page.
This API provides a special process that wraps the anura daemon. This process claims PID 1 and manages all anura init scripts.
An example of an init script can be found in this document
Note: All anurad init scripts must have the .init.ajs
extension, and contain a
shebang-like header like so:
#! {"lang":"module"}
This is an array of running init scripts, which are a special process type. This array is not stateful and direct mutations are allowed.
This API allows you to add an init script to the anura daemon.
Usage:
const script = `
export name = "example";
export description = "This is an example init script";
export start = async () => {
console.log("Hello, World!");
};
`;
await anura.anurad.addInitScript(script);
This function kills the anura daemon and all running init scripts.
Usage:
anura.anurad.kill();
This API returns a usable wsproxy url for any TCP application.
Usage:
let webSocket = new WebSocket(anura.wsproxyURL + "alicesworld.tech:80", [
"binary",
]);
webSocket.onmessage = async (event) => {
const text = await (await event.data).text();
console.log(text);
};
webSocket.onopen = (event) => {
webSocket.send("GET / HTTP/1.1\r\nHost: alicesworld.tech\r\n\r\n");
};
// Sends HTTP 1.1 request to alicesworld.tech using wsproxy
This API creates a anura style context menu you can use in your apps.
Usage:
const contextmenu = new anura.ContextMenu();
contextmenu.addItem("Log to console", function () {
console.log("hello world!");
});
element.addEventListener("contextmenu", (e) => {
e.preventDefault();
const boundingRect = window.frameElement.getBoundingClientRect();
contextmenu.show(e.pageX + boundingRect.x, e.pageY + boundingRect.y);
document.onclick = (e) => {
document.onclick = null;
contextmenu.hide();
e.preventDefault();
};
});
This adds an item to the context menu item with a callback thats executed on selection of that menu item.
const contextmenu = new anura.ContextMenu();
contextmenu.addItem("Log to console", function () {
console.log("hello world!");
});
This makes the context menu visible to the user, it also takes arguments on where to place it on the page.
const contextmenu = new anura.ContextMenu();
contextmenu.addItem("Log to console", function () {
console.log("hello world!");
});
contextmenu.show(e.pageX + boundingRect.x, e.pageY + boundingRect.y); // place context menu where the mouse is
This hides the context menu from the user.
const contextmenu = new anura.ContextMenu();
contextmenu.addItem("Log to console", function () {
console.log("hello world!");
});
contextmenu.hide();
This API provides dialogs for Anura. For app developers, these should be used instead of using native browser dialogs to keep the user inside of the desktop environment and to make your app integrate better with Anura.
This creates a alert dialog window.
Usage:
anura.dialog.alert("Hello World!");
This creates a dialog window that gives the user a prompt to confirm an action. This function returns a boolean
you can use.
Usage:
let confirm = await anura.dialog.confirm("Are you sure?");
if (confirm) {
console.log("They were sure.");
}
This gives a user a dialog prompt where the user can enter text. If the user decides to not input text and a default value exists, it returns that instead or returns null if none of those are met.
Usage:
let input = await anura.dialog.prompt("What is your favorite number?");
if (input) {
console.log(input);
}
// default value mode
let input = await anura.dialog.prompt("What is your favorite number?", "3");
if (input) {
console.log(input);
}
This property contains the element that contains all of the
This property contains alll of the icons in the systray in an array.
This function allows you to create an object in the systray, you can pass in an icon and a tooltip to be rendered.
Usage:
const sysicon = anura.systray.create({
icon: "",
tooltip: "Anura AdBlock Active",
});
sysicon.onclick = (event) => {
console.log("got left click event");
};
sysicon.onrightclick = (event) => {
console.log("got right click event");
};
This API provides information about the platform that Anura is running on.
This property returns the type of platform that Anura is running on. This can be one of the following values:
desktop
- Anura is running on a desktop.mobile
- Anura is running on a mobile phone.tablet
- Anura is running on a tablet.
This property returns a boolean indicating whether the platform supports touch input.
Returns a CSS style you can append to your document's head
to provide styles for your application:
Example:
// Append theme css
document.head.appendChild(
html`<><style data-id="anura-theme">${anura.ui.theme.css()}</style></>`,
);
document.addEventListener("anura-theme-change", () => {
document.head.querySelector('style[data-id="anura-theme"]').innerHTML =
anura.ui.theme.css();
});
You now have the following CSS variables to use, corresponding to the properties listed below.
--theme-fg
--theme-secondary-fg
--theme-border
--theme-dark-border
--theme-bg
--theme-secondary-bg
--theme-dark-bg
--theme-accent
The accent of the theme in hex.
The background color of the theme in hex.
The dark background color of the theme in hex.
The secondary background color of the theme in hex.
The border color of the theme in hex.
The dark border color of the theme in hex.
The foreground/text color of the theme in hex.
The secondary foreground color of the theme in hex.