CLI
Fast command-line client for code execution and interactive sessions. 42+ languages, 30+ shells/REPLs.
Official OpenAPI Swagger Docs ↗Quick Start — Haskell
# Download + setup
curl -O https://git.unturf.com/engineering/unturf/un-inception/-/raw/main/clients/haskell/sync/src/un.hs
export UNSANDBOX_PUBLIC_KEY="unsb-pk-xxxx-xxxx-xxxx-xxxx"
export UNSANDBOX_SECRET_KEY="unsb-sk-xxxxx-xxxxx-xxxxx-xxxxx"
# Run code
./un script.hs
Downloads
Install Guide →Features
- 42+ languages - Python, JS, Go, Rust, C++, Java...
- Sessions - 30+ shells/REPLs, tmux persistence
- Files - Upload files, collect artifacts
- Services - Persistent containers with domains
- Snapshots - Point-in-time backups
- Images - Publish, share, transfer
Integration Quickstart ⚡
Add unsandbox superpowers to your existing Haskell app:
curl -O https://git.unturf.com/engineering/unturf/un-inception/-/raw/main/clients/haskell/sync/src/un.hs
# Option A: Environment variables
export UNSANDBOX_PUBLIC_KEY="unsb-pk-xxxx-xxxx-xxxx-xxxx"
export UNSANDBOX_SECRET_KEY="unsb-sk-xxxxx-xxxxx-xxxxx-xxxxx"
# Option B: Config file (persistent)
mkdir -p ~/.unsandbox
echo "unsb-pk-xxxx-xxxx-xxxx-xxxx,unsb-sk-xxxxx-xxxxx-xxxxx-xxxxx" > ~/.unsandbox/accounts.csv
-- In your Haskell app:
import Un (executeCode)
main = do
result <- executeCode "haskell" "putStrLn \"Hello from Haskell running on unsandbox!\""
putStrLn (stdout result) -- Hello from Haskell running on unsandbox!
ghc -o myapp Main.hs Un.hs && ./myapp
b7f7e0ba9ca4c24065450f069d2323b0
SHA256: 31db1c120df5bce5cd5482aec2f31646528f6d89070e3d63772307f2c2c8ee50
-- PUBLIC DOMAIN - NO LICENSE, NO WARRANTY
--
-- This is free public domain software for the public good of a permacomputer hosted
-- at permacomputer.com - an always-on computer by the people, for the people. One
-- which is durable, easy to repair, and distributed like tap water for machine
-- learning intelligence.
--
-- The permacomputer is community-owned infrastructure optimized around four values:
--
-- TRUTH - First principles, math & science, open source code freely distributed
-- FREEDOM - Voluntary partnerships, freedom from tyranny & corporate control
-- HARMONY - Minimal waste, self-renewing systems with diverse thriving connections
-- LOVE - Be yourself without hurting others, cooperation through natural law
--
-- This software contributes to that vision by enabling code execution across 42+
-- programming languages through a unified interface, accessible to all. Code is
-- seeds to sprout on any abandoned technology.
--
-- Learn more: https://www.permacomputer.com
--
-- Anyone is free to copy, modify, publish, use, compile, sell, or distribute this
-- software, either in source code form or as a compiled binary, for any purpose,
-- commercial or non-commercial, and by any means.
--
-- NO WARRANTY. THE SOFTWARE IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND.
--
-- That said, our permacomputer's digital membrane stratum continuously runs unit,
-- integration, and functional tests on all of it's own software - with our
-- permacomputer monitoring itself, repairing itself, with minimal human in the
-- loop guidance. Our agents do their best.
--
-- Copyright 2025 TimeHexOn & foxhop & russell@unturf
-- https://www.timehexon.com
-- https://www.foxhop.net
-- https://www.unturf.com/software
#!/usr/bin/env runhaskell
{-
Haskell UN CLI - Unsandbox CLI Client
Full-featured CLI matching un.py capabilities:
- Execute code with env vars, input files, artifacts
- Interactive sessions with shell/REPL support
- Persistent services with domains and ports
Usage:
chmod +x un.hs
export UNSANDBOX_API_KEY="your_key_here"
./un.hs [options] <source_file>
./un.hs session [options]
./un.hs service [options]
Uses curl for HTTP (no external dependencies)
-}
import System.Environment (getArgs, getEnv, lookupEnv)
import System.Exit (exitWith, ExitCode(..), exitFailure)
import System.FilePath (takeExtension, takeFileName)
import System.Process (readProcessWithExitCode)
import System.IO (hPutStrLn, hPutStr, hFlush, stderr, stdout)
import System.Directory (createDirectoryIfMissing, setPermissions, getPermissions, setOwnerExecutable)
import Data.List (isPrefixOf, intercalate)
import Data.Char (isDigit, ord)
import Text.Printf (printf)
import Control.Monad (when, unless, forM_)
import Control.Exception (try, catch, IOError)
import qualified Data.ByteString as BS
import qualified Data.ByteString.Char8 as BSC
import qualified Data.ByteString.Base64 as B64
import Crypto.Hash.SHA256 (hmac)
import Numeric (showHex)
import Data.Time.Clock.POSIX (getPOSIXTime, posixSecondsToUTCTime)
import Data.Time.Clock (diffUTCTime)
import qualified Data.Time.Clock
import qualified System.Posix.Files
-- API constants
apiBase :: String
apiBase = "https://api.unsandbox.com"
portalBase :: String
portalBase = "https://unsandbox.com"
languagesCacheTtl :: Int
languagesCacheTtl = 3600 -- 1 hour in seconds
-- ANSI colors
blue, red, green, yellow, reset :: String
blue = "\x1b[34m"
red = "\x1b[31m"
green = "\x1b[32m"
yellow = "\x1b[33m"
reset = "\x1b[0m"
-- Extension to language mapping
extToLang :: String -> Maybe String
extToLang ext = lookup ext extMap
where
extMap = [ (".hs", "haskell"), (".ml", "ocaml"), (".clj", "clojure")
, (".scm", "scheme"), (".lisp", "commonlisp"), (".erl", "erlang")
, (".ex", "elixir"), (".exs", "elixir"), (".py", "python")
, (".js", "javascript"), (".ts", "typescript"), (".rb", "ruby")
, (".go", "go"), (".rs", "rust"), (".c", "c"), (".cpp", "cpp")
, (".cc", "cpp"), (".cxx", "cpp"), (".java", "java")
, (".kt", "kotlin"), (".cs", "csharp"), (".fs", "fsharp")
, (".jl", "julia"), (".r", "r"), (".cr", "crystal")
, (".d", "d"), (".nim", "nim"), (".zig", "zig"), (".v", "v")
, (".dart", "dart"), (".groovy", "groovy"), (".scala", "scala")
, (".sh", "bash"), (".pl", "perl"), (".lua", "lua"), (".php", "php")
]
-- Escape JSON string
escapeJSON :: String -> String
escapeJSON = concatMap escape
where
escape '\\' = "\\\\"
escape '"' = "\\\""
escape '\n' = "\\n"
escape '\r' = "\\r"
escape '\t' = "\\t"
escape c = [c]
-- Parse command line arguments
data Command = Execute ExecuteOpts | Session SessionOpts | Service ServiceOpts | Key KeyOpts | Snapshot SnapshotOpts | Image ImageOpts | Languages LanguagesOpts | Help
data LanguagesOpts = LanguagesOpts
{ langJson :: Bool
}
data ExecuteOpts = ExecuteOpts
{ exFile :: String
, exEnv :: [(String, String)]
, exFiles :: [String]
, exArtifacts :: Bool
, exOutDir :: Maybe String
, exNetwork :: Maybe String
, exVcpu :: Maybe Int
}
data SessionOpts = SessionOpts
{ sessAction :: SessionAction
, sessShell :: Maybe String
, sessNetwork :: Maybe String
, sessVcpu :: Maybe Int
, sessFiles :: [String]
, sessSnapshotName :: Maybe String
, sessSnapshotFrom :: Maybe String
, sessHot :: Bool
}
data SessionAction = SessionList | SessionKill String | SessionCreate
| SessionSnapshot String | SessionRestore String
data ServiceOpts = ServiceOpts
{ svcAction :: ServiceAction
, svcName :: Maybe String
, svcPorts :: Maybe String
, svcType :: Maybe String
, svcBootstrap :: Maybe String
, svcBootstrapFile :: Maybe String
, svcNetwork :: Maybe String
, svcVcpu :: Maybe Int
, svcFiles :: [String]
, svcSnapshotName :: Maybe String
, svcSnapshotFrom :: Maybe String
, svcHot :: Bool
, svcEnvs :: [(String, String)]
, svcEnvFile :: Maybe String
}
data ServiceAction = ServiceList | ServiceInfo String | ServiceLogs String
| ServiceSleep String | ServiceWake String | ServiceDestroy String
| ServiceResize String | ServiceExecute String String | ServiceDumpBootstrap String (Maybe String)
| ServiceCreate | ServiceSnapshot String | ServiceRestore String
| ServiceEnv String (Maybe String) -- action, target
| ServiceSetUnfreezeOnDemand String Bool -- service_id, enabled
data SnapshotOpts = SnapshotOpts
{ snapAction :: SnapshotAction
, snapCloneType :: Maybe String
, snapCloneName :: Maybe String
, snapClonePorts :: Maybe String
}
data SnapshotAction = SnapshotList | SnapshotInfo String | SnapshotDelete String
| SnapshotClone String
data ImageOpts = ImageOpts
{ imgAction :: ImageAction
, imgName :: Maybe String
, imgPorts :: Maybe String
, imgSourceType :: Maybe String
, imgVisibilityMode :: Maybe String
}
data ImageAction = ImageList | ImageInfo String | ImageDelete String
| ImageLock String | ImageUnlock String | ImagePublish String
| ImageVisibility String | ImageSpawn String | ImageClone String
data KeyOpts = KeyOpts
{ keyExtend :: Bool
}
-- Parse arguments
parseArgs :: [String] -> IO Command
parseArgs ("session":rest) = Session <$> parseSession rest
parseArgs ("service":rest) = Service <$> parseService rest
parseArgs ("key":rest) = Key <$> parseKey rest
parseArgs ("snapshot":rest) = Snapshot <$> parseSnapshot rest
parseArgs ("image":rest) = Image <$> parseImage rest
parseArgs ("languages":rest) = Languages <$> parseLanguages rest
parseArgs args = parseExecute args
parseLanguages :: [String] -> IO LanguagesOpts
parseLanguages args = return $ parseLanguagesArgs args defaultLanguagesOpts
where
defaultLanguagesOpts = LanguagesOpts False
parseLanguagesArgs [] opts = opts
parseLanguagesArgs ("--json":rest) opts = parseLanguagesArgs rest opts { langJson = True }
parseLanguagesArgs (_:rest) opts = parseLanguagesArgs rest opts
parseKey :: [String] -> IO KeyOpts
parseKey args = return $ parseKeyArgs args defaultKeyOpts
where
defaultKeyOpts = KeyOpts False
parseKeyArgs [] opts = opts
parseKeyArgs ("--extend":rest) opts = parseKeyArgs rest opts { keyExtend = True }
parseKeyArgs (_:rest) opts = parseKeyArgs rest opts
parseSnapshot :: [String] -> IO SnapshotOpts
parseSnapshot args = return $ parseSnapshotArgs args defaultSnapshotOpts
where
defaultSnapshotOpts = SnapshotOpts SnapshotList Nothing Nothing Nothing
parseSnapshotArgs [] opts = opts
parseSnapshotArgs ("--list":rest) opts = parseSnapshotArgs rest opts { snapAction = SnapshotList }
parseSnapshotArgs ("-l":rest) opts = parseSnapshotArgs rest opts { snapAction = SnapshotList }
parseSnapshotArgs ("--info":id:rest) opts = parseSnapshotArgs rest opts { snapAction = SnapshotInfo id }
parseSnapshotArgs ("--delete":id:rest) opts = parseSnapshotArgs rest opts { snapAction = SnapshotDelete id }
parseSnapshotArgs ("--clone":id:rest) opts = parseSnapshotArgs rest opts { snapAction = SnapshotClone id }
parseSnapshotArgs ("--type":t:rest) opts = parseSnapshotArgs rest opts { snapCloneType = Just t }
parseSnapshotArgs ("--name":n:rest) opts = parseSnapshotArgs rest opts { snapCloneName = Just n }
parseSnapshotArgs ("--ports":p:rest) opts = parseSnapshotArgs rest opts { snapClonePorts = Just p }
parseSnapshotArgs (_:rest) opts = parseSnapshotArgs rest opts
parseImage :: [String] -> IO ImageOpts
parseImage args = return $ parseImageArgs args defaultImageOpts
where
defaultImageOpts = ImageOpts ImageList Nothing Nothing Nothing Nothing
parseImageArgs [] opts = opts
parseImageArgs ("--list":rest) opts = parseImageArgs rest opts { imgAction = ImageList }
parseImageArgs ("-l":rest) opts = parseImageArgs rest opts { imgAction = ImageList }
parseImageArgs ("--info":id:rest) opts = parseImageArgs rest opts { imgAction = ImageInfo id }
parseImageArgs ("--delete":id:rest) opts = parseImageArgs rest opts { imgAction = ImageDelete id }
parseImageArgs ("--lock":id:rest) opts = parseImageArgs rest opts { imgAction = ImageLock id }
parseImageArgs ("--unlock":id:rest) opts = parseImageArgs rest opts { imgAction = ImageUnlock id }
parseImageArgs ("--publish":id:rest) opts = parseImageArgs rest opts { imgAction = ImagePublish id }
parseImageArgs ("--source-type":t:rest) opts = parseImageArgs rest opts { imgSourceType = Just t }
parseImageArgs ("--visibility":id:mode:rest) opts = parseImageArgs rest opts { imgAction = ImageVisibility id, imgVisibilityMode = Just mode }
parseImageArgs ("--spawn":id:rest) opts = parseImageArgs rest opts { imgAction = ImageSpawn id }
parseImageArgs ("--clone":id:rest) opts = parseImageArgs rest opts { imgAction = ImageClone id }
parseImageArgs ("--name":n:rest) opts = parseImageArgs rest opts { imgName = Just n }
parseImageArgs ("--ports":p:rest) opts = parseImageArgs rest opts { imgPorts = Just p }
parseImageArgs (_:rest) opts = parseImageArgs rest opts
parseSession :: [String] -> IO SessionOpts
parseSession args = return $ parseSessionArgs args defaultSessionOpts
where
defaultSessionOpts = SessionOpts SessionCreate Nothing Nothing Nothing [] Nothing Nothing False
parseSessionArgs [] opts = opts
parseSessionArgs ("--list":rest) opts = parseSessionArgs rest opts { sessAction = SessionList }
parseSessionArgs ("--kill":id:rest) opts = parseSessionArgs rest opts { sessAction = SessionKill id }
parseSessionArgs ("--snapshot":id:rest) opts = parseSessionArgs rest opts { sessAction = SessionSnapshot id }
parseSessionArgs ("--restore":id:rest) opts = parseSessionArgs rest opts { sessAction = SessionRestore id }
parseSessionArgs ("--shell":sh:rest) opts = parseSessionArgs rest opts { sessShell = Just sh }
parseSessionArgs ("-s":sh:rest) opts = parseSessionArgs rest opts { sessShell = Just sh }
parseSessionArgs ("--snapshot-name":n:rest) opts = parseSessionArgs rest opts { sessSnapshotName = Just n }
parseSessionArgs ("--from":f:rest) opts = parseSessionArgs rest opts { sessSnapshotFrom = Just f }
parseSessionArgs ("--hot":rest) opts = parseSessionArgs rest opts { sessHot = True }
parseSessionArgs ("-n":net:rest) opts = parseSessionArgs rest opts { sessNetwork = Just net }
parseSessionArgs ("-v":v:rest) opts = parseSessionArgs rest opts { sessVcpu = Just (read v) }
parseSessionArgs ("-f":f:rest) opts = parseSessionArgs rest opts { sessFiles = sessFiles opts ++ [f] }
parseSessionArgs (_:rest) opts = parseSessionArgs rest opts
parseService :: [String] -> IO ServiceOpts
parseService args = return $ parseServiceArgs args defaultServiceOpts
where
defaultServiceOpts = ServiceOpts ServiceCreate Nothing Nothing Nothing Nothing Nothing Nothing Nothing [] Nothing Nothing False [] Nothing
parseServiceArgs [] opts = opts
parseServiceArgs ("--list":rest) opts = parseServiceArgs rest opts { svcAction = ServiceList }
parseServiceArgs ("--info":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceInfo id }
parseServiceArgs ("--logs":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceLogs id }
parseServiceArgs ("--freeze":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceSleep id }
parseServiceArgs ("--unfreeze":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceWake id }
parseServiceArgs ("--destroy":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceDestroy id }
parseServiceArgs ("--resize":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceResize id }
parseServiceArgs ("--snapshot":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceSnapshot id }
parseServiceArgs ("--restore":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceRestore id }
parseServiceArgs ("--unfreeze-on-demand":id:"true":rest) opts = parseServiceArgs rest opts { svcAction = ServiceSetUnfreezeOnDemand id True }
parseServiceArgs ("--unfreeze-on-demand":id:"false":rest) opts = parseServiceArgs rest opts { svcAction = ServiceSetUnfreezeOnDemand id False }
parseServiceArgs ("--execute":id:"--command":cmd:rest) opts = parseServiceArgs rest opts { svcAction = ServiceExecute id cmd }
parseServiceArgs ("--dump-bootstrap":id:file:rest) opts = parseServiceArgs rest opts { svcAction = ServiceDumpBootstrap id (Just file) }
parseServiceArgs ("--dump-bootstrap":id:rest) opts = parseServiceArgs rest opts { svcAction = ServiceDumpBootstrap id Nothing }
parseServiceArgs ("env":action:target:rest) opts = parseServiceArgs rest opts { svcAction = ServiceEnv action (Just target) }
parseServiceArgs ("env":action:rest) opts = parseServiceArgs rest opts { svcAction = ServiceEnv action Nothing }
parseServiceArgs ("--name":n:rest) opts = parseServiceArgs rest opts { svcName = Just n }
parseServiceArgs ("--ports":p:rest) opts = parseServiceArgs rest opts { svcPorts = Just p }
parseServiceArgs ("--type":t:rest) opts = parseServiceArgs rest opts { svcType = Just t }
parseServiceArgs ("--bootstrap":b:rest) opts = parseServiceArgs rest opts { svcBootstrap = Just b }
parseServiceArgs ("--bootstrap-file":f:rest) opts = parseServiceArgs rest opts { svcBootstrapFile = Just f }
parseServiceArgs ("--snapshot-name":n:rest) opts = parseServiceArgs rest opts { svcSnapshotName = Just n }
parseServiceArgs ("--from":f:rest) opts = parseServiceArgs rest opts { svcSnapshotFrom = Just f }
parseServiceArgs ("--hot":rest) opts = parseServiceArgs rest opts { svcHot = True }
parseServiceArgs ("-n":net:rest) opts = parseServiceArgs rest opts { svcNetwork = Just net }
parseServiceArgs ("-v":v:rest) opts = parseServiceArgs rest opts { svcVcpu = Just (read v) }
parseServiceArgs ("-f":f:rest) opts = parseServiceArgs rest opts { svcFiles = svcFiles opts ++ [f] }
parseServiceArgs ("-e":kv:rest) opts =
let (k, v) = span (/= '=') kv
in parseServiceArgs rest opts { svcEnvs = svcEnvs opts ++ [(k, drop 1 v)] }
parseServiceArgs ("--env-file":f:rest) opts = parseServiceArgs rest opts { svcEnvFile = Just f }
parseServiceArgs (_:rest) opts = parseServiceArgs rest opts
parseExecute :: [String] -> IO Command
parseExecute args =
case parseExecArgs args defaultExecOpts of
Just opts -> return $ Execute opts
Nothing -> return Help
where
defaultExecOpts = ExecuteOpts "" [] [] False Nothing Nothing Nothing
parseExecArgs [] opts = if null (exFile opts) then Nothing else Just opts
parseExecArgs (arg:rest) opts
| "-e" `isPrefixOf` arg = parseExecArgs rest opts { exEnv = parseEnv rest : exEnv opts }
| "-f" `isPrefixOf` arg = parseExecArgs rest opts { exFiles = head rest : exFiles opts }
| "-a" == arg = parseExecArgs rest opts { exArtifacts = True }
| "-o" `isPrefixOf` arg = parseExecArgs rest opts { exOutDir = Just (head rest) }
| "-n" `isPrefixOf` arg = parseExecArgs rest opts { exNetwork = Just (head rest) }
| "-v" `isPrefixOf` arg = parseExecArgs rest opts { exVcpu = Just (read (head rest)) }
| "-" `isPrefixOf` arg = do
hPutStrLn stderr $ red ++ "Unknown option: " ++ arg ++ reset
exitFailure
| null (exFile opts) = parseExecArgs rest opts { exFile = arg }
| otherwise = parseExecArgs rest opts
parseEnv (kv:rest) =
let (k, v) = span (/= '=') kv
in (k, drop 1 v)
-- ============================================================================
-- Library API
-- ============================================================================
-- SDK Version
sdkVersion :: String
sdkVersion = "4.2.0"
-- | Return the SDK version
version :: String
version = sdkVersion
-- | Check API health
healthCheck :: IO Bool
healthCheck = do
(exitCode, stdout, _) <- readProcessWithExitCode "curl"
["-s", "-o", "/dev/null", "-w", "%{http_code}", apiBase ++ "/health"] ""
return $ filter isDigit stdout == "200"
-- | Generate HMAC-SHA256 signature for a message
hmacSign :: String -> String -> String
hmacSign = hmacSha256
-- | Detect language from filename extension
detectLanguage :: String -> Maybe String
detectLanguage filename = extToLang (takeExtension filename)
-- | Get list of supported languages (list of strings)
getLanguages :: IO [String]
getLanguages = do
cached <- loadLanguagesCache
case cached of
Just languages -> return languages
Nothing -> do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/languages")
let languages = maybe [] id (extractJsonArray stdout "languages")
when (not (null languages)) $ saveLanguagesCache languages
return languages
-- | Execute code synchronously
execute :: String -> String -> IO (Either String (Bool, String, String, Int))
execute language code = do
apiKey <- getApiKey
let json = "{\"language\":\"" ++ escapeJSON language ++ "\",\"code\":\"" ++ escapeJSON code ++ "\"}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/execute") json
case exitCode of
ExitSuccess ->
let success = case extractJsonString stdout "exit_code" of
Just "0" -> True
_ -> False
stdoutVal = maybe "" id (extractJsonString stdout "stdout")
stderrVal = maybe "" id (extractJsonString stdout "stderr")
exitCodeVal = case extractJsonString stdout "exit_code" of
Just s -> read (filter isDigit s) :: Int
_ -> 0
in return $ Right (success, stdoutVal, stderrVal, exitCodeVal)
_ -> return $ Left "Execution failed"
-- | Execute code asynchronously, returning a job ID
executeAsync :: String -> String -> IO (Maybe String)
executeAsync language code = do
apiKey <- getApiKey
let json = "{\"language\":\"" ++ escapeJSON language ++ "\",\"code\":\"" ++ escapeJSON code ++ "\"}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/execute/async") json
case exitCode of
ExitSuccess -> return $ extractJsonString stdout "job_id"
_ -> return Nothing
-- | Get job status
getJob :: String -> IO (Maybe (String, String))
getJob jobId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlGet apiKey (apiBase ++ "/jobs/" ++ jobId)
case exitCode of
ExitSuccess ->
let status = maybe "unknown" id (extractJsonString stdout "status")
language = maybe "" id (extractJsonString stdout "language")
in return $ Just (status, language)
_ -> return Nothing
-- | Wait for job completion
waitJob :: String -> IO (Either String (Bool, String, String, Int))
waitJob jobId = waitJobLoop jobId 0 100
where
pollDelays = [300, 450, 700, 900, 650, 1600, 2000]
terminalStates = ["completed", "failed", "timeout", "cancelled"]
waitJobLoop jid pollCount maxPolls
| pollCount >= maxPolls = return $ Left "Max polls exceeded"
| otherwise = do
let delayIdx = min pollCount (length pollDelays - 1)
let delayMs = pollDelays !! delayIdx
threadDelay (delayMs * 1000) -- threadDelay takes microseconds
result <- getJob jid
case result of
Just (status, _) | status `elem` terminalStates -> do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/jobs/" ++ jid)
let success = case extractJsonString stdout "exit_code" of
Just "0" -> True
_ -> False
stdoutVal = maybe "" id (extractJsonString stdout "stdout")
stderrVal = maybe "" id (extractJsonString stdout "stderr")
exitCodeVal = case extractJsonString stdout "exit_code" of
Just s -> read (filter isDigit s) :: Int
_ -> 1
return $ Right (success, stdoutVal, stderrVal, exitCodeVal)
_ -> waitJobLoop jid (pollCount + 1) maxPolls
-- | Cancel a running job
cancelJob :: String -> IO Bool
cancelJob jobId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlDelete apiKey (apiBase ++ "/jobs/" ++ jobId)
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | List all active jobs
listJobs :: IO String
listJobs = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/jobs")
return stdout
-- | List all sessions
sessionList :: IO String
sessionList = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/sessions")
return stdout
-- | Get session details
sessionGet :: String -> IO String
sessionGet sessionId = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/sessions/" ++ sessionId)
return stdout
-- | Create a new session
sessionCreate :: Maybe String -> Maybe String -> IO (Maybe String)
sessionCreate shell network = do
apiKey <- getApiKey
let shellVal = maybe "bash" id shell
let networkJson = maybe "" (\n -> ",\"network\":\"" ++ n ++ "\"") network
let json = "{\"shell\":\"" ++ shellVal ++ "\"" ++ networkJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/sessions") json
return $ extractJsonString stdout "id"
-- | Destroy a session
sessionDestroy :: String -> IO Bool
sessionDestroy sessionId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlDelete apiKey (apiBase ++ "/sessions/" ++ sessionId)
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Freeze a session
sessionFreeze :: String -> IO Bool
sessionFreeze sessionId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/sessions/" ++ sessionId ++ "/freeze") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Unfreeze a session
sessionUnfreeze :: String -> IO Bool
sessionUnfreeze sessionId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/sessions/" ++ sessionId ++ "/unfreeze") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Boost session resources
sessionBoost :: String -> Int -> IO Bool
sessionBoost sessionId vcpu = do
apiKey <- getApiKey
let json = "{\"vcpu\":" ++ show vcpu ++ "}"
(exitCode, stdout, _) <- curlPatch apiKey (apiBase ++ "/sessions/" ++ sessionId) json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Unboost session
sessionUnboost :: String -> IO Bool
sessionUnboost sessionId = sessionBoost sessionId 1
-- | Execute a command in a session
sessionExecute :: String -> String -> IO String
sessionExecute sessionId command = do
apiKey <- getApiKey
let json = "{\"command\":\"" ++ escapeJSON command ++ "\"}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/sessions/" ++ sessionId ++ "/execute") json
return stdout
-- | List all services
serviceList :: IO String
serviceList = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/services")
return stdout
-- | Get service details
serviceGet :: String -> IO String
serviceGet serviceId = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/services/" ++ serviceId)
return stdout
-- | Create a new service
serviceCreate :: String -> Maybe String -> Maybe String -> Maybe String -> IO (Maybe String)
serviceCreate name ports bootstrap network = do
apiKey <- getApiKey
let portsJson = maybe "" (\p -> ",\"ports\":[" ++ p ++ "]") ports
let bootstrapJson = maybe "" (\b -> ",\"bootstrap\":\"" ++ escapeJSON b ++ "\"") bootstrap
let networkJson = maybe "" (\n -> ",\"network\":\"" ++ n ++ "\"") network
let json = "{\"name\":\"" ++ escapeJSON name ++ "\"" ++ portsJson ++ bootstrapJson ++ networkJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/services") json
return $ extractJsonString stdout "id"
-- | Destroy a service
serviceDestroy :: String -> IO Bool
serviceDestroy serviceId = do
result <- curlDeleteWithSudo "" (apiBase ++ "/services/" ++ serviceId)
case result of
SudoSuccess _ -> return True
_ -> return False
-- | Freeze a service
serviceFreeze :: String -> IO Bool
serviceFreeze serviceId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/freeze") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Unfreeze a service
serviceUnfreeze :: String -> IO Bool
serviceUnfreeze serviceId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/unfreeze") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Lock a service
serviceLock :: String -> IO Bool
serviceLock serviceId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/lock") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Unlock a service
serviceUnlock :: String -> IO Bool
serviceUnlock serviceId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/unlock") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Set unfreeze-on-demand for a service
serviceSetUnfreezeOnDemand :: String -> Bool -> IO Bool
serviceSetUnfreezeOnDemand serviceId enabled = do
apiKey <- getApiKey
let enabledStr = if enabled then "true" else "false"
let json = "{\"unfreeze_on_demand\":" ++ enabledStr ++ "}"
(exitCode, stdout, _) <- curlPatch apiKey (apiBase ++ "/services/" ++ serviceId) json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Redeploy a service
serviceRedeploy :: String -> Maybe String -> IO Bool
serviceRedeploy serviceId bootstrap = do
apiKey <- getApiKey
let bootstrapJson = maybe "" (\b -> "\"bootstrap\":\"" ++ escapeJSON b ++ "\"") bootstrap
let json = "{" ++ bootstrapJson ++ "}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/redeploy") json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Get service logs
serviceLogs :: String -> Bool -> IO String
serviceLogs serviceId allLogs = do
apiKey <- getApiKey
let endpoint = if allLogs
then "/services/" ++ serviceId ++ "/logs?all=true"
else "/services/" ++ serviceId ++ "/logs"
(_, stdout, _) <- curlGet apiKey (apiBase ++ endpoint)
return stdout
-- | Execute a command in a service
serviceExecute :: String -> String -> IO String
serviceExecute serviceId command = do
apiKey <- getApiKey
let json = "{\"command\":\"" ++ escapeJSON command ++ "\"}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/execute") json
return stdout
-- | Resize a service
serviceResize :: String -> Int -> IO Bool
serviceResize serviceId vcpu = do
apiKey <- getApiKey
let json = "{\"vcpu\":" ++ show vcpu ++ "}"
(exitCode, stdout, _) <- curlPatch apiKey (apiBase ++ "/services/" ++ serviceId) json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | List all snapshots
snapshotList :: IO String
snapshotList = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/snapshots")
return stdout
-- | Get snapshot details
snapshotGet :: String -> IO String
snapshotGet snapshotId = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/snapshots/" ++ snapshotId)
return stdout
-- | Create a snapshot of a session
snapshotSession :: String -> Maybe String -> Bool -> IO (Maybe String)
snapshotSession sessionId name hot = do
apiKey <- getApiKey
let nameJson = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\",") name
let hotJson = if hot then "\"hot\":true" else "\"hot\":false"
let json = "{" ++ nameJson ++ hotJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/sessions/" ++ sessionId ++ "/snapshot") json
return $ extractJsonString stdout "id"
-- | Create a snapshot of a service
snapshotService :: String -> Maybe String -> Bool -> IO (Maybe String)
snapshotService serviceId name hot = do
apiKey <- getApiKey
let nameJson = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\",") name
let hotJson = if hot then "\"hot\":true" else "\"hot\":false"
let json = "{" ++ nameJson ++ hotJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/snapshot") json
return $ extractJsonString stdout "id"
-- | Restore from a snapshot
snapshotRestore :: String -> IO (Maybe String)
snapshotRestore snapshotId = do
apiKey <- getApiKey
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/snapshots/" ++ snapshotId ++ "/restore") "{}"
return $ extractJsonString stdout "id"
-- | Delete a snapshot
snapshotDelete :: String -> IO Bool
snapshotDelete snapshotId = do
result <- curlDeleteWithSudo "" (apiBase ++ "/snapshots/" ++ snapshotId)
case result of
SudoSuccess _ -> return True
_ -> return False
-- | Lock a snapshot
snapshotLock :: String -> IO Bool
snapshotLock snapshotId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/snapshots/" ++ snapshotId ++ "/lock") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Unlock a snapshot
snapshotUnlock :: String -> IO Bool
snapshotUnlock snapshotId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/snapshots/" ++ snapshotId ++ "/unlock") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Clone a snapshot to create a new session or service
snapshotClone :: String -> String -> Maybe String -> Maybe String -> Maybe String -> IO (Maybe String)
snapshotClone snapshotId cloneType name ports shell = do
apiKey <- getApiKey
let typeJson = "\"type\":\"" ++ cloneType ++ "\""
let nameJson = maybe "" (\n -> ",\"name\":\"" ++ escapeJSON n ++ "\"") name
let portsJson = maybe "" (\p -> ",\"ports\":[" ++ p ++ "]") ports
let shellJson = maybe "" (\s -> ",\"shell\":\"" ++ s ++ "\"") shell
let json = "{" ++ typeJson ++ nameJson ++ portsJson ++ shellJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/snapshots/" ++ snapshotId ++ "/clone") json
return $ extractJsonString stdout "id"
-- | List images
imageList :: Maybe String -> IO String
imageList filter' = do
apiKey <- getApiKey
let endpoint = maybe "/images" (\f -> "/images?filter=" ++ f) filter'
(_, stdout, _) <- curlGet apiKey (apiBase ++ endpoint)
return stdout
-- | Get image details
imageGet :: String -> IO String
imageGet imageId = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/images/" ++ imageId)
return stdout
-- | Publish an image
imagePublish :: String -> String -> Maybe String -> Maybe String -> IO (Maybe String)
imagePublish sourceType sourceId name description = do
apiKey <- getApiKey
let nameJson = maybe "" (\n -> ",\"name\":\"" ++ escapeJSON n ++ "\"") name
let descJson = maybe "" (\d -> ",\"description\":\"" ++ escapeJSON d ++ "\"") description
let json = "{\"source_type\":\"" ++ sourceType ++ "\",\"source_id\":\"" ++ sourceId ++ "\"" ++ nameJson ++ descJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/images/publish") json
return $ extractJsonString stdout "id"
-- | Delete an image
imageDelete :: String -> IO Bool
imageDelete imageId = do
result <- curlDeleteWithSudo "" (apiBase ++ "/images/" ++ imageId)
case result of
SudoSuccess _ -> return True
_ -> return False
-- | Lock an image
imageLock :: String -> IO Bool
imageLock imageId = do
apiKey <- getApiKey
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/lock") "{}"
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Unlock an image
imageUnlock :: String -> IO Bool
imageUnlock imageId = do
result <- curlPostWithSudo "" (apiBase ++ "/images/" ++ imageId ++ "/unlock") "{}"
case result of
SudoSuccess _ -> return True
_ -> return False
-- | Set image visibility
imageSetVisibility :: String -> String -> IO Bool
imageSetVisibility imageId visibility = do
apiKey <- getApiKey
let json = "{\"visibility\":\"" ++ visibility ++ "\"}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/visibility") json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Grant access to an image
imageGrantAccess :: String -> String -> IO Bool
imageGrantAccess imageId trustedApiKey = do
apiKey <- getApiKey
let json = "{\"api_key\":\"" ++ trustedApiKey ++ "\"}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/access/grant") json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Revoke access to an image
imageRevokeAccess :: String -> String -> IO Bool
imageRevokeAccess imageId trustedApiKey = do
apiKey <- getApiKey
let json = "{\"api_key\":\"" ++ trustedApiKey ++ "\"}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/access/revoke") json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | List trusted API keys for an image
imageListTrusted :: String -> IO String
imageListTrusted imageId = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/images/" ++ imageId ++ "/access")
return stdout
-- | Transfer image ownership
imageTransfer :: String -> String -> IO Bool
imageTransfer imageId toApiKey = do
apiKey <- getApiKey
let json = "{\"to_api_key\":\"" ++ toApiKey ++ "\"}"
(exitCode, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/transfer") json
return $ exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') stdout)
-- | Spawn a service from an image
imageSpawn :: String -> Maybe String -> Maybe String -> Maybe String -> Maybe String -> IO (Maybe String)
imageSpawn imageId name ports bootstrap network = do
apiKey <- getApiKey
let nameJson = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\"") name
let portsJson = maybe "" (\p -> (if null nameJson then "" else ",") ++ "\"ports\":[" ++ p ++ "]") ports
let bootstrapJson = maybe "" (\b -> ",\"bootstrap\":\"" ++ escapeJSON b ++ "\"") bootstrap
let networkJson = maybe "" (\n -> ",\"network\":\"" ++ n ++ "\"") network
let json = "{" ++ nameJson ++ portsJson ++ bootstrapJson ++ networkJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/spawn") json
return $ extractJsonString stdout "id"
-- | Clone an image
imageClone :: String -> Maybe String -> Maybe String -> IO (Maybe String)
imageClone imageId name description = do
apiKey <- getApiKey
let nameJson = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\"") name
let descJson = maybe "" (\d -> (if null nameJson then "" else ",") ++ "\"description\":\"" ++ escapeJSON d ++ "\"") description
let json = "{" ++ nameJson ++ descJson ++ "}"
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/images/" ++ imageId ++ "/clone") json
return $ extractJsonString stdout "id"
-- | Validate API keys
validateKeys :: IO String
validateKeys = do
apiKey <- getApiKey
(_, stdout, _) <- curlPostPortal apiKey (portalBase ++ "/keys/validate") "{}"
return stdout
-- Helper for threadDelay (microseconds)
threadDelay :: Int -> IO ()
threadDelay us = do
let ms = us `div` 1000
_ <- readProcessWithExitCode "sleep" [show (fromIntegral ms / 1000.0 :: Double)] ""
return ()
-- Main
main :: IO ()
main = do
args <- getArgs
cmd <- parseArgs args
case cmd of
Execute opts -> executeCommand opts
Session opts -> sessionCommand opts
Service opts -> serviceCommand opts
Key opts -> keyCommand opts
Snapshot opts -> snapshotCommand opts
Image opts -> imageCommand opts
Languages opts -> languagesCommand opts
Help -> printHelp
printHelp :: IO ()
printHelp = do
putStrLn "Usage:"
putStrLn " un.hs [options] <source_file> Execute code"
putStrLn " un.hs session [options] Manage sessions"
putStrLn " un.hs service [options] Manage services"
putStrLn " un.hs service env <action> <id> Manage service vault"
putStrLn " un.hs snapshot [options] Manage snapshots"
putStrLn " un.hs image [options] Manage images"
putStrLn " un.hs languages [--json] List available languages"
putStrLn " un.hs key [options] Validate/extend API key"
putStrLn ""
putStrLn "Execute options:"
putStrLn " -e KEY=VALUE Environment variable"
putStrLn " -f FILE Input file"
putStrLn " -a Return artifacts"
putStrLn " -o DIR Output directory"
putStrLn " -n MODE Network mode (zerotrust|semitrusted)"
putStrLn " -v N vCPU count (1-8)"
putStrLn ""
putStrLn "Service options:"
putStrLn " -e KEY=VALUE Set vault env var (with --name or env set)"
putStrLn " --env-file FILE Load vault vars from file"
putStrLn " --freeze ID Freeze service"
putStrLn " --unfreeze ID Unfreeze service"
putStrLn " --destroy ID Destroy service"
putStrLn " --resize ID Resize service (requires -v N)"
putStrLn " --unfreeze-on-demand ID true|false Enable/disable auto-unfreeze on HTTP request"
putStrLn ""
putStrLn "Service env commands:"
putStrLn " env status ID Check vault status"
putStrLn " env set ID Set vault (use -e or --env-file)"
putStrLn " env export ID Export vault contents"
putStrLn " env delete ID Delete vault"
putStrLn ""
putStrLn "Session snapshot options:"
putStrLn " --snapshot ID Create snapshot of session"
putStrLn " --restore ID Restore session from snapshot"
putStrLn " --from SNAPSHOT_ID Snapshot ID to restore from"
putStrLn " --snapshot-name NAME Optional snapshot name"
putStrLn " --hot Take live snapshot without freezing"
putStrLn ""
putStrLn "Service snapshot options:"
putStrLn " --snapshot ID Create snapshot of service"
putStrLn " --restore ID Restore service from snapshot"
putStrLn " --from SNAPSHOT_ID Snapshot ID to restore from"
putStrLn " --snapshot-name NAME Optional snapshot name"
putStrLn " --hot Take live snapshot without freezing"
putStrLn ""
putStrLn "Snapshot management options:"
putStrLn " -l, --list List all snapshots"
putStrLn " --info ID Get snapshot details"
putStrLn " --delete ID Delete a snapshot"
putStrLn " --clone ID Clone snapshot to new session/service"
putStrLn " --type TYPE Type for clone (session|service)"
putStrLn " --name NAME Name for cloned instance"
putStrLn " --ports PORTS Ports for cloned service"
putStrLn ""
putStrLn "Key options:"
putStrLn " --extend Open browser to extend/renew key"
putStrLn ""
putStrLn "Image options:"
putStrLn " -l, --list List all images"
putStrLn " --info ID Get image details"
putStrLn " --delete ID Delete an image"
putStrLn " --lock ID Lock image to prevent deletion"
putStrLn " --unlock ID Unlock image"
putStrLn " --publish ID Publish image from service/snapshot (requires --source-type)"
putStrLn " --source-type TYPE Source type: service or snapshot"
putStrLn " --visibility ID MODE Set visibility: private, unlisted, or public"
putStrLn " --spawn ID Spawn new service from image"
putStrLn " --clone ID Clone an image"
putStrLn " --name NAME Name for spawned service or cloned image"
putStrLn " --ports PORTS Ports for spawned service"
putStrLn ""
putStrLn "Languages options:"
putStrLn " --json Output as JSON array"
exitFailure
-- Execute command
executeCommand :: ExecuteOpts -> IO ()
executeCommand opts = do
apiKey <- getApiKey
let file = exFile opts
-- Detect language
let ext = takeExtension file
lang <- case extToLang ext of
Just l -> return l
Nothing -> do
hPutStrLn stderr $ "Error: Unknown extension: " ++ ext
exitFailure
-- Read file
code <- readFile file
-- Build JSON payload
let envJSON = if null (exEnv opts) then ""
else ",\"env\":{" ++ intercalate "," (map (\(k,v) -> printf "\"%s\":\"%s\"" k (escapeJSON v)) (exEnv opts)) ++ "}"
let filesJSON = if null (exFiles opts) then ""
else ",\"input_files\":[" ++ intercalate "," (map fileToJSON (exFiles opts)) ++ "]"
where fileToJSON _ = "" -- simplified for now
let artifactsJSON = if exArtifacts opts then ",\"return_artifacts\":true" else ""
let networkJSON = maybe "" (\n -> ",\"network\":\"" ++ n ++ "\"") (exNetwork opts)
let vcpuJSON = maybe "" (\v -> ",\"vcpu\":" ++ show v) (exVcpu opts)
let json = "{\"language\":\"" ++ lang ++ "\",\"code\":\"" ++ escapeJSON code ++ "\""
++ envJSON ++ filesJSON ++ artifactsJSON ++ networkJSON ++ vcpuJSON ++ "}"
-- Call API
(exitCode, stdout, stderr) <- curlPost apiKey "https://api.unsandbox.com/execute" json
-- Print output
unless (null stdout) $ putStr $ blue ++ stdout ++ reset
unless (null stderr) $ putStr $ red ++ stderr ++ reset
-- Parse exit code from response
let responseExitCode = parseExitCode stdout
exitWith $ if responseExitCode == 0 then ExitSuccess else ExitFailure responseExitCode
-- Session command
sessionCommand :: SessionOpts -> IO ()
sessionCommand opts = do
apiKey <- getApiKey
case sessAction opts of
SessionList -> do
(_, stdout, _) <- curlGet apiKey "https://api.unsandbox.com/sessions"
putStrLn stdout
SessionKill sid -> do
(_, stdout, _) <- curlDelete apiKey ("https://api.unsandbox.com/sessions/" ++ sid)
putStrLn $ green ++ "Session terminated: " ++ sid ++ reset
SessionSnapshot sid -> do
let nameJSON = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\",") (sessSnapshotName opts)
let hotJSON = if sessHot opts then "\"hot\":true" else "\"hot\":false"
let json = "{" ++ nameJSON ++ hotJSON ++ "}"
hPutStrLn stderr $ "Creating snapshot of session " ++ sid ++ "..."
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/sessions/" ++ sid ++ "/snapshot") json
putStrLn $ green ++ "Snapshot created" ++ reset
putStrLn stdout
SessionRestore snapshotId -> do
-- --restore takes snapshot ID directly, calls /snapshots/:id/restore
hPutStrLn stderr $ "Restoring from snapshot " ++ snapshotId ++ "..."
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/snapshots/" ++ snapshotId ++ "/restore") "{}"
putStrLn $ green ++ "Session restored from snapshot" ++ reset
putStrLn stdout
SessionCreate -> do
let shell = maybe "bash" id (sessShell opts)
let networkJSON = maybe "" (\n -> ",\"network\":\"" ++ n ++ "\"") (sessNetwork opts)
let vcpuJSON = maybe "" (\v -> ",\"vcpu\":" ++ show v) (sessVcpu opts)
-- Input files
filesJSON <- if null (sessFiles opts)
then return ""
else do
fileEntries <- mapM (\f -> do
content <- BS.readFile f
let b64 = BSC.unpack $ B64.encode content
let fname = takeFileName f
return $ "{\"filename\":\"" ++ fname ++ "\",\"content_base64\":\"" ++ b64 ++ "\"}"
) (sessFiles opts)
return $ ",\"input_files\":[" ++ intercalate "," fileEntries ++ "]"
let json = "{\"shell\":\"" ++ shell ++ "\"" ++ networkJSON ++ vcpuJSON ++ filesJSON ++ "}"
(_, stdout, _) <- curlPost apiKey "https://api.unsandbox.com/sessions" json
putStrLn $ yellow ++ "Session created (WebSocket required for interactivity)" ++ reset
putStrLn stdout
-- Service command
serviceCommand :: ServiceOpts -> IO ()
serviceCommand opts = do
apiKey <- getApiKey
case svcAction opts of
ServiceList -> do
(_, stdout, _) <- curlGet apiKey "https://api.unsandbox.com/services"
putStrLn stdout
ServiceInfo sid -> do
(_, stdout, _) <- curlGet apiKey ("https://api.unsandbox.com/services/" ++ sid)
putStrLn stdout
ServiceLogs sid -> do
(_, stdout, _) <- curlGet apiKey ("https://api.unsandbox.com/services/" ++ sid ++ "/logs")
putStrLn stdout
ServiceSleep sid -> do
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/services/" ++ sid ++ "/freeze") "{}"
putStrLn $ green ++ "Service frozen: " ++ sid ++ reset
ServiceWake sid -> do
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/services/" ++ sid ++ "/unfreeze") "{}"
putStrLn $ green ++ "Service unfreezing: " ++ sid ++ reset
ServiceDestroy sid -> do
result <- curlDeleteWithSudo apiKey ("https://api.unsandbox.com/services/" ++ sid)
case result of
SudoSuccess _ -> putStrLn $ green ++ "Service destroyed: " ++ sid ++ reset
SudoCancelled -> exitFailure
SudoError msg -> do
hPutStrLn stderr $ red ++ "Error: " ++ msg ++ reset
exitFailure
ServiceResize sid -> do
case svcVcpu opts of
Nothing -> do
hPutStrLn stderr $ red ++ "Error: --resize requires -v N (1-8)" ++ reset
exitFailure
Just vcpu -> do
let json = "{\"vcpu\":" ++ show vcpu ++ "}"
let ram = vcpu * 2
(_, stdout, _) <- curlPatch apiKey ("https://api.unsandbox.com/services/" ++ sid) json
putStrLn $ green ++ "Service resized to " ++ show vcpu ++ " vCPU, " ++ show ram ++ " GB RAM" ++ reset
ServiceExecute sid cmd -> do
let json = "{\"command\":\"" ++ escapeJSON cmd ++ "\"}"
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/services/" ++ sid ++ "/execute") json
unless (null stdout) $ putStr $ blue ++ stdout ++ reset
ServiceDumpBootstrap sid maybeFile -> do
hPutStrLn stderr $ "Fetching bootstrap script from " ++ sid ++ "..."
let json = "{\"command\":\"cat /tmp/bootstrap.sh\"}"
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/services/" ++ sid ++ "/execute") json
-- Extract stdout from JSON response
let bootstrapScript = extractJsonString stdout "stdout"
case bootstrapScript of
Just script | not (null script) -> do
case maybeFile of
Just file -> do
writeFile file script
perms <- getPermissions file
setPermissions file (setOwnerExecutable True perms)
putStrLn $ "Bootstrap saved to " ++ file
Nothing -> putStr script
_ -> do
hPutStrLn stderr $ red ++ "Error: Failed to fetch bootstrap (service not running or no bootstrap file)" ++ reset
exitFailure
ServiceSnapshot sid -> do
let nameJSON = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\",") (svcSnapshotName opts)
let hotJSON = if svcHot opts then "\"hot\":true" else "\"hot\":false"
let json = "{" ++ nameJSON ++ hotJSON ++ "}"
hPutStrLn stderr $ "Creating snapshot of service " ++ sid ++ "..."
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/services/" ++ sid ++ "/snapshot") json
putStrLn $ green ++ "Snapshot created" ++ reset
putStrLn stdout
ServiceRestore snapshotId -> do
-- --restore takes snapshot ID directly, calls /snapshots/:id/restore
hPutStrLn stderr $ "Restoring from snapshot " ++ snapshotId ++ "..."
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/snapshots/" ++ snapshotId ++ "/restore") "{}"
putStrLn $ green ++ "Service restored from snapshot" ++ reset
putStrLn stdout
ServiceSetUnfreezeOnDemand sid enabled -> do
let json = "{\"unfreeze_on_demand\":" ++ (if enabled then "true" else "false") ++ "}"
(_, stdout, _) <- curlPatch apiKey ("https://api.unsandbox.com/services/" ++ sid) json
putStrLn $ green ++ "Service unfreeze_on_demand set to " ++ (if enabled then "true" else "false") ++ ": " ++ sid ++ reset
ServiceEnv action maybeTarget -> do
case action of
"status" -> case maybeTarget of
Just target -> do
result <- serviceEnvStatus target
let hasVault = "\"has_vault\":true" `isPrefixOf` dropWhile (/= 'h') result
if hasVault
then do
putStrLn $ green ++ "Vault: configured" ++ reset
case extractJsonString result "env_count" of
Just count -> putStrLn $ "Variables: " ++ count
Nothing -> return ()
case extractJsonString result "updated_at" of
Just updated -> putStrLn $ "Updated: " ++ updated
Nothing -> return ()
else putStrLn $ yellow ++ "Vault: not configured" ++ reset
Nothing -> do
hPutStrLn stderr $ red ++ "Error: service env status requires service ID" ++ reset
exitFailure
"set" -> case maybeTarget of
Just target -> do
if null (svcEnvs opts) && svcEnvFile opts == Nothing
then do
hPutStrLn stderr $ red ++ "Error: service env set requires -e or --env-file" ++ reset
exitFailure
else do
envContent <- buildEnvContent (svcEnvs opts) (svcEnvFile opts)
success <- serviceEnvSet target envContent
if success
then putStrLn $ green ++ "Vault updated for service " ++ target ++ reset
else do
hPutStrLn stderr $ red ++ "Error: Failed to update vault" ++ reset
exitFailure
Nothing -> do
hPutStrLn stderr $ red ++ "Error: service env set requires service ID" ++ reset
exitFailure
"export" -> case maybeTarget of
Just target -> do
result <- serviceEnvExport target
case extractJsonString result "content" of
Just content -> putStr content
Nothing -> return ()
Nothing -> do
hPutStrLn stderr $ red ++ "Error: service env export requires service ID" ++ reset
exitFailure
"delete" -> case maybeTarget of
Just target -> do
success <- serviceEnvDelete target
if success
then putStrLn $ green ++ "Vault deleted for service " ++ target ++ reset
else do
hPutStrLn stderr $ red ++ "Error: Failed to delete vault" ++ reset
exitFailure
Nothing -> do
hPutStrLn stderr $ red ++ "Error: service env delete requires service ID" ++ reset
exitFailure
_ -> do
hPutStrLn stderr $ red ++ "Error: Unknown env action: " ++ action ++ reset
hPutStrLn stderr "Usage: un.hs service env <status|set|export|delete> <service_id>"
exitFailure
ServiceCreate -> do
case svcName opts of
Nothing -> do
hPutStrLn stderr "Error: --name required to create service"
exitFailure
Just name -> do
let portsJSON = maybe "" (\p -> ",\"ports\":[" ++ p ++ "]") (svcPorts opts)
let typeJSON = maybe "" (\t -> ",\"service_type\":\"" ++ t ++ "\"") (svcType opts)
let bootstrapJSON = maybe "" (\b -> ",\"bootstrap\":\"" ++ escapeJSON b ++ "\"") (svcBootstrap opts)
bootstrapContentJSON <- case svcBootstrapFile opts of
Just f -> do
content <- readFile f
return $ ",\"bootstrap_content\":\"" ++ escapeJSON content ++ "\""
Nothing -> return ""
let networkJSON = maybe "" (\n -> ",\"network\":\"" ++ n ++ "\"") (svcNetwork opts)
let vcpuJSON = maybe "" (\v -> ",\"vcpu\":" ++ show v) (svcVcpu opts)
-- Input files
filesJSON <- if null (svcFiles opts)
then return ""
else do
fileEntries <- mapM (\f -> do
content <- BS.readFile f
let b64 = BSC.unpack $ B64.encode content
let fname = takeFileName f
return $ "{\"filename\":\"" ++ fname ++ "\",\"content_base64\":\"" ++ b64 ++ "\"}"
) (svcFiles opts)
return $ ",\"input_files\":[" ++ intercalate "," fileEntries ++ "]"
let json = "{\"name\":\"" ++ name ++ "\"" ++ portsJSON ++ typeJSON ++ bootstrapJSON ++ bootstrapContentJSON ++ networkJSON ++ vcpuJSON ++ filesJSON ++ "}"
(_, stdout, _) <- curlPost apiKey "https://api.unsandbox.com/services" json
putStrLn $ green ++ "Service created" ++ reset
putStrLn stdout
-- Auto-set vault if env vars were provided
when (not (null (svcEnvs opts)) || svcEnvFile opts /= Nothing) $ do
case extractJsonString stdout "id" of
Just serviceId -> do
envContent <- buildEnvContent (svcEnvs opts) (svcEnvFile opts)
when (not (null envContent)) $ do
success <- serviceEnvSet serviceId envContent
if success
then putStrLn $ green ++ "Vault configured with environment variables" ++ reset
else hPutStrLn stderr $ yellow ++ "Warning: Failed to set vault" ++ reset
Nothing -> return ()
-- Check for clock drift error
checkClockDriftError :: String -> IO ()
checkClockDriftError response = do
let hasTimestamp = "timestamp" `isPrefixOf` dropWhile (/= 't') response ||
"\"timestamp\"" `isInfixOf` response
let has401 = "401" `isInfixOf` response
let hasExpired = "expired" `isInfixOf` response
let hasInvalid = "invalid" `isInfixOf` response
when (hasTimestamp && (has401 || hasExpired || hasInvalid)) $ do
hPutStrLn stderr $ red ++ "Error: Request timestamp expired (must be within 5 minutes of server time)" ++ reset
hPutStrLn stderr $ yellow ++ "Your computer's clock may have drifted." ++ reset
hPutStrLn stderr "Check your system time and sync with NTP if needed:"
hPutStrLn stderr " Linux: sudo ntpdate -s time.nist.gov"
hPutStrLn stderr " macOS: sudo sntp -sS time.apple.com"
hPutStrLn stderr " Windows: w32tm /resync"
exitFailure
where
isInfixOf needle haystack = any (isPrefixOf needle) (tails haystack)
tails [] = [[]]
tails s@(_:xs) = s : tails xs
-- HTTP helpers using curl
curlPost :: String -> String -> String -> IO (ExitCode, String, String)
curlPost apiKey url body = do
(publicKey, secretKey) <- getApiKeys
-- Extract path from URL
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "POST" path body
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "POST"
, url
, "-H", "Content-Type: application/json"
] ++ authHeaders ++ ["-d", body]) ""
-- Check for clock drift error
checkClockDriftError stdout
return (exitCode, stdout, stderr)
curlGet :: String -> String -> IO (ExitCode, String, String)
curlGet apiKey url = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "GET" path ""
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", url ] ++ authHeaders) ""
-- Check for clock drift error
checkClockDriftError stdout
return (exitCode, stdout, stderr)
curlDelete :: String -> String -> IO (ExitCode, String, String)
curlDelete apiKey url = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "DELETE" path ""
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "DELETE", url ] ++ authHeaders) ""
-- Check for clock drift error
checkClockDriftError stdout
return (exitCode, stdout, stderr)
curlPatch :: String -> String -> String -> IO (ExitCode, String, String)
curlPatch apiKey url body = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "PATCH" path body
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "PATCH"
, url
, "-H", "Content-Type: application/json"
] ++ authHeaders ++ ["-d", body]) ""
checkClockDriftError stdout
return (exitCode, stdout, stderr)
curlPut :: String -> String -> String -> IO (ExitCode, String, String)
curlPut apiKey url body = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "PUT" path body
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "PUT"
, url
, "-H", "Content-Type: text/plain"
] ++ authHeaders ++ ["-d", body]) ""
checkClockDriftError stdout
return (exitCode, stdout, stderr)
-- Result type for sudo challenge operations
data SudoResult = SudoSuccess String | SudoError String | SudoCancelled
-- Handle 428 sudo OTP challenge - prompts user for OTP and retries the request
handleSudoChallenge :: String -> String -> String -> String -> IO SudoResult
handleSudoChallenge response method endpoint body = do
let challengeId = extractJsonString response "challenge_id"
hPutStrLn stderr $ yellow ++ "Confirmation required. Check your email for a one-time code." ++ reset
hPutStr stderr "Enter OTP: "
hFlush stderr
otpRaw <- getLine
let otp = filter (/= '\n') $ filter (/= '\r') otpRaw
if null otp
then do
hPutStrLn stderr $ red ++ "Error: Operation cancelled" ++ reset
return SudoCancelled
else do
-- Retry the request with sudo headers
(publicKey, secretKey) <- getApiKeys
authHeaders <- buildAuthHeaders publicKey secretKey method endpoint body
-- Build sudo headers
let sudoHeaders = ["-H", "X-Sudo-OTP: " ++ otp] ++
case challengeId of
Just cid -> ["-H", "X-Sudo-Challenge: " ++ cid]
Nothing -> []
let baseArgs = case method of
"DELETE" -> ["-s", "-X", "DELETE", apiBase ++ endpoint]
"POST" -> ["-s", "-X", "POST", apiBase ++ endpoint, "-H", "Content-Type: application/json", "-d", body]
_ -> ["-s", apiBase ++ endpoint]
(exitCode, retryStdout, _) <- readProcessWithExitCode "curl"
(baseArgs ++ authHeaders ++ sudoHeaders) ""
if exitCode == ExitSuccess && not ("\"error\"" `isPrefixOf` dropWhile (/= '"') retryStdout)
then return $ SudoSuccess retryStdout
else return $ SudoError retryStdout
-- Curl DELETE with 428 handling
curlDeleteWithSudo :: String -> String -> IO SudoResult
curlDeleteWithSudo apiKey url = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "DELETE" path ""
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "DELETE", "-w", "\n%{http_code}", url ] ++ authHeaders) ""
-- Split response and status code
let allLines = lines stdout
let (bodyLines, statusLines) = splitAt (length allLines - 1) allLines
let body = intercalate "\n" bodyLines
let httpCode = case statusLines of
[s] -> read (filter (`elem` "0123456789") s) :: Int
_ -> 200
checkClockDriftError body
if httpCode == 428
then handleSudoChallenge body "DELETE" path ""
else return $ SudoSuccess body
-- Curl POST with 428 handling
curlPostWithSudo :: String -> String -> String -> IO SudoResult
curlPostWithSudo apiKey url body = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length "https://api.unsandbox.com") url
authHeaders <- buildAuthHeaders publicKey secretKey "POST" path body
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "POST", "-w", "\n%{http_code}"
, url
, "-H", "Content-Type: application/json"
] ++ authHeaders ++ ["-d", body]) ""
-- Split response and status code
let allLines = lines stdout
let (bodyLines, statusLines) = splitAt (length allLines - 1) allLines
let bodyStr = intercalate "\n" bodyLines
let httpCode = case statusLines of
[s] -> read (filter (`elem` "0123456789") s) :: Int
_ -> 200
checkClockDriftError bodyStr
if httpCode == 428
then handleSudoChallenge bodyStr "POST" path body
else return $ SudoSuccess bodyStr
-- Vault helper functions
maxEnvContentSize :: Int
maxEnvContentSize = 65536
readEnvFile :: String -> IO String
readEnvFile path = do
content <- readFile path
return content
buildEnvContent :: [(String, String)] -> Maybe String -> IO String
buildEnvContent envs maybeEnvFile = do
-- Add from -e flags
let envLines = map (\(k, v) -> k ++ "=" ++ v) envs
-- Add from --env-file
fileLines <- case maybeEnvFile of
Just path -> do
content <- readEnvFile path
return $ filter (not . null) $ filter (not . isPrefixOf "#") $ map (filter (/= '\r')) $ lines content
Nothing -> return []
return $ intercalate "\n" (envLines ++ fileLines)
serviceEnvStatus :: String -> IO String
serviceEnvStatus serviceId = do
apiKey <- getApiKey
(_, stdout, _) <- curlGet apiKey (apiBase ++ "/services/" ++ serviceId ++ "/env")
return stdout
serviceEnvSet :: String -> String -> IO Bool
serviceEnvSet serviceId envContent = do
if length envContent > maxEnvContentSize
then do
hPutStrLn stderr $ red ++ "Error: Env content exceeds maximum size of 64KB" ++ reset
return False
else do
apiKey <- getApiKey
(exitCode, _, _) <- curlPut apiKey (apiBase ++ "/services/" ++ serviceId ++ "/env") envContent
return (exitCode == ExitSuccess)
serviceEnvExport :: String -> IO String
serviceEnvExport serviceId = do
apiKey <- getApiKey
(_, stdout, _) <- curlPost apiKey (apiBase ++ "/services/" ++ serviceId ++ "/env/export") "{}"
return stdout
serviceEnvDelete :: String -> IO Bool
serviceEnvDelete serviceId = do
apiKey <- getApiKey
(exitCode, _, _) <- curlDelete apiKey (apiBase ++ "/services/" ++ serviceId ++ "/env")
return (exitCode == ExitSuccess)
-- Get API keys from environment
getApiKeys :: IO (String, Maybe String)
getApiKeys = do
publicKey <- lookupEnv "UNSANDBOX_PUBLIC_KEY"
secretKey <- lookupEnv "UNSANDBOX_SECRET_KEY"
apiKey <- lookupEnv "UNSANDBOX_API_KEY"
case (publicKey, secretKey, apiKey) of
(Just pk, Just sk, _) -> return (pk, Just sk)
(_, _, Just ak) -> return (ak, Nothing)
_ -> do
hPutStrLn stderr "Error: UNSANDBOX_PUBLIC_KEY and UNSANDBOX_SECRET_KEY not set (or UNSANDBOX_API_KEY for backwards compat)"
exitFailure
getApiKey :: IO String
getApiKey = do
(publicKey, _) <- getApiKeys
return publicKey
-- HMAC-SHA256
hmacSha256 :: String -> String -> String
hmacSha256 secret message =
let secretBS = BSC.pack secret
messageBS = BSC.pack message
mac = hmac secretBS messageBS
in concatMap (printf "%02x") (BS.unpack mac)
makeSignature :: String -> String -> String -> String -> String -> String
makeSignature secretKey timestamp method path body =
let message = timestamp ++ ":" ++ method ++ ":" ++ path ++ ":" ++ body
in hmacSha256 secretKey message
buildAuthHeaders :: String -> Maybe String -> String -> String -> String -> IO [String]
buildAuthHeaders publicKey maybeSecretKey method path body =
case maybeSecretKey of
Just secretKey -> do
now <- getPOSIXTime
let timestamp = show (floor now :: Integer)
let signature = makeSignature secretKey timestamp method path body
return [ "-H", "Authorization: Bearer " ++ publicKey
, "-H", "X-Timestamp: " ++ timestamp
, "-H", "X-Signature: " ++ signature
]
Nothing ->
return ["-H", "Authorization: Bearer " ++ publicKey]
-- Parse exit code from JSON response
parseExitCode :: String -> Int
parseExitCode resp =
case extractField "exit_code" resp of
Just s -> read s
Nothing -> 0
where
extractField field str =
case break (== ':') <$> words str >>= find (\(k,_) -> field `elem` words k) of
Just (_, ':':v) -> Just $ takeWhile isDigit v
_ -> Nothing
find f = foldr (\x acc -> if f x then Just x else acc) Nothing
-- Extract JSON string field (simple parser for basic cases)
extractJsonString :: String -> String -> Maybe String
extractJsonString json field =
case break (== '"') rest of
(_, '"':value) ->
case break (== '"') value of
(v, _) -> Just v
_ -> Nothing
where
needle = "\"" ++ field ++ "\":"
rest = case dropWhile (/= '"') $ dropWhile (not . isPrefixOf needle) $ tails json of
(_:xs) -> case dropWhile (/= ':') xs of
(_:ys) -> dropWhile (`elem` " \t\n") ys
_ -> ""
_ -> ""
tails [] = [[]]
tails s@(_:xs) = s : tails xs
-- Snapshot command
snapshotCommand :: SnapshotOpts -> IO ()
snapshotCommand opts = do
apiKey <- getApiKey
case snapAction opts of
SnapshotList -> do
(_, stdout, _) <- curlGet apiKey "https://api.unsandbox.com/snapshots"
putStrLn stdout
SnapshotInfo sid -> do
(_, stdout, _) <- curlGet apiKey ("https://api.unsandbox.com/snapshots/" ++ sid)
putStrLn stdout
SnapshotDelete sid -> do
result <- curlDeleteWithSudo apiKey ("https://api.unsandbox.com/snapshots/" ++ sid)
case result of
SudoSuccess _ -> putStrLn $ green ++ "Snapshot deleted: " ++ sid ++ reset
SudoCancelled -> exitFailure
SudoError msg -> do
hPutStrLn stderr $ red ++ "Error: " ++ msg ++ reset
exitFailure
SnapshotClone sid -> do
case snapCloneType opts of
Nothing -> do
hPutStrLn stderr "Error: --type (session|service) required for clone"
exitFailure
Just cloneType -> do
let nameJSON = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\",") (snapCloneName opts)
let portsJSON = maybe "" (\p -> "\"ports\":[" ++ p ++ "],") (snapClonePorts opts)
let json = "{\"type\":\"" ++ cloneType ++ "\"," ++ nameJSON ++ portsJSON ++ "}"
hPutStrLn stderr $ "Cloning snapshot " ++ sid ++ " to create new " ++ cloneType ++ "..."
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/snapshots/" ++ sid ++ "/clone") json
putStrLn $ green ++ "Snapshot cloned" ++ reset
putStrLn stdout
-- Image command
imageCommand :: ImageOpts -> IO ()
imageCommand opts = do
apiKey <- getApiKey
case imgAction opts of
ImageList -> do
(_, stdout, _) <- curlGet apiKey "https://api.unsandbox.com/images"
putStrLn stdout
ImageInfo iid -> do
(_, stdout, _) <- curlGet apiKey ("https://api.unsandbox.com/images/" ++ iid)
putStrLn stdout
ImageDelete iid -> do
result <- curlDeleteWithSudo apiKey ("https://api.unsandbox.com/images/" ++ iid)
case result of
SudoSuccess _ -> putStrLn $ green ++ "Image deleted: " ++ iid ++ reset
SudoCancelled -> exitFailure
SudoError msg -> do
hPutStrLn stderr $ red ++ "Error: " ++ msg ++ reset
exitFailure
ImageLock iid -> do
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/images/" ++ iid ++ "/lock") "{}"
putStrLn $ green ++ "Image locked: " ++ iid ++ reset
ImageUnlock iid -> do
result <- curlPostWithSudo apiKey ("https://api.unsandbox.com/images/" ++ iid ++ "/unlock") "{}"
case result of
SudoSuccess _ -> putStrLn $ green ++ "Image unlocked: " ++ iid ++ reset
SudoCancelled -> exitFailure
SudoError msg -> do
hPutStrLn stderr $ red ++ "Error: " ++ msg ++ reset
exitFailure
ImagePublish sourceId -> do
case imgSourceType opts of
Nothing -> do
hPutStrLn stderr $ red ++ "Error: --source-type required (service or snapshot)" ++ reset
exitFailure
Just sourceType -> do
let nameJSON = maybe "" (\n -> ",\"name\":\"" ++ escapeJSON n ++ "\"") (imgName opts)
let json = "{\"source_type\":\"" ++ sourceType ++ "\",\"source_id\":\"" ++ sourceId ++ "\"" ++ nameJSON ++ "}"
(_, stdout, _) <- curlPost apiKey "https://api.unsandbox.com/images/publish" json
putStrLn $ green ++ "Image published" ++ reset
putStrLn stdout
ImageVisibility iid -> do
case imgVisibilityMode opts of
Nothing -> do
hPutStrLn stderr $ red ++ "Error: --visibility requires MODE (private, unlisted, or public)" ++ reset
exitFailure
Just mode -> do
let json = "{\"visibility\":\"" ++ mode ++ "\"}"
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/images/" ++ iid ++ "/visibility") json
putStrLn $ green ++ "Image visibility set to " ++ mode ++ ": " ++ iid ++ reset
ImageSpawn iid -> do
let nameJSON = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\"") (imgName opts)
let portsJSON = maybe "" (\p -> ",\"ports\":[" ++ p ++ "]") (imgPorts opts)
let json = if null nameJSON then "{" ++ (if null portsJSON then "" else drop 1 portsJSON) ++ "}"
else "{" ++ nameJSON ++ portsJSON ++ "}"
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/images/" ++ iid ++ "/spawn") json
putStrLn $ green ++ "Service spawned from image" ++ reset
putStrLn stdout
ImageClone iid -> do
let nameJSON = maybe "" (\n -> "\"name\":\"" ++ escapeJSON n ++ "\"") (imgName opts)
let json = "{" ++ nameJSON ++ "}"
(_, stdout, _) <- curlPost apiKey ("https://api.unsandbox.com/images/" ++ iid ++ "/clone") json
putStrLn $ green ++ "Image cloned" ++ reset
putStrLn stdout
-- Languages cache functions
getLanguagesCachePath :: IO FilePath
getLanguagesCachePath = do
home <- lookupEnv "HOME"
let homeDir = maybe "." id home
return $ homeDir ++ "/.unsandbox/languages.json"
loadLanguagesCache :: IO (Maybe [String])
loadLanguagesCache = do
cachePath <- getLanguagesCachePath
exists <- doesFileExist cachePath
if not exists
then return Nothing
else do
-- Check if cache is fresh (< 1 hour old)
modTime <- getModificationTime cachePath
now <- getCurrentTime
let ageSeconds = floor $ diffUTCTime now modTime
if ageSeconds >= languagesCacheTtl
then return Nothing
else do
content <- readFile cachePath
return $ extractJsonArray content "languages"
where
doesFileExist path = do
result <- try (readFile path) :: IO (Either IOError String)
case result of
Left _ -> return False
Right _ -> return True
getModificationTime path = do
status <- System.Posix.Files.getFileStatus path
return $ posixSecondsToUTCTime $ realToFrac $ modificationTime status
getCurrentTime = Data.Time.Clock.getCurrentTime
saveLanguagesCache :: [String] -> IO ()
saveLanguagesCache languages = do
cachePath <- getLanguagesCachePath
let cacheDir = takeDirectory cachePath
-- Create directory if needed
createDirectoryIfMissing True cacheDir
now <- getPOSIXTime
let timestamp = show (floor now :: Integer)
let langsJson = "[" ++ intercalate "," (map (\l -> "\"" ++ l ++ "\"") languages) ++ "]"
let content = "{\"languages\":" ++ langsJson ++ ",\"timestamp\":" ++ timestamp ++ "}"
writeFile cachePath content
`catch` (\(_ :: IOError) -> return ()) -- Cache failures are non-fatal
where
takeDirectory path = reverse $ dropWhile (/= '/') $ reverse path
-- Languages command
languagesCommand :: LanguagesOpts -> IO ()
languagesCommand opts = do
apiKey <- getApiKey
-- Try cache first
cached <- loadLanguagesCache
langs <- case cached of
Just languages -> return languages
Nothing -> do
(_, stdout, _) <- curlGet apiKey "https://api.unsandbox.com/languages"
let languages = maybe [] id (extractJsonArray stdout "languages")
-- Save to cache
when (not (null languages)) $ saveLanguagesCache languages
return languages
let jsonOutput = langJson opts
if jsonOutput
then do
-- Extract languages array and print as JSON
putStrLn $ "[" ++ intercalate "," (map (\l -> "\"" ++ l ++ "\"") langs) ++ "]"
else do
-- Print each language on its own line
mapM_ putStrLn langs
-- Extract JSON array of strings from response
extractJsonArray :: String -> String -> Maybe [String]
extractJsonArray json field =
let needle = "\"" ++ field ++ "\":["
rest = dropWhile (not . isPrefixOf needle) (tails json)
in case rest of
(x:_) -> Just $ parseArrayItems $ drop (length needle) x
_ -> Nothing
where
tails [] = [[]]
tails s@(_:xs) = s : tails xs
parseArrayItems :: String -> [String]
parseArrayItems s = go (dropWhile (`elem` " \t\n") s) []
where
go (']':_) acc = reverse acc
go ('"':rest) acc =
let (item, remaining) = span (/= '"') rest
in go (dropWhile (`elem` ",] \t\n") (drop 1 remaining)) (item : acc)
go (_:rest) acc = go rest acc
go [] acc = reverse acc
-- Key command
keyCommand :: KeyOpts -> IO ()
keyCommand opts = do
apiKey <- getApiKey
if keyExtend opts
then extendKey apiKey
else validateKey apiKey
-- Validate API key and show status
validateKey :: String -> IO ()
validateKey apiKey = do
let url = portalBase ++ "/keys/validate"
(exitCode, stdout, stderr) <- curlPostPortal apiKey url "{}"
-- Check if valid:false appears in response
let isInvalid = "\"valid\":false" `isPrefixOf` dropWhile (/= 'v') stdout
if exitCode /= ExitSuccess || isInvalid
then do
-- Parse error response
let reason = extractJsonString stdout "reason"
case reason of
Just "expired" -> do
putStrLn $ red ++ "Expired" ++ reset ++ "\n"
-- Show key details
case extractJsonString stdout "public_key" of
Just pk -> putStrLn $ "Public Key: " ++ pk
Nothing -> return ()
case extractJsonString stdout "tier" of
Just tier -> putStrLn $ "Tier: " ++ tier
Nothing -> return ()
case extractJsonString stdout "expired_at_datetime" of
Just expiredAt -> do
putStr $ "Expired: " ++ expiredAt
case extractJsonString stdout "expired_ago" of
Just ago -> putStrLn $ " (" ++ ago ++ ")"
Nothing -> putStrLn ""
Nothing -> return ()
putStrLn ""
putStrLn $ yellow ++ "To renew:" ++ reset ++ " Visit https://unsandbox.com/keys/extend"
exitFailure
Just "invalid_key" -> do
putStrLn $ red ++ "Invalid" ++ reset ++ ": key not found"
exitFailure
Just "suspended" -> do
putStrLn $ red ++ "Suspended" ++ reset ++ ": key has been suspended"
exitFailure
_ -> do
putStrLn $ red ++ "Invalid key" ++ reset
exitFailure
else do
-- Parse valid response
putStrLn $ green ++ "Valid" ++ reset ++ "\n"
case extractJsonString stdout "public_key" of
Just pk -> putStrLn $ "Public Key: " ++ pk
Nothing -> return ()
case extractJsonString stdout "tier" of
Just tier -> putStrLn $ "Tier: " ++ tier
Nothing -> return ()
case extractJsonString stdout "status" of
Just status -> putStrLn $ "Status: " ++ status
Nothing -> return ()
case extractJsonString stdout "valid_through_datetime" of
Just validThrough -> putStrLn $ "Expires: " ++ validThrough
Nothing -> return ()
case extractJsonString stdout "valid_for_human" of
Just validFor -> putStrLn $ "Time Remaining: " ++ validFor
Nothing -> return ()
case extractJsonString stdout "rate_per_minute" of
Just rate -> putStrLn $ "Rate Limit: " ++ rate ++ "/min"
Nothing -> return ()
case extractJsonString stdout "burst" of
Just burst -> putStrLn $ "Burst: " ++ burst
Nothing -> return ()
case extractJsonString stdout "concurrency" of
Just conc -> putStrLn $ "Concurrency: " ++ conc
Nothing -> return ()
-- Extend key (open browser to extend page)
extendKey :: String -> IO ()
extendKey apiKey = do
let url = portalBase ++ "/keys/validate"
(exitCode, stdout, _) <- curlPostPortal apiKey url "{}"
case extractJsonString stdout "public_key" of
Nothing -> do
hPutStrLn stderr "Error: Invalid key or could not retrieve public key"
exitFailure
Just publicKey -> do
let extendUrl = portalBase ++ "/keys/extend?pk=" ++ publicKey
putStrLn "Opening extension page in browser..."
putStrLn $ "If browser doesn't open, visit: " ++ extendUrl
-- Try to open URL in browser (Linux-specific)
_ <- readProcessWithExitCode "sh"
["-c", "xdg-open '" ++ extendUrl ++ "' 2>/dev/null || sensible-browser '" ++ extendUrl ++ "' 2>/dev/null || true"]
""
return ()
-- HTTP helper for portal API
curlPostPortal :: String -> String -> String -> IO (ExitCode, String, String)
curlPostPortal apiKey url body = do
(publicKey, secretKey) <- getApiKeys
let path = drop (length portalBase) url
authHeaders <- buildAuthHeaders publicKey secretKey "POST" path body
(exitCode, stdout, stderr) <- readProcessWithExitCode "curl"
([ "-s", "-X", "POST"
, url
, "-H", "Content-Type: application/json"
] ++ authHeaders ++ ["-d", body]) ""
-- Check for clock drift error
checkClockDriftError stdout
return (exitCode, stdout, stderr)
Documentation clarifications
Dependencies
C Binary (un1) — requires libcurl and libwebsockets:
sudo apt install build-essential libcurl4-openssl-dev libwebsockets-dev
wget unsandbox.com/downloads/un.c && gcc -O2 -o un un.c -lcurl -lwebsockets
SDK Implementations — most use stdlib only (Ruby, JS, Go, etc). Some require minimal deps:
pip install requests # Python
Execute Code
Run a Script
./un hello.py
./un app.js
./un main.rs
With Environment Variables
./un -e DEBUG=1 -e NAME=World script.py
With Input Files (teleport files into sandbox)
./un -f data.csv -f config.json process.py
Get Compiled Binary (teleport artifacts out)
./un -a -o ./bin main.c
Interactive Sessions
Start a Shell Session
# Default bash shell
./un session
# Choose your shell
./un session --shell zsh
./un session --shell fish
# Jump into a REPL
./un session --shell python3
./un session --shell node
./un session --shell julia
Session with Network Access
./un session -n semitrusted
Session Auditing (full terminal recording)
# Record everything (including vim, interactive programs)
./un session --audit -o ./logs
# Replay session later
zcat session.log*.gz | less -R
Collect Artifacts from Session
# Files in /tmp/artifacts/ are collected on exit
./un session -a -o ./outputs
Session Persistence (tmux/screen)
# Default: session terminates on disconnect (clean exit)
./un session
# With tmux: session persists, can reconnect later
./un session --tmux
# Press Ctrl+b then d to detach
# With screen: alternative multiplexer
./un session --screen
# Press Ctrl+a then d to detach
List Active Sessions
./un session --list
# Output:
# Active sessions: 2
#
# SESSION ID CONTAINER SHELL TTL STATUS
# abc123... unsb-vm-12345 python3 45m30s active
# def456... unsb-vm-67890 bash 1h2m active
Reconnect to Existing Session
# Reconnect by container name (requires --tmux or --screen)
./un session --attach unsb-vm-12345
# Use exit to terminate session, or detach to keep it running
Terminate a Session
./un session --kill unsb-vm-12345
Available Shells & REPLs
Shells: bash, dash, sh, zsh, fish, ksh, tcsh, csh, elvish, xonsh, ash
REPLs: python3, bpython, ipython # Python
node # JavaScript
ruby, irb # Ruby
lua # Lua
php # PHP
perl # Perl
guile, scheme # Scheme
ghci # Haskell
erl, iex # Erlang/Elixir
sbcl, clisp # Common Lisp
r # R
julia # Julia
clojure # Clojure
API Key Management
Check Key Status
# Check if your API key is valid
./un key
# Output:
# Valid: key expires in 30 days
Extend Expired Key
# Open the portal to extend an expired key
./un key --extend
# This opens the unsandbox.com portal where you can
# add more credits to extend your key's expiration
Authentication
Credentials are loaded in priority order (highest first):
# 1. CLI flags (highest priority)
./un -p unsb-pk-xxxx -k unsb-sk-xxxxx script.py
# 2. Environment variables
export UNSANDBOX_PUBLIC_KEY=unsb-pk-xxxx-xxxx-xxxx-xxxx
export UNSANDBOX_SECRET_KEY=unsb-sk-xxxxx-xxxxx-xxxxx-xxxxx
./un script.py
# 3. Config file (lowest priority)
# ~/.unsandbox/accounts.csv format: public_key,secret_key
mkdir -p ~/.unsandbox
echo "unsb-pk-xxxx-xxxx-xxxx-xxxx,unsb-sk-xxxxx-xxxxx-xxxxx-xxxxx" > ~/.unsandbox/accounts.csv
./un script.py
Requests are signed with HMAC-SHA256. The bearer token contains only the public key; the secret key computes the signature (never transmitted).
Resource Scaling
Set vCPU Count
# Default: 1 vCPU, 2GB RAM
./un script.py
# Scale up: 4 vCPUs, 8GB RAM
./un -v 4 script.py
# Maximum: 8 vCPUs, 16GB RAM
./un --vcpu 8 heavy_compute.py
Live Session Boosting
# Boost a running session to 2 vCPU, 4GB RAM
./un session --boost sandbox-abc
# Boost to specific vCPU count (4 vCPU, 8GB RAM)
./un session --boost sandbox-abc --boost-vcpu 4
# Return to base resources (1 vCPU, 2GB RAM)
./un session --unboost sandbox-abc
Session Freeze/Unfreeze
Freeze and Unfreeze Sessions
# Freeze a session (stop billing, preserve state)
./un session --freeze sandbox-abc
# Unfreeze a frozen session
./un session --unfreeze sandbox-abc
# Note: Requires --tmux or --screen for persistence
Persistent Services
Create a Service
# Web server with ports
./un service --name web --ports 80,443 --bootstrap "python -m http.server 80"
# With custom domains
./un service --name blog --ports 8000 --domains blog.example.com
# Game server with SRV records
./un service --name mc --type minecraft --bootstrap ./setup.sh
# Deploy app tarball with bootstrap script
./un service --name app --ports 8000 -f app.tar.gz --bootstrap-file ./setup.sh
# setup.sh: cd /tmp && tar xzf app.tar.gz && ./app/start.sh
Manage Services
# List all services
./un service --list
# Get service details
./un service --info abc123
# View bootstrap logs
./un service --logs abc123
./un service --tail abc123 # last 9000 lines
# Execute command in running service
./un service --execute abc123 'journalctl -u myapp -n 50'
# Dump bootstrap script (for migrations)
./un service --dump-bootstrap abc123
./un service --dump-bootstrap abc123 backup.sh
# Freeze/unfreeze service
./un service --freeze abc123
./un service --unfreeze abc123
# Service settings (auto-wake, freeze page display)
./un service --auto-unfreeze abc123 # enable auto-wake on HTTP
./un service --no-auto-unfreeze abc123 # disable auto-wake
./un service --show-freeze-page abc123 # show HTML payment page (default)
./un service --no-show-freeze-page abc123 # return JSON error instead
# Redeploy with new bootstrap
./un service --redeploy abc123 --bootstrap ./new-setup.sh
# Destroy service
./un service --destroy abc123
Snapshots
List Snapshots
./un snapshot --list
# Output:
# Snapshots: 3
#
# SNAPSHOT ID NAME SOURCE SIZE CREATED
# unsb-snapshot-a1b2-c3d4-e5f6-g7h8 before-upgrade session 512 MB 2h ago
# unsb-snapshot-i9j0-k1l2-m3n4-o5p6 stable-v1.0 service 1.2 GB 1d ago
Create Session Snapshot
# Snapshot with name
./un session --snapshot unsb-vm-12345 --name "before upgrade"
# Quick snapshot (auto-generated name)
./un session --snapshot unsb-vm-12345
Create Service Snapshot
# Standard snapshot (pauses container briefly)
./un service --snapshot unsb-service-abc123 --name "stable v1.0"
# Hot snapshot (no pause, may be inconsistent)
./un service --snapshot unsb-service-abc123 --hot
Restore from Snapshot
# Restore session from snapshot
./un session --restore unsb-snapshot-a1b2-c3d4-e5f6-g7h8
# Restore service from snapshot
./un service --restore unsb-snapshot-i9j0-k1l2-m3n4-o5p6
Delete Snapshot
./un snapshot --delete unsb-snapshot-a1b2-c3d4-e5f6-g7h8
Images
Images are independent, transferable container images that survive container deletion. Unlike snapshots (which live with their container), images can be shared with other users, transferred between API keys, or made public in the marketplace.
List Images
# List all images (owned + shared + public)
./un image --list
# List only your images
./un image --list owned
# List images shared with you
./un image --list shared
# List public marketplace images
./un image --list public
# Get image details
./un image --info unsb-image-xxxx-xxxx-xxxx-xxxx
Publish Images
# Publish from a stopped or frozen service
./un image --publish-service unsb-service-abc123 \
--name "My App v1.0" --description "Production snapshot"
# Publish from a snapshot
./un image --publish-snapshot unsb-snapshot-xxxx-xxxx-xxxx-xxxx \
--name "Stable Release"
# Note: Cannot publish from running containers - stop or freeze first
Create Services from Images
# Spawn a new service from an image
./un image --spawn unsb-image-xxxx-xxxx-xxxx-xxxx \
--name new-service --ports 80,443
# Clone an image (creates a copy you own)
./un image --clone unsb-image-xxxx-xxxx-xxxx-xxxx
Image Protection
# Lock image to prevent accidental deletion
./un image --lock unsb-image-xxxx-xxxx-xxxx-xxxx
# Unlock image to allow deletion
./un image --unlock unsb-image-xxxx-xxxx-xxxx-xxxx
# Delete image (must be unlocked)
./un image --delete unsb-image-xxxx-xxxx-xxxx-xxxx
Visibility & Sharing
# Set visibility level
./un image --visibility unsb-image-xxxx-xxxx-xxxx-xxxx private # owner only (default)
./un image --visibility unsb-image-xxxx-xxxx-xxxx-xxxx unlisted # can be shared
./un image --visibility unsb-image-xxxx-xxxx-xxxx-xxxx public # marketplace
# Share with specific user
./un image --grant unsb-image-xxxx-xxxx-xxxx-xxxx \
--key unsb-pk-friend-friend-friend-friend
# Revoke access
./un image --revoke unsb-image-xxxx-xxxx-xxxx-xxxx \
--key unsb-pk-friend-friend-friend-friend
# List who has access
./un image --trusted unsb-image-xxxx-xxxx-xxxx-xxxx
Transfer Ownership
# Transfer image to another API key
./un image --transfer unsb-image-xxxx-xxxx-xxxx-xxxx \
--to unsb-pk-newowner-newowner-newowner-newowner
Usage Reference
Usage: ./un [options] <source_file>
./un session [options]
./un service [options]
./un snapshot [options]
./un image [options]
./un key
Commands:
(default) Execute source file in sandbox
session Open interactive shell/REPL session
service Manage persistent services
snapshot Manage container snapshots
image Manage container images (publish, share, transfer)
key Check API key validity and expiration
Options:
-e KEY=VALUE Set environment variable (can use multiple times)
-f FILE Add input file (can use multiple times)
-a Return and save artifacts from /tmp/artifacts/
-o DIR Output directory for artifacts (default: current dir)
-p KEY Public key (or set UNSANDBOX_PUBLIC_KEY env var)
-k KEY Secret key (or set UNSANDBOX_SECRET_KEY env var)
-n MODE Network mode: zerotrust (default) or semitrusted
-v N, --vcpu N vCPU count 1-8, each vCPU gets 2GB RAM (default: 1)
-y Skip confirmation for large uploads (>1GB)
-h Show this help
Authentication (priority order):
1. -p and -k flags (public and secret key)
2. UNSANDBOX_PUBLIC_KEY + UNSANDBOX_SECRET_KEY env vars
3. ~/.unsandbox/accounts.csv (format: public_key,secret_key per line)
Session options:
-s, --shell SHELL Shell/REPL to use (default: bash)
-l, --list List active sessions
--attach ID Reconnect to existing session (ID or container name)
--kill ID Terminate a session (ID or container name)
--freeze ID Freeze a session (requires --tmux/--screen)
--unfreeze ID Unfreeze a frozen session
--boost ID Boost session resources (2 vCPU, 4GB RAM)
--boost-vcpu N Specify vCPU count for boost (1-8)
--unboost ID Return to base resources
--audit Record full session for auditing
--tmux Enable session persistence with tmux (allows reconnect)
--screen Enable session persistence with screen (allows reconnect)
Service options:
--name NAME Service name (creates new service)
--ports PORTS Comma-separated ports (e.g., 80,443)
--domains DOMAINS Custom domains (e.g., example.com,www.example.com)
--type TYPE Service type: minecraft, mumble, teamspeak, source, tcp, udp
--bootstrap CMD Bootstrap command/file/URL to run on startup
-f FILE Upload file to /tmp/ (can use multiple times)
-l, --list List all services
--info ID Get service details
--tail ID Get last 9000 lines of bootstrap logs
--logs ID Get all bootstrap logs
--freeze ID Freeze a service
--unfreeze ID Unfreeze a service
--auto-unfreeze ID Enable auto-wake on HTTP request
--no-auto-unfreeze ID Disable auto-wake on HTTP request
--show-freeze-page ID Show HTML payment page when frozen (default)
--no-show-freeze-page ID Return JSON error when frozen
--destroy ID Destroy a service
--redeploy ID Re-run bootstrap script (requires --bootstrap)
--execute ID CMD Run a command in a running service
--dump-bootstrap ID [FILE] Dump bootstrap script (for migrations)
--snapshot ID Create snapshot of session or service
--snapshot-name User-friendly name for snapshot
--hot Create snapshot without pausing (may be inconsistent)
--restore ID Restore session/service from snapshot ID
Snapshot options:
-l, --list List all snapshots
--info ID Get snapshot details
--delete ID Delete a snapshot permanently
Image options:
-l, --list [owned|shared|public] List images (all, owned, shared, or public)
--info ID Get image details
--publish-service ID Publish image from stopped/frozen service
--publish-snapshot ID Publish image from snapshot
--name NAME Name for published image
--description DESC Description for published image
--delete ID Delete image (must be unlocked)
--clone ID Clone image (creates copy you own)
--spawn ID Create service from image (requires --name)
--lock ID Lock image to prevent deletion
--unlock ID Unlock image to allow deletion
--visibility ID LEVEL Set visibility (private|unlisted|public)
--grant ID --key KEY Grant access to another API key
--revoke ID --key KEY Revoke access from API key
--transfer ID --to KEY Transfer ownership to API key
--trusted ID List API keys with access
Key options:
(no options) Check API key validity
--extend Open portal to extend an expired key
Examples:
./un script.py # execute Python script
./un -e DEBUG=1 script.py # with environment variable
./un -f data.csv process.py # with input file
./un -a -o ./bin main.c # save compiled artifacts
./un -v 4 heavy.py # with 4 vCPUs, 8GB RAM
./un session # interactive bash session
./un session --tmux # bash with reconnect support
./un session --list # list active sessions
./un session --attach unsb-vm-12345 # reconnect to session
./un session --kill unsb-vm-12345 # terminate a session
./un session --freeze unsb-vm-12345 # freeze session
./un session --unfreeze unsb-vm-12345 # unfreeze session
./un session --boost unsb-vm-12345 # boost resources
./un session --unboost unsb-vm-12345 # return to base
./un session --shell python3 # Python REPL
./un session --shell node # Node.js REPL
./un session -n semitrusted # session with network access
./un session --audit -o ./logs # record session for auditing
./un service --name web --ports 80 # create web service
./un service --list # list all services
./un service --logs abc123 # view bootstrap logs
./un key # check API key
./un key --extend # extend expired key
./un snapshot --list # list all snapshots
./un session --snapshot unsb-vm-123 # snapshot a session
./un service --snapshot abc123 # snapshot a service
./un session --restore unsb-snapshot-xxxx # restore from snapshot
./un image --list # list all images
./un image --list owned # list your images
./un image --publish-service abc # publish image from service
./un image --spawn img123 --name x # create service from image
./un image --grant img --key pk # share image with user
CLI Inception
The UN CLI has been implemented in 42 programming languages, demonstrating that the unsandbox API can be accessed from virtually any environment.
License
PUBLIC DOMAIN - NO LICENSE, NO WARRANTY
This is free public domain software for the public good of a permacomputer hosted
at permacomputer.com - an always-on computer by the people, for the people. One
that is durable, easy to repair, and distributed like tap water for machine
learning intelligence.
The permacomputer is community-owned infrastructure optimized around four values:
TRUTH - First principles, math & science, open source code freely distributed
FREEDOM - Voluntary partnerships, freedom from tyranny & corporate control
HARMONY - Minimal waste, self-renewing systems with diverse thriving connections
LOVE - Be yourself without hurting others, cooperation through natural law
This software contributes to that vision by enabling code execution across all 42
programming languages through a unified interface, accessible to everyone. Code is
seeds to sprout on any abandoned technology.
Learn more: https://www.permacomputer.com
Anyone is free to copy, modify, publish, use, compile, sell, or distribute this
software, either in source code form or as a compiled binary, for any purpose,
commercial or non-commercial, and by any means.
NO WARRANTY. THE SOFTWARE IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND.
That said, our permacomputer's digital membrane stratum continuously runs unit,
integration, and functional tests on all its own software - with our permacomputer
monitoring itself, repairing itself, with minimal human guidance in the loop.
Our agents do their best.
Copyright 2025 TimeHexOn & foxhop & russell@unturf
https://www.timehexon.com
https://www.foxhop.net
https://www.unturf.com/software