Skip to content

osal #

Operating System Abstraction Layer (OSAL)

A comprehensive operating system abstraction layer for V that provides platform-independent system operations, process management, and network utilities.

Features

  • Platform detection and system information
  • Process execution and management
  • Network utilities (ping, TCP port testing)
  • Environment variable handling
  • File system operations
  • SSH key management
  • Profile path management

Platform Detection

import freeflowuniverse.crystallib.osal

// Get platform type
platform := osal.platform()
if platform == .osx {
    // macOS specific code
}

// Platform-specific checks
if osal.is_linux() {
    // Linux specific code
}
if osal.is_osx_arm() {
    // Apple Silicon specific code
}

// CPU architecture
cpu := osal.cputype()
if cpu == .arm {
    // ARM specific code
}

// System information
hostname := osal.hostname()!
init_system := osal.initname()!  // e.g., systemd, bash, zinit

Process Execution

The module provides flexible process execution with extensive configuration options:

// Simple command execution
job := osal.exec(cmd: 'ls -la')!
println(job.output)

// Execute with error handling
job := osal.exec(Command{
    cmd: 'complex_command'
    timeout: 3600        // timeout in seconds
    retry: 3             // retry count
    work_folder: '/tmp'  // working directory
    environment: {       // environment variables
        'PATH': '/usr/local/bin'
    }
    stdout: true         // show output
    raise_error: true    // raise error on failure
})!

// Silent execution
output := osal.execute_silent('command')!

// Interactive shell execution
osal.execute_interactive('bash command')!

// Debug mode execution
output := osal.execute_debug('command')!

Job Status and Error Handling

// Check job status
if job.status == .done {
    println('Success!')
} else if job.status == .error_timeout {
    println('Command timed out')
}

// Error handling with specific error types
job := osal.exec(cmd: 'invalid_command') or {
    match err.error_type {
        .exec { println('Execution error') }
        .timeout { println('Command timed out') }
        .args { println('Invalid arguments') }
        else { println(err) }
    }
    return
}

Network Utilities

Ping

// Simple ping
result := osal.ping(address: '8.8.8.8')!
assert result == .ok

// Advanced ping configuration
result := osal.ping(PingArgs{
    address: '8.8.8.8'
    count: 3        // number of pings
    timeout: 2      // timeout in seconds
    retry: 1        // retry attempts
})!

match result {
    .ok { println('Host is reachable') }
    .timeout { println('Host timed out') }
    .unknownhost { println('Unknown host') }
}

TCP Port Testing

// Test if port is open
is_open := osal.tcp_port_test(TcpPortTestArgs{
    address: '192.168.1.1'
    port: 22
    timeout: 2000  // milliseconds
})

if is_open {
    println('Port is open')
}

// Get public IP address
pub_ip := osal.ipaddr_pub_get()!
println('Public IP: ${pub_ip}')

Profile Management

Manage system PATH and other profile settings:

// Add/remove paths from system PATH
osal.profile_path_add_remove(
    paths2delete: 'go/bin',
    paths2add: '~/hero/bin,~/usr/local/bin'
)!

Environment Variables

// Get environment variable
value := osal.env_get('PATH')!

// Set environment variable
osal.env_set('MY_VAR', 'value')!

// Check if environment variable exists
exists := osal.env_exists('MY_VAR')

Notes

  • All commands are executed from temporary scripts in /tmp/execscripts
  • Failed script executions are preserved for debugging
  • Successful script executions are automatically cleaned up
  • Platform-specific behavior is automatically handled
  • Timeout and retry mechanisms are available for robust execution
  • Environment variables and working directories can be specified per command
  • Interactive and non-interactive modes are supported
  • Debug mode provides additional execution information

Error Handling

The module provides detailed error information:

  • Exit codes
  • Standard output and error streams
  • Execution time and duration
  • Process status
  • Retry counts
  • Error types (execution, timeout, arguments)

Platform Support

  • macOS (Intel and ARM)
  • Ubuntu
  • Alpine Linux
  • Arch Linux
  • SUSE (partial)

CPU architectures:- Intel (x86_64)

  • ARM (arm64/aarch64)
  • 32-bit variants (intel32, arm32)

fn bin_path #

fn bin_path() !string

fn cmd_add #

fn cmd_add(args_ CmdAddArgs) !

copy a binary to the right location on the local computer . e.g. is /usr/local/bin on linux . e.g. is ~/hero/bin on osx . will also add the bin location to the path of .zprofile and .zshrc (different per platform)

fn cmd_delete #

fn cmd_delete(cmd string) !

delete cmds from found locations can be one command of multiple

fn cmd_exists #

fn cmd_exists(cmd string) bool

fn cmd_exists_profile #

fn cmd_exists_profile(cmd string) bool

fn cmd_path #

fn cmd_path(cmd string) !string

is same as executing which in OS returns path or error

fn cmd_to_script_path #

fn cmd_to_script_path(cmd Command) !string

will return temporary path which then can be executed, is a helper function for making script out of command

fn cputype #

fn cputype() CPUType

fn cputype_enum_from_string #

fn cputype_enum_from_string(cpytype string) CPUType

Returns the enum value that matches the provided string for CPUType

fn dir_delete #

fn dir_delete(path string) !

remove all if it exists

fn dir_ensure #

fn dir_ensure(path string) !

remove all if it exists

fn dir_reset #

fn dir_reset(path string) !

remove all if it exists and then (re-)create

fn done_delete #

fn done_delete(key string) !

fn done_exists #

fn done_exists(key string) bool

fn done_get #

fn done_get(key string) ?string

fn done_get_int #

fn done_get_int(key string) int

fn done_get_str #

fn done_get_str(key string) string

fn done_print #

fn done_print() !

fn done_reset #

fn done_reset() !

fn done_set #

fn done_set(key string, val string) !

fn download #

fn download(args_ DownloadArgs) !pathlib.Path

if name is not specified, then will be the filename part if the last ends in an extension like .md .txt .log .text ... the file will be downloaded

fn env_get #

fn env_get(key string) !string

Returns the requested environment variable if it exists or throws an error if it does not

fn env_get_all #

fn env_get_all() map[string]string

Returns all existing environment variables

fn env_get_default #

fn env_get_default(key string, def string) string

Returns the requested environment variable if it exists or returns the provided default value if it does not

fn env_set #

fn env_set(args EnvSet)

Sets an environment if it was not set before, it overwrites the enviroment variable if it exists and if overwrite was set to true (default)

fn env_set_all #

fn env_set_all(args EnvSetAll)

Allows to set multiple enviroment variables in one go, if clear_before_set is true all existing environment variables will be unset before the operation, if overwrite_if_exists is set to true it will overwrite all existing enviromnent variables

fn env_unset #

fn env_unset(key string)

Unsets an environment variable

fn env_unset_all #

fn env_unset_all()

Unsets all environment variables

fn exec #

fn exec(cmd Command) !Job

cmd is the cmd to execute can use ' ' and spaces . if \n in cmd it will write it to ext and then execute with bash . if die==false then will just return returncode,out but not return error . if stdout will show stderr and stdout . . if cmd starts with find or ls, will give to bash -c so it can execute . if cmd has no path, path will be found . . Command argument: .

 name                             string // to give a name to your command, good to see logs...
 cmd                              string
 description                      string
 timeout                          int  = 3600 // timeout in sec
 stdout                           bool = true
 stdout_log                       bool = true
 raise_error                      bool = true // if false, will not raise an error but still error report
 ignore_error                     bool // means if error will just exit and not raise, there will be no error reporting
 work_folder                      string // location where cmd will be executed
 environment                      map[string]string // env variables
 ignore_error_codes               []int
 scriptpath                       string // is the path where the script will be put which is executed
 scriptkeep                       bool   // means we don't remove the script
 debug                            bool   // if debug will put +ex in the script which is being executed and will make sure script stays
 shell                            bool   // means we will execute it in a shell interactive
 retry                            int
 interactive 					 	bool = true // make sure we run on non interactive way
 async							 bool
 runtime							 RunTime (.bash, .python)

 returns Job:
 start  time.Time
 end    time.Time
 cmd    Command
 output []string
 error    []string
 exit_code int
 status JobStatus
 process os.Process

return Job .

fn exec_string #

fn exec_string(cmd Command) !string

cmd is the cmd to execute can use ' ' and spaces if \n in cmd it will write it to ext and then execute with bash if die==false then will just return returncode,out but not return error if stdout will show stderr and stdout

if cmd starts with find or ls, will give to bash -c so it can execute if cmd has no path, path will be found $... are remplaced by environment arguments TODO:implement

Command argument: cmd string timeout int = 600 stdout bool = true die bool = true debug bool

return what needs to be executed can give it to bash -c ...

fn execute_debug #

fn execute_debug(cmd string) !string

fn execute_interactive #

fn execute_interactive(cmd string) !

shortcut to execute a job interactive means in shell

fn execute_ok #

fn execute_ok(cmd string) bool

executes a cmd, if not error return true

fn execute_silent #

fn execute_silent(cmd string) !string

shortcut to execute a job silent

fn execute_stdout #

fn execute_stdout(cmd string) !string

shortcut to execute a job to stdout

fn file_read #

fn file_read(path string) !string

fn file_write #

fn file_write(path string, text string) !

fn get_logger #

fn get_logger() log.Log

Returns a logger object and allows you to specify via environment argument OSAL_LOG_LEVEL the debug level

fn get_ssh_key #

fn get_ssh_key(key_name string, config SSHConfig) ?SSHKey

Returns a specific SSH key with the given name from the default SSH directory (~/.ssh)

fn hero_path #

fn hero_path() !string

fn hostname #

fn hostname() !string

fn initname #

fn initname() !string

e.g. systemd, bash, zinit

fn ipaddr_pub_get #

fn ipaddr_pub_get() !string

Returns the ipaddress as known on the public side is using resolver4.opendns.com

fn is_linux #

fn is_linux() bool

fn is_linux_arm #

fn is_linux_arm() bool

fn is_linux_intel #

fn is_linux_intel() bool

fn is_osx #

fn is_osx() bool

fn is_osx_arm #

fn is_osx_arm() bool

fn is_osx_intel #

fn is_osx_intel() bool

fn is_ubuntu #

fn is_ubuntu() bool

fn load_env_file #

fn load_env_file(file_path string) !

fn memdb_exists #

fn memdb_exists(key string) bool

fn memdb_get #

fn memdb_get(key string) string

fn memdb_set #

fn memdb_set(key string, val string)

fn new_ssh_key #

fn new_ssh_key(key_name string, config SSHConfig) !SSHKey

Creates a new SSH key pair to the specified directory

fn package_install #

fn package_install(name_ string) !

install a package will use right commands per platform

fn package_refresh #

fn package_refresh() !

update the package list

fn package_remove #

fn package_remove(name_ string) !

Remove a package using the appropriate command for each platform

fn ping #

fn ping(args PingArgs) !PingResult

if reached in timout result will be True address is e.g. 8.8.8.8 ping means we check if the destination responds

fn platform #

fn platform() PlatformType

fn platform_enum_from_string #

fn platform_enum_from_string(platform string) PlatformType

fn process_exists #

fn process_exists(pid int) bool

fn process_exists_byname #

fn process_exists_byname(name string) !bool

fn process_kill_recursive #

fn process_kill_recursive(args ProcessKillArgs) !

kill process and all the ones underneith

fn processinfo_children #

fn processinfo_children(pid int) !ProcessMap

get all children of 1 process

fn processinfo_get #

fn processinfo_get(pid int) !ProcessInfo

get process info from 1 specific process returns

 pub struct ProcessInfo {
 pub mut:
    cpu_perc	f32
    mem_perc	f32
    cmd 		string
    pid 		int
    ppid 		int
    //resident memory
    rss			int
 }

fn processinfo_get_byname #

fn processinfo_get_byname(name string) ![]ProcessInfo

fn processinfo_with_children #

fn processinfo_with_children(pid int) !ProcessMap

return the process and its children

fn processmap_get #

fn processmap_get() !ProcessMap

make sure to use new first, so that the connection has been initted then you can get it everywhere

fn profile_path #

fn profile_path() string

fn profile_path_add_hero #

fn profile_path_add_hero() !string

fn profile_path_add_remove #

fn profile_path_add_remove(args_ ProfilePathAddRemoveArgs) !

add and/or remove paths from profiles if paths_profile not specified it will walk over all of them

fn profile_path_source #

fn profile_path_source() string

return the source statement if the profile exists

fn profile_path_source_and #

fn profile_path_source_and() string

return source $path && . or empty if it doesn't exist

fn profile_paths_all #

fn profile_paths_all() ![]string

return possible profile paths in OS

fn profile_paths_preferred #

fn profile_paths_preferred() ![]string

fn rm #

fn rm(todelete_ string) !

can be list of dirs, files ~ supported can be \n or , separated

fn sleep #

fn sleep(duration int)

sleep in seconds

fn tcp_port_test #

fn tcp_port_test(args TcpPortTestArgs) bool

test if a tcp port answers

 address string //192.168.8.8
 port int = 22
 timeout u16 = 2000 // total time in milliseconds to keep on trying

fn user_add #

fn user_add(args UserArgs) !int

add's a user if the user does not exist yet

fn user_exists #

fn user_exists(username string) bool

fn user_id_get #

fn user_id_get(username string) !int

fn usr_local_path #

fn usr_local_path() !string

/usr/local on linux, ${os.home_dir()}/hero on osx

fn whoami #

fn whoami() !string

fn write_flags #

fn write_flags[T](options T) string

enum CPUType #

enum CPUType {
	unknown
	intel
	arm
	intel32
	arm32
}

enum ErrorType #

enum ErrorType {
	exec
	timeout
	args
}

enum JobStatus #

enum JobStatus {
	init
	running
	error_exec
	error_timeout
	error_args
	done
}

enum PMState #

enum PMState {
	init
	ok
	old
}

enum PingResult #

enum PingResult {
	ok
	timeout     // timeout from ping
	unknownhost // means we don't know the hostname its a dns issue
}

enum PlatformType #

enum PlatformType {
	unknown
	osx
	ubuntu
	alpine
	arch
	suse
}

enum RunTime #

enum RunTime {
	bash
	python
	heroscript
	herocmd
	v
}

struct CmdAddArgs #

struct CmdAddArgs {
pub mut:
	cmdname string
	source  string @[required] // path where the binary is
	symlink bool // if rather than copy do a symlink
	reset   bool = true // if existing cmd will delete
	// bin_repo_url string = 'https://github.com/freeflowuniverse/freeflow_binary' // binary where we put the results
}

struct Command #

@[params]
struct Command {
pub mut:
	name               string // to give a name to your command, good to see logs...
	cmd                string
	description        string
	timeout            int  = 3600 // timeout in sec
	stdout             bool = true
	stdout_log         bool = true
	raise_error        bool = true // if false, will not raise an error but still error report
	ignore_error       bool              // means if error will just exit and not raise, there will be no error reporting
	work_folder        string            // location where cmd will be executed
	environment        map[string]string // env variables
	ignore_error_codes []int
	scriptpath         string // is the path where the script will be put which is executed
	scriptkeep         bool   // means we don't remove the script
	debug              bool   // if debug will put +ex in the script which is being executed and will make sure script stays
	shell              bool   // means we will execute it in a shell interactive
	retry              int
	interactive        bool = true
	async              bool
	runtime            RunTime
}

struct DownloadArgs #

@[params]
struct DownloadArgs {
pub mut:
	name        string // optional (otherwise derived out of filename)
	url         string
	reset       bool   // will remove
	hash        string // if hash is known, will verify what hash is
	dest        string // if specified will copy to that destination	
	timeout     int = 180
	retry       int = 3
	minsize_kb  u32 = 10 // is always in kb
	maxsize_kb  u32
	expand_dir  string
	expand_file string
}

struct EnvSet #

@[params]
struct EnvSet {
pub mut:
	key       string @[required]
	value     string @[required]
	overwrite bool = true
}

struct EnvSetAll #

@[params]
struct EnvSetAll {
pub mut:
	env                 map[string]string
	clear_before_set    bool
	overwrite_if_exists bool = true
}

struct Job #

struct Job {
pub mut:
	start     time.Time
	end       time.Time
	cmd       Command
	output    string
	error     string
	exit_code int
	status    JobStatus
	process   ?&os.Process @[skip; str: skip]
	runnr     int // nr of time it runs, is for retry
}

fn (Job) execute_retry #

fn (mut job Job) execute_retry() !

execute the job and wait on result will retry as specified

fn (Job) execute #

fn (mut job Job) execute() !

execute the job, start process, process will not be closed . important you need to close the process later by job.close()! otherwise we get zombie processes

fn (Job) wait #

fn (mut job Job) wait() !

wait till the job finishes or goes in error

fn (Job) process #

fn (mut job Job) process() !

process (read std.err and std.out of process)

fn (Job) close #

fn (mut job Job) close() !

will wait & close

struct JobError #

struct JobError {
	Error
pub mut:
	job        Job
	error_type ErrorType
}

struct PingArgs #

@[params]
struct PingArgs {
pub mut:
	address string @[required]
	count   u8  = 1 // the ping is successful if it got count amount of replies from the other side
	timeout u16 = 1 // the time in which the other side should respond in seconds
	retry   u8
}

struct ProcessInfo #

@[heap]
struct ProcessInfo {
pub mut:
	cpu_perc f32
	mem_perc f32
	cmd      string
	pid      int
	ppid     int // parentpid
	// resident memory
	rss int
}

fn (ProcessInfo) str #

fn (mut p ProcessInfo) str() string

struct ProcessKillArgs #

@[params]
struct ProcessKillArgs {
pub mut:
	name string
	pid  int
}

struct ProcessMap #

@[heap]
struct ProcessMap {
pub mut:
	processes []ProcessInfo
	lastscan  time.Time
	state     PMState
	pids      []int
}

struct ProfilePathAddRemoveArgs #

@[params]
struct ProfilePathAddRemoveArgs {
pub mut:
	paths_profile string
	paths2add     string
	paths2delete  string
	allprofiles   bool
}

struct SSHConfig #

@[params]
struct SSHConfig {
pub:
	directory string = os.join_path(os.home_dir(), '.ssh')
}

struct SSHKey #

@[noinit]
struct SSHKey {
pub:
	name      string
	directory string
}

fn (SSHKey) public_key_path #

fn (key SSHKey) public_key_path() !pathlib.Path

returns the public ssh key's path of the keypair

fn (SSHKey) private_key_path #

fn (key SSHKey) private_key_path() !pathlib.Path

returns the private ssh key's path of the keypair

fn (SSHKey) public_key #

fn (key SSHKey) public_key() !string

returns the public ssh key of the keypair

fn (SSHKey) private_key #

fn (key SSHKey) private_key() !string

returns the private ssh key of the keypair

struct TcpPortTestArgs #

@[params]
struct TcpPortTestArgs {
pub mut:
	address string @[required] // 192.168.8.8
	port    int = 22
	timeout u16 = 2000 // total time in milliseconds to keep on trying
}

struct UserArgs #

@[params]
struct UserArgs {
pub mut:
	name string @[required]
}