starcm
"star-cm"
- A rudimentary configuration management language that utilizes Starlark instead of json or yaml.
- Why Starlark? It provides variables, functions, loops, and lots more "for free" inside of the configuration files!
- Starcm is not intended to be a full replacement for tools like Chef or Ansible, but starcm can be used to bootstrap these tools and many others through features like
exec()
for calling binaries, template()
for rendering templated files, load_dynamic()
for loading additional starcm config files dynamically, and much more!
Goal
Starcm is intended to become a viable alternative for tools like macadmins/installapplications, facebookincubator/go2chef, and google/glazier.
Introduction to starcm language
exec
Let's look at a simple starcm file that uses the exec
function: examples/echo.star
% cat examples/echo/echo.star
load("starcm", "exec")
exec(
name = "hello_from_starcm",
cmd = "echo",
args = ["hello from echo.star!"],
live_output = True
)
We can execute it with
go run main.go --root_file examples/echo/echo.star
which outputs
% go run main.go --root_file examples/echo/echo.star
INFO: 2024/06/01 23:46:17 starting starcm...
INFO: 2024/06/01 23:46:17 [hello_from_starcm]: Starting...
hello from echo.star!
Starcm's exec
can also handle non-zero exit codes.
handling non-zero exit codes
See examples/exec/exit_non_zero/expect_exit_code_non_zero.star.
If exec
exits non-zero but we don't provide expected_exit_code
load("starcm", "exec")
a = exec(
name = "explicitly exit 2",
cmd = "sh",
args = ["-c", "echo 'we expect to exit 2'; exit 2"],
# expected_exit_code = 2,
)
print(a)
INFO: 2024/05/29 22:49:07 starting starcm...
INFO: 2024/05/29 22:49:07 [explicitly exit 2]: Starting...
result(changed = True, diff = "", error = "exit status 2", name = "explicitly exit 2", output = "we expect to exit 2\n", success = False)
there will be a failure (result(..., success=False)
) because the default expected_error_code
is 0
. But if we set it to 2
then this succeeds!
load("starcm", "exec")
a = exec(
name = "explicitly exit 2",
cmd = "sh",
args = ["-c", "echo 'we expect to exit 2'; exit 2"],
expected_exit_code = 2,
)
print(a)
INFO: 2024/05/29 22:51:04 starting starcm...
INFO: 2024/05/29 22:51:04 [explicitly exit 2]: Starting...
result(changed = True, diff = "", error = "exit status 2", name = "explicitly exit 2", output = "we expect to exit 2\n", success = True)
Nearly all starcm functions return this result()
struct which we can combine with conditionals to create powerful and flexible workflows.
if statements
Starlark, and by extension starcm, supports if
statements. Take examples/if_statements/if_statements.star for example. If the exec()
succeeds, we print party!
.
load("starcm", "exec")
a = exec(
name = "explicitly exit 2",
cmd = "sh",
args = ["-c", "echo 'we expect to exit 2'; exit 2"],
expected_exit_code = 2,
)
if a.success == True:
print("party!")
else:
print("no party :(")
Running go run main.go --root_file examples/if_statements/if_statements.star
results in
% go run main.go --root_file examples/if_statements.star
INFO: 2024/06/01 23:52:56 starting starcm...
INFO: 2024/06/01 23:52:56 [explicitly exit 2]: Starting...
party!
We can also implement this same conditional behavior with a starcm-specific construct called only_if
.
only_if
See examples/only_if/only_if.star:
load("shellout", "exec")
load("write", "write")
a = exec(
name = "explicitly exit 2",
cmd = "sh",
args = ["-c", "echo 'we expect to exit 2'; exit 2"],
expected_exit_code = 2,
live_output = True,
)
if not(a.success):
write(
name = "print_not_success_#1",
str = "a.success: %s #1" % (a.success),
)
write(
name = "print_not_success_#2",
str = "a.success: %s #2" % (a.success),
only_if = a.success == False
)
In this example
if not(a.success):
write(
name = "print_not_success_#1",
str = "a.success: %s #1" % (a.success),
)
is essentially equivalent to
write(
name = "print_not_success_#2",
str = "a.success: %s #2" % (a.success),
only_if = a.success == False
)
with one key difference: only_if
produces a log message indicating that write(name=print_not_success, ...)
was skipped due to the only_if
condition being false. This is can be useful for debugging.
% go run main.go -v 2 --root_file examples/only_if.star
INFO: 2024/06/04 23:04:00 starting starcm...
INFO: 2024/06/04 23:04:00 [LoadFromFile]: loading file "examples/only_if.star"
INFO: 2024/06/04 23:04:00 [explicitly exit 2]: Executing...
we expect to exit 2
INFO: 2024/06/04 23:04:00 [explicitly exit 2]: expectedExitCode: 2
INFO: 2024/06/04 23:04:00 [explicitly exit 2]: actualExitCode: 2
INFO: 2024/06/04 23:04:00 [print_not_success_#2]: skipping write(name="print_not_success_#2") because only_if was false
Notice that there is no log message regarding print_not_success_#1
. Normal if
statements are not executed at all if the condition is false, whereas only_if
logs that print_not_success_#2
was skipped.
Advanced functionality
See the examples folder for more examples of what starcm can do. There's lots it can do such as downloading files (with hash checking), dynamically loading additional .star
files, rendering templates, and combining all the cabilities via macros, thanks to Starlark.