Arc is an infrastructure automation tool similar to Ansible, but using Lua for scripting. It allows you to define tasks that are executed on remote systems via SSH, with a powerful and flexible API.
Clone the repository and build the project:
git clone https://github.com/myFavShrimp/arc.git
cd arc
cargo build --release
The binary will be available at target/release/arc
.
Arc uses a Lua script (arc.lua
) to define targets and tasks.
-- Define a target system
targets.systems["frontend-server"] = {
address = "192.168.1.100",
user = "root",
}
targets.systems["api-server"] = {
address = "192.168.1.101",
user = "root",
port = 42, -- defaults to 22 if not specified
}
-- Define a group of systems
targets.groups["web-servers"] = {
members = {"frontend-server", "api-server"}
}
-- Define tasks
tasks["check_nginx"] = {
handler = function (system)
local result = system:run_command("nginx -v")
return result.exit_code == 0
end,
tags = {"nginx"}
}
tasks["install_nginx"] = {
handler = function (system)
local nginx_installed = tasks["check_nginx"].result
if nginx_installed == false then
return system:run_command("apt install nginx -y")
end
end,
dependencies = {"check_nginx"},
tags = {"nginx"}
}
To run all tasks on all systems:
arc
To run tasks with a specific tag:
arc --tag nginx
To run tasks on specific a group:
arc --group web-servers
Arc provides several command-line options:
Usage:
arc [OPTIONS]
Options:
-v, --verbose... Enable verbose output (repeat for increased verbosity)
-t, --tag <TAG> Filter tasks by tag
-g, --group <GROUP> Run tasks only on specific groups
-d, --dry-run Perform a dry run without executing commands or modifying the file system
-h, --help Print help information
Run tasks with the "nginx" tag:
arc -t nginx
Run tasks on the "web-servers" group with verbose output:
arc -g web-servers -v
Run tasks with multiple tags and groups:
arc -t nginx -t security -g web-servers -g database-servers
Perform a dry run without executing commands:
arc --dry-run
The system
object represents a connection to a remote system and is provided to task handlers.
name
: The name of the system as defined in targets.systemsaddress
: The IP address of the systemport
: The SSH port of the systemuser
: The SSH user used to connect to the system
-
run_command(cmd)
: Execute a command on the remote system- Parameters:
cmd
(string) - The command to execute - Returns: A table with
stdout
,stderr
, andexit_code
- Parameters:
-
file(path)
: Get a File object representing a file on the remote system- Parameters:
path
(string) - Path to the file - Returns: A File object
- Parameters:
-
directory(path)
: Get a Directory object representing a directory on the remote system- Parameters:
path
(string) - Path to the directory - Returns: A Directory object
- Parameters:
The File object represents a file on a remote system.
path
: Path to the file (can be read and set, setting renames the file)content
: Text content of the file (can be read and set)permissions
: File permissions (can be read and set as numeric mode)
-
metadata()
: Get file metadata- Returns: A table with file metadata (see Metadata Structure below)
-
remove()
: Remove the file
Example:
-- Create and write to a file
local config_file = system:file("/etc/nginx/sites-available/default")
config_file.content = "server {\n listen 80 default_server;\n root /var/www/html;\n}"
print(format.to_json(config_file.metadata()))
-- Create, move and then delete a file
local file = system:file("/path/to/file.txt")
file.content = "New content" -- Write to file
file.permissions = tonumber("755", 8) -- Set permissions
file.path = "/new-path/to/renamed-file.txt" -- Rename file
file:remove() -- Delete file
The Directory object represents a directory on a remote system.
path
: Path to the directory (can be read and set, setting renames the directory)permissions
: Directory permissions (can be read and set as numeric mode)
create()
: Create the directoryremove()
: Remove the directorymetadata()
: Get directory metadata
Example:
-- Create directory structure
local app_dir = system:directory("/var/www/myapp")
app_dir:create()
-- Set permissions
app_dir.permissions = tonumber("755", 8)
-- Create, move and then delete a directory
local dir = system:directory("/path/to/dir")
dir:create() -- Create directory
dir.path = "/path/to/renamed-dir" -- Rename directory
dir:remove() -- Delete directory
When calling the metadata()
method on File or Directory objects, a table with the following fields is returned:
path
: Path to the file or directorysize
: Size in bytes (number)permissions
: Permission mode (number)type
: Type of the item ("file", "directory", or "unknown")uid
: User ID of the owner (number)gid
: Group ID of the owner (number)accessed
: Last access time as a Unix timestamp (number)modified
: Last modification time as a Unix timestamp (number)
Example:
local file = system:file("/etc/hostname")
local metadata = file:metadata()
print("File size: " .. metadata.size)
print("File type: " .. metadata.type)
print("File permissions: " .. metadata.permissions)
print("Last modified: " .. metadata.modified)
The env
module provides access to environment variables.
Arc automatically loads variables from .env
files in your project directory, making it easy to manage secrets, configuration values, and environment-specific settings without hardcoding sensitive information in your scripts.
get(var_name)
: Get the value of an environment variable- Parameters:
var_name
(string) - Name of the environment variable - Returns: Value of the environment variable (string) or nil if not set
- Parameters:
Example:
local home_dir = env.get("HOME")
local user = env.get("USER")
if home_dir then
print("Home directory: " .. home_dir)
end
-- Use environment variables in tasks
tasks["deploy_app"] = {
handler = function(system)
local app_version = env.get("APP_VERSION") or "latest"
system:run_command("docker pull myapp:" .. app_version)
end
}
The host
module provides functions for interacting with the local system (where Arc is running).
-
run_command(cmd)
: Execute a command on the local system- Parameters:
cmd
(string) - The command to execute - Returns: A table with
stdout
,stderr
, andexit_code
- Parameters:
-
file(path)
: Get a File object representing a file on the local system- Parameters:
path
(string) - Path to the file - Returns: A File object
- Parameters:
-
directory(path)
: Get a Directory object representing a directory on the local system- Parameters:
path
(string) - Path to the directory - Returns: A Directory object
- Parameters:
Example:
-- Execute a local command
local result = host:run_command("ls -la")
print("Command output: " .. result.stdout)
-- Work with a local file
local local_file = host:file("/tmp/example.txt")
local_file.content = "This is a local file"
-- Create a local directory
local local_dir = host:directory("/tmp/arc_test")
local_dir:create()
-- Copy files between systems
tasks["copy_config"] = {
handler = function(system)
-- Read a local template
local template = host:file("templates/nginx.conf").content
-- Apply configuration and write to remote system
local remote_config = system:file("/etc/nginx/nginx.conf")
remote_config:write(template)
-- Restart service
system:run_command("systemctl restart nginx")
end
}
The format
object provides utilities for working with JSON.
-
to_json(value)
: Convert a Lua value to JSON- Parameters:
value
(any) - Value to convert - Returns: JSON string
- Parameters:
-
to_json_pretty(value)
: Convert a Lua value to pretty-printed JSON- Parameters:
value
(any) - Value to convert - Returns: JSON string
- Parameters:
-
from_json(json_string)
: Parse a JSON string to a Lua value- Parameters:
json_string
(string) - JSON string to parse - Returns: Parsed Lua value
- Parameters:
Example:
-- Read a JSON configuration file
local config_file = system:file("/etc/myapp/config.json")
local config = format.from_json(config_file.content)
-- Modify configuration
config.debug = true
config.log_level = "info"
-- Write back to the file
config_file.content = format.to_json(config)
-- Working with JSON APIs
tasks["update_api_config"] = {
handler = function(system)
-- Get current config from an API
local result = system:run_command("curl -s http://localhost:8080/api/config")
local api_config = format.from_json(result.stdout)
-- Update configuration
api_config.settings.cache_ttl = 3600
-- Send updated config back to API
local json_config = format.to_json(api_config)
system:run_command('curl -X POST -H "Content-Type: application/json" -d \'' .. json_config .. '\' http://localhost:8080/api/config')
end
}
The template
object allows rendering templates.
render(template_content, context)
: Render a template with given context- Parameters:
template_content
(string) - Template contentcontext
(table) - Variables to use for template rendering
- Returns: Rendered template as string
- Parameters:
Example:
-- Complex example with configuration management
tasks["configure_web_server"] = {
handler = function(system)
-- Load a template from a local file
local template_content = local:file("templates/nginx.conf.template").content
-- Define context variables
local context = {
worker_processes = 4,
worker_connections = 1024,
server_name = system.name .. ".example.com",
document_root = "/var/www/" .. system.name,
environment = env.get("ENVIRONMENT") or "production"
}
-- Render and deploy configuration
local config = template.render(template_content, context)
system:file("/etc/nginx/nginx.conf").content = config
-- Validate and reload
local validation = system:run_command("nginx -t")
if validation.exit_code == 0 then
system:run_command("systemctl reload nginx")
return true
else
error("Nginx configuration is invalid: " .. validation.stderr)
end
end
}
Various logging functions are available on the log
object.
debug(message)
: Log a debug messageinfo(message)
: Log an info messagewarn(message)
: Log a warning messageerror(message)
: Log an error message
Example:
tasks["provision_database"] = {
handler = function(system)
log.info("Starting database provisioning on " .. system.name)
local result = system:run_command("systemctl status postgresql")
if result.exit_code ~= 0 then
log.warn("PostgreSQL not running, attempting to install")
local install = system:run_command("apt-get install -y postgresql")
if install.exit_code ~= 0 then
log.error("Failed to install PostgreSQL: " .. install.stderr)
return false
end
}
log.debug("PostgreSQL installed and running")
return true
end
}
Tasks are defined using the tasks
global table:
tasks["task_name"] = {
handler = function(system)
-- Task implementation
return result
end,
dependencies = {"other_task"}, -- Optional
tags = {"tag1", "tag2"}, -- Optional
groups = {"group1", "group2"}, -- Optional
}
handler
: The function that implements the task. Takes a system object and returns a result.dependencies
: Array of task names that must be executed before this task.tags
: Array of tags associated with the task, used for filtering.groups
: Array of group names this task should run on. If not specified, the task runs on all groups.
Within a task, you can access the result of a previously executed dependency using:
local dependency_result = tasks["dependency_task_name"].result
Contributions are welcome! Please feel free to submit a Pull Request.
This project is licensed under the MIT License - see the LICENSE file for details.