Scriptish is a Golang library. It helps you port UNIX shell scripts to Golang.
It is released under the 3-clause New BSD license. See LICENSE.md for details.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountLines(),
).Exec().ParseInt()
(We're going to create Scriptish for other languages too, and we'll update this README when those are available!)
- Introduction
- Why Use Scriptish?
- How Does It Work?
- Creating A Pipeline
- Running An Existing Pipeline
- Passing Parameters Into Pipelines
- Calling A Pipeline From Another Pipeline
- Capturing The Output
- Pipelines vs Lists
- Creating A List
- Running An Existing List
- Passing Parameters Into Lists
- Calling A List From Another List Or Pipeline
- Pipelines, Lists and Sequences
- UNIX Shell String Expansion
- From Bash To Scriptish
- Sources
- Filters
- AppendToTempFile()
- CountLines()
- CountWords()
- CutFields()
- DropEmptyLines()
- Grep()
- GrepV()
- Head()
- Rsort()
- RunPipeline()
- Sort()
- StripExtension()
- SwapExtensions()
- Tail()
- Tr()
- TrimSuffix()
- TrimWhitespace()
- Uniq()
- XargsBasename()
- XargsCat()
- XargsDirname()
- XargsRmFile()
- XargsTestFilepathExists()
- XargsTruncateFiles()
- Sinks
- Redirects
- What Are Redirects?
- How Do We Use Redirects?
- AppendStdoutToFilename()
- AppendStderrToFilename()
- AttachOsStdin()
- OverwriteFilenameWithStdout()
- OverwriteFilenameWithStderr()
- RedirectStderrToStdout()
- RedirectStderrToDevNull()
- RedirectStderrToTextReaderWriter()
- RedirectStdoutToDevNull()
- RedirectStdoutToStderr()
- RedirectStdoutToTextReaderWriter()
- Builtins
- Capture Methods
- Logic Calls
- Errors
- Inspirations
We've built Scriptish for anyone who needs to replace UNIX shell scripts with compiled Golang binaries.
We're going to be doing that ourselves for some of our projects:
- Dockhand - Docker management utility
- HubFlow - the GitFlow extension for Git
- SimpleCA - local TLS certificate authority for internal infrastructure
We'll add links to those projects when they're available.
UNIX shell scripts are one of the most practical inventions in computer programming.
- They're very quick to write.
- They're very powerful.
- They treat everything as text.
They're fantastic for knocking up utilities in a matter of minutes, for automating things, and for gluing things together. Our hard drives are littered with little shell scripts - and some large ones too! - and we create new ones all the time.
If you're using any sort of UNIX system (Linux, or MacOS), shell scripting is a must-have skill - whether you're a developer or a sysadmin.
UNIX shell scripts are great until you want to share them with other people. They're just not a great choice if you want to distribute your work outside your team, organisation or community.
-
If someone else is going to run your shell scripts, they need to make sure that they've installed all the commands that your shell scripts call. This can end up being a trial-and-error process. And what happens if they can't install those commands for any reason?
-
Creating portable shell scripts (e.g. scripts that run on both Linux and MacOS) isn't always easy, and is very difficult (if not impossible) to test via a CI process.
-
What about your Windows users? UNIX shell scripts don't work on a vanilla Windows box.
If you want to distribute shell scripts, it's best not to write them as shell scripts. Use Scriptish to quickly do the same thing in Golang:
- There's one binary to ship to your users.
- Scriptish is self-contained. No need to worry about installing additional commands (unless you call scriptish.Exec() ...)
- Use Golang's
go test
to create tests for your tools. - Use the power of Golang to cross-compile binaries for Linux, MacOS and Windows.
Import Scriptish into your Golang code:
import scriptish "github.com/ganbarodigital/go_scriptish"
Create a pipeline, and provide it with a list of commands:
pipeline := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
)
Once you have your pipeline, run it:
pipeline.Exec()
Once you've run your pipeline, call one of the capture methods to find out what happened:
result, err := pipeline.ParseInt()
UNIX shell scripts compose UNIX commands into a pipeline:
cat /path/to/file.txt | wc -w
The UNIX commands execute from left to right. The output (known as stdout
) of each command becomes the input (known as stdin
) of the next command.
The output of the final command can be captured by your shell script to become the value of a variable:
current_branch=$(git branch --no-color | grep "^[*] " | sed -e 's/^[*] //')
Scriptish works the same way. You create a pipeline of Scriptish commands:
pipeline := scriptish.NewPipeline(
scriptish.Exec([]string{"git", "branch", "--no-color"}),
scriptish.Grep("^[* ]"),
scriptish.Tr([]string{"* "}, []string{""}),
)
and then you run it:
pipeline.Exec()
The output of the final command can be captured by your Golang code to become the value of a variable, using capture methods:
current_branch, err := pipeline.TrimmedString()
UNIX commands in a pipeline:
- read text input from
stdin
- write their results (as text!) to
stdout
- write any errors (as text!) out to
stderr
- return a status code to indicate what happened
Each Scriptish command works the same way:
- they read text input from the pipeline's
Stdin
- they write their results to the pipeline's
Stdout
- they write any error messages out to the pipeline's
Stderr
- they return a status code and a Golang error to indicate what happened
When a single command has finished, its Stdout
becomes the Stdin
for the next command in the pipeline.
One difference between UNIX commands and Golang is error handling. Scriptish combines the best of both.
- UNIX commands return a status code to indicate what happened. A status code of 0 (zero) means success.
- Scriptish commands return the UNIX-like status code, and any Golang error that has occurred. We store these in the pipeline.
If you're calling external commands using scriptish.Exec()
, you've still got access to the UNIX status code exactly like a shell script does. And you've always got access to any Golang errors that have occurred too.
Unlike UNIX shell scripts, a Scriptish pipeline stops executing if any command returns an error.
You might not be aware of it, but by default, a pipeline in a UNIX shell script continues to run even if one of the commands returns an error. This causes error values to propagate - and error propagation is a major cause of robustness issues in software.
Philosophically, we believe that good software engineering practices are more important than UNIX shell compatibility.
Scriptish commands fall into one of five categories:
- Sources create content in the pipeline, e.g.
scriptish.CatFile()
. They ignore whatever's already in the pipeline. - Filters do something with (or to) the pipeline's content, and they write the results back into the pipeline. These results form the input content for the next pipeline command.
- Sinks do something with (or two) the pipeline's content, and don't write any new content back into the pipeline.
- Logic implement support for
if
-like statements directly in Scriptish. - Redirects allow you to send output to somewhere else, such as a temporary file or /dev/null.
A pipeline normally:
- starts with a source
- applies one or more filters
- finishes with a sink to send the results somewhere
But what if we want to get the results back into our Golang code, to reuse in some way? Instead of using a sink, use a capture method instead.
A capture method isn't a Scriptish command. It's a method on the Pipeline
struct:
fileExists = scriptish.ExecPipeline(
scriptish.TestFilepathExists("/path/to/file.txt")
).Okay()
You can create a pipeline in several ways.
Pipeline | Produces | Best For |
---|---|---|
scriptish.NewPipeline() |
Pipeline that's ready to run | Reusable pipelines |
scriptish.NewPipelineFunc() |
Function that will run your pipeline | Getting results back into Golang |
scriptish.ExecPipeline() |
Pipeline that has been run once | Throwaway pipelines |
Call NewPipeline()
when you want to build a pipeline:
pipeline := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
)
pipeline
can now be executed as often as you want.
result, err := pipeline.Exec().ParseInt()
Most of the examples in this README (and most of the unit tests) use scriptish.NewPipeline()
.
NewPipelineFunc()
builds the pipeline and turns it into a function.
fileExistsFunc := scriptish.NewPipelineFunc(
scriptish.FileExists("/path/to/file")
)
Whenever you call the function, the pipeline executes. The function returns a *Pipeline
. Use any of the capture methods to find out what happened when the pipeline executed.
fileExists := fileExistsFunc().Okay()
You can re-use the function as often as you want.
NewPipelineFunc()
is great for pipelines where you want to get the results back into your Golang code:
getCurrentBranch := scriptish.NewPipelineFunc(
scriptish.Exec([]string{"git", "branch", "--no-color"}),
scriptish.Grep("^[* ]"),
scriptish.Tr([]string{"* "}, []string{""}),
)
currentBranch, err := getCurrentBranch().TrimmedString()
ExecPipeline()
builds a pipeline and executes it in a single step.
pipeline := scriptish.ExecPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
)
Behind the scenes, it simply does a scriptish.NewPipeline(...).Exec()
for you.
You can then use any of the capture methods to find out what happened:
result, err = pipeline.ParseInt()
You can re-use the resulting pipeline as often as you want.
ExecPipeline()
is great for pipelines that you want to throw away after use:
result, err := scriptish.ExecPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
).ParseInt()
Once you have built a pipeline, call the Exec()
method to execute it:
pipeline := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
)
pipeline.Exec()
Exec()
always returns a pointer to the same pipeline, so that you can use method chaining to create nicer-looking code.
// in this example, `pipeline` is available to be used more than once
pipeline := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
)
result, err := pipeline.Exec().ParseInt()
// in this example, we don't keep a reference to the pipeline
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
).Exec().ParseInt()
Both Pipeline.Exec()
and the function returned by NewPipelineFunc()
accept a list of parameters.
pipeline := scriptish.NewPipeline(
scriptish.CatFile("$1"),
scriptish.CountWords()
)
wordCount, _ := pipeline.Exec("/path/to/file").ParseInt()
fmt.Printf("file has %d words\n", wordCount)
countWordsInFile := scriptish.NewPipelineFunc(
scriptish.CatFile("$1"),
scriptish.CountWords()
)
wordCount, _ := countWordsInFile("/path/to/file").ParseInt()
fmt.Printf("file has %d words\n", wordCount)
The positional variables $1
, $2
, $3
et al are available inside the pipeline, just like they would be in a UNIX shell script function.
UNIX shell scripts can be broken up into functions to make them easier to maintain. You can do something similar in Scriptish, by calling a pipeline from another pipeline:
// this will parse the output of Git to find the selected branch
//
// the selected branch depends on the Git command called
filterSelectedBranch := scriptish.NewPipeline(
scriptish.Grep("^[*] "),
scriptish.Tr([]string{"* "}, []string{""}),
)
// which local branch are we working on?
localBranch, err := scriptish.NewPipeline(
scriptish.Exec([]string{"git branch --no-color"}),
scriptish.RunPipeline(filterSelectedBranch),
).Exec().TrimmedString()
// what's the tracking branch?
remoteBranch, err := scriptish.NewPipeline(
scriptish.Exec([]string{"git", "branch", "-av", "--no-color"}),
scriptish.RunPipeline(filterSelectedBranch),
).Exec().TrimmedString()
If you're familiar with UNIX shell scripting, you'll know that every shell command creates three different outputs:
stdout
- normal text outputstderr
- any error messages- status code - an integer representing what happened. 0 (zero) means success, any other value means an error occurred.
Scriptish commands work the same way. They also track any Golang errors that occur when the commands run.
Property | Description |
---|---|
pipeline.Stdout |
normal text output |
pipeline.Stderr |
an error messages (this is normally blank, because we have Golang errors too) |
pipeline.Err |
Golang errors |
pipeline.StatusCode |
an integer representing what happened. Normally 0 for success |
When the pipeline has executed, you can call one of the capture methods to find out what happened:
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.CountWords()
).Exec().ParseInt()
// if the pipeline worked ...
// - result now contains the number of words in the file
// - err is nil
//
// and if the pipeline didn't work ...
// - result is 0
// - err contains a Golang error
If you want to run a Scriptish command and you don't care about capturing the output, call Pipeline.Okay()
:
success := scriptish.NewPipeline(
scriptish.RmFile("/path/to/file.txt")
).Exec().Okay()
// if the pipeline worked ...
// - success is `true`
//
// and if the pipeline didn't work ...
// - success is `false`
UNIX shell scripts support two main ways (known as sequences) to string individual commands together:
- pipelines feed the output from one command into the next one
- lists simply append the output from each command to
stdout
andstderr
Most of the time, you'll want to stick to pipelines, and port the rest of your shell script's behaviour over to native Golang code. That gives you the convenience of Scriptish's emulation of classic UNIX shell commands and the power of everything that Golang can do.
Sometimes, you'll find it less effort to use a few lists too.
A classic example is die()
. It's very common for UNIX shell scripts to define their own die()
function like this:
die() {
echo "*** error: $*"
exit 1
}
[[ -e ./Dockerfile ]] || die "cannot find Dockerfile"
Here's the equivalent Scriptish:
dieFunc := scriptish.NewList(
scriptish.Echo("*** error: $*"),
scriptish.ToStderr(),
scriptish.Exit(1),
)
scriptish.ExecList(
scriptish.TestFileExists("./Dockerfile"),
scriptish.Or(dieFunc("cannot find Dockerfile")),
)
You can create a list in several ways.
Pipeline | Produces | Best For |
---|---|---|
scriptish.NewList() |
List that's ready to run | Reusable lists |
scriptish.NewListFunc() |
Function that will run your list | Getting results back into Golang |
scriptish.ExecList() |
List that has been run once | Throwaway lists |
Call NewList()
when you want to build a list:
list := scriptish.NewList(
scriptish.Echo("*** warning: $*"),
)
list
can now be executed as often as you want.
list.Exec("cannot find Dockerfile")
NewListFunc()
builds the list and turns it into a function.
fileExistsFunc := scriptish.NewListFunc(
scriptish.FileExists("/path/to/file")
)
Whenever you call the function, the list executes. The function returns a *List
. Use any of the capture methods to find out what happened when the list executed.
fileExists := fileExistsFunc().Okay()
You can re-use the function as often as you want.
NewListFunc()
is great for lists where you want to get the results back into your Golang code.
ExecList()
builds a list and executes it in a single step.
list := scriptish.ExecList(
scriptish.CatFile("/path/to/file1.txt"),
scriptish.CatFile("/path/to/file2.txt"),
)
Behind the scenes, it simply does a scriptish.NewList(...).Exec()
for you.
You can then use any of the capture methods to find out what happened:
result, err = list.ParseInt()
You can re-use the resulting list as often as you want.
ExecList()
is great for lists that you want to throw away after use.
Once you have built a list, call the Exec()
method to execute it:
list := scriptish.NewList(
scriptish.CatFile("/path/to/file1.txt"),
scriptish.CatFile("/path/to/file2.txt"),
)
list.Exec()
Exec()
always returns a pointer to the same list, so that you can use method chaining to create nicer-looking code.
// in this example, `list` is available to be used more than once
list := scriptish.NewList(
scriptish.CatFile("/path/to/file1.txt"),
scriptish.CatFile("/path/to/file2.txt"),
)
result, err := list.Exec().String()
// in this example, we don't keep a reference to the list
result, err := scriptish.NewList(
scriptish.CatFile("/path/to/file1.txt"),
scriptish.CatFile("/path/to/file2.txt"),
).Exec().String()
Both List.Exec()
and the function returned by NewListFunc()
accept a list of parameters.
list := scriptish.NewList(
scriptish.CatFile("$1"),
scriptish.CatFile("$2"),
)
fileContents := list.Exec("/path/to/file1.txt", "/path/to/file2.txt").String()
getTwoFileContents := scriptish.NewListFunc(
scriptish.CatFile("$1"),
scriptish.CatFile("$2"),
)
fileContents := getTwoFileContents("/path/to/file1.txt", "/path/to/file2.txt").String()
The positional variables $1
, $2
, $3
et al are available inside the list, just like they would be in a UNIX shell script function.
UNIX shell scripts can be broken up into functions to make them easier to maintain. You can do something similar in Scriptish, by calling a list from another pipeline or list:
// this will fetch the latest changes from the upstream Git repo
fetch_changes_from_origin := scriptish.NewList(
scriptish.Exec([]string{"git", "remote", "update", "$ORIGIN"}),
scriptish.Or(dieFunc("Unable to get list of changes from '$ORIGIN'")),
scriptish.Exec([]string{"git", "fetch", "$ORIGIN"}),
scriptish.Or(dieFunc("Unable to fetch latest changes from '$ORIGIN'")),
scriptish.Exec([]string{"git", "fetch", "--tags"}),
scriptish.Or(dieFunc("Unable to fetch latest tags from '$ORIGIN'")),
)
// this will fetch the latest changes from upstream, and then
// merge them into local branches
merge_latest_changes = scriptish.NewList(
scriptish.RunList(fetch_changes_from_origin),
...
)
In UNIX shell programming, pipelines and lists are both examples of a sequence of commands. Each one is a set of commands that are wrapped in slightly different execution logic.
In Scriptish, a Pipeline
and a List
are type aliases for a Sequence
. A call to NewPipeline()
or NewList()
creates a Sequence
that also has the right execution logic for a pipeline or a list. We've done it this way so that you can use both pipelines and lists in our logic calls.
All of our logic calls accept Sequence
parameters. You can pass in a Pipeline
or a List
to suit, and either will work just fine.
One of the things that makes UNIX shells so powerful is the way they expand a string, or a line of code, before executing it.
#!/usr/bin/env bash
PARAM1=hello world
# output: "Hello world"
echo ${PARAM1^H}
We've integrated the ShellExpand package so that string expansion is available to you.
The positional parameters are $1
, $2
, $3
all the way up to $9
, as well as $#
and $*
. These are exactly the same as their equivalents in shell scripts.
To set these, pass parameters into pipeline or into lists.
Every Pipeline
and List
struct comes with a LocalVars
member. You can call its Setenv()
method to create more variables to use in string expansion:
// create a reusable List
fetch_changes_from_remote := scriptish.NewList(
scriptish.Exec([]string{"git", "remote", "update", "$REMOTE"}),
scriptish.Or(dieFunc("Unable to get list of changes from '$REMOTE'")),
scriptish.Exec([]string{"git", "fetch", "$REMOTE"}),
scriptish.Or(dieFunc("Unable to fetch latest changes from '$REMOTE'")),
scriptish.Exec([]string{"git", "fetch", "--tags"}),
scriptish.Or(dieFunc("Unable to fetch latest tags from '$REMOTE'")),
)
// set the value of '$REMOTE'
fetch_changes_from_remote.LocalVars.Setenv("REMOTE", "origin")
// run it
fetch_changes_from_remote.Exec()
Any local variables that you set will remain set if you reuse the pipeline or list - ie, they are persistent.
The one downside of string expansion is that you will need to escape characters in your strings, to avoid them being interpreted as instructions to the string expansion engine.
The basic rule of thumb is that if you'd need to escape it in a shell script, you'll also need to escape it in a string passed into Scriptish.
At the moment, the string expansion does not support globbing (properly known as pathname expansion). That means you can't use wildcards in filepaths anywhere.
This is something we might add in a future release.
Here's a handy table to help you quickly translate an action from a Bash shell script to the equivalent Scriptish command.
Sources get data from outside the pipeline, and write it into the pipeline's Stdout
.
Every pipeline normally begins with a source, and is then followed by one or more filters.
Basename()
treats the input as a filepath. Any parent elements are stripped from the input, and the results written to the pipeline's Stdout
.
Any blank lines are preserved.
result, err := scriptish.NewPipeline(
scriptish.Basename("/path/to/folder/or/file"),
).Exec().TrimmedString()
Cat()
writes the remaining contents of the pipe's Stdin to the pipe's Stdout.
result, err := scriptish.NewPipeline(
// read from the program's os.Stdin
scriptish.Cat(AttachOsStdin()),
).Exec().String()
CatFile()
writes the contents of a file to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.CatFile("a/file.txt"),
).Exec().String()
CatOsStdin()
copies the program's stdin
(os.Stdin
in Golang terms) to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.CatOsStdin(),
).Exec().String()
Dirname()
treats the input as a filepath. It removes the last element from the input. It writes the result to the pipeline's Stdout
.
If the input is blank, Dirname() returns a '.'
result, err := scriptish.NewPipeline(
scriptish.Dirname("/path/to/folder/or/file")
).Exec().TrimmedString()
Echo()
writes a string to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.Echo("hello world"),
).Exec().String()
EchoArgs()
writes the program's arguments to the pipeline's Stdout
, one line per argument.
result, err := scriptish.NewPipeline(
scriptish.EchoArgs(),
).Exec().String()
EchoSlice()
writes an array of strings to the pipeline's Stdout
, one line per array entry.
myStrings := []string{"hello world", "have a nice day"}
result, err := scriptish.NewPipeline(
scriptish.EchoSlice(myStrings),
).Exec().String()
EchoToStderr()
writes a string to the pipeline's Stderr
.
result, err := scriptish.NewPipeline(
scriptish.EchoToStderr("*** error: file not found"),
).Exec()
Exec()
executes an operating system command. The command's stdin
will be the pipeline's Stdin
, and it will write to the pipeline's Stdout
and Stderr
.
The command's status code will be stored in the pipeline's StatusCode
.
localBranch, err := scriptish.ExecPipeline(
scriptish.Exec([]string{"git", "branch", "--no-color"}),
scriptish.Grep("^[* ]"),
scriptish.Tr([]string{"* "}, []string{""}),
).TrimmedString()
Use the .Okay()
capture method if you simply want to know if the command worked or not:
success := scriptish.ExecPipeline(scriptish.Exec([]string{"git", "push"})).Okay()
Golang will set err
to an exec.ExitError
if the command's status code is not 0 (zero).
Golang will set err
to an os.PathError
if the command could not be found in the first place.
ListFiles()
writes a list of matching files to the pipeline's Stdout
, one line per filename found.
- If
path
is a file, ListFiles writes the file to the pipeline'sStdout
- If
path
is a folder, ListFiles writes the contents of the folder to the pipeline'sStdout
. The path to the folder is included. - If
path
contains wildcards, ListFiles writes any files that matches to the pipeline'sStdout
.ListFiles()
uses Golang'sos.Glob()
to expand the wildcards.
// list a single file, if it exists
result, err := scriptish.NewPipeline(
scriptish.ListFiles("path/to/file"),
).Exec().String()
// list all files in a folder, if the folder exists
result, err := scriptish.NewPipeline(
scriptish.ListFiles("path/to/folder"),
).Exec().String()
// list all files in a folder that match wildcards, if the folder exists
result, err := scriptish.NewPipeline(
scriptish.ListFiles("path/to/folder/*.txt"),
).Exec().String()
Lsmod()
writes the permissions of the given filepath to the pipe's stdout.
- Symlinks are followed.
- Permissions are in the form '-rwxrwxrwx'.
It ignores the contents of the pipeline.
result, err := scriptish.NewPipeline(
scriptish.Lsmod("/path/to/file"),
).Exec().TrimmedString()
MkTempDir()
creates a temporary folder, and writes the filename to the pipeline's Stdout
.
tmpDir, err := scriptish.NewPipeline(
scriptish.MkTempFile(os.TempDir(), "scriptish-")
).Exec().TrimmedString()
MkTempFile()
creates a temporary file, and writes the filename to the pipeline's Stdout
.
tmpFilename, err := scriptish.NewPipeline(
scriptish.MkTempFile(os.TempDir(), "scriptish-*")
).Exec().TrimmedString()
MkTempFilename()
generates a temporary filename, and writes the filename to the pipeline's Stdout
.
tmpFilename, err := scriptish.NewPipeline(
scriptish.MkTempFilename(os.TempDir(), "scriptish-*")
).Exec().TrimmedString()
Which()
searches the current PATH to find the given path. If one is found, the command's path is written to the pipeline's Stdout
.
It ignores the contents of the pipeline.
result, err := scriptish.NewPipeline(
scriptish.Which("git"),
).Exec().String()
Filters read the contents of the pipeline's Stdin
, do something to that data, and write the results out to the pipeline's Stdout
.
When you've finished adding filters to your pipeline, you should either add a sink, or call one of the capture methods to get the results back into your Golang code.
AppendToTempFile()
writes the contents of the pipeline's Stdin
to a
temporary file. The temporary file's filename is then written to
the pipeline's Stdout
.
If the file does not exist, it is created.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.AppendToTempFile(os.TempDir(), "scriptish-*"),
).Exec().TrimmedString()
// result now contains the temporary filename
CountLines()
counts the number of lines in the pipeline's Stdin
, and writes that to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("path/to/folder/*.txt"),
scriptish.CountLines(),
).Exec().ParseInt()
CountWords()
counts the number of words in the pipeline's Stdin
, and writes that to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.CatFile("path/to/file.txt"),
scriptish.CountWords(),
).Exec().ParseInt()
CutFields()
retrieves only the fields specified on each line of the pipeline's Stdin
, and writes them to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.Echo("one two three four five"),
scriptish.CutFields("2-3,5")
).Exec().String()
DropEmptyLines()
removes any lines that are blank, or that only contain whitespace.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.DropEmptyLines()
).Exec().String()
Grep()
filters out lines that do not match the given regex.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Grep("second|third"),
).Exec().String()
GrepV()
filters out lines that match the given regex.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.GrepV("second|third"),
).Exec().String()
Head()
copies the first N lines of the pipeline's Stdin
to its Stdout
.
If N is zero or negative, Head()
copies no lines.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Head(100),
).Exec().String()
Rsort()
sorts the contents of the pipeline into descending alphabetical order.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Rsort(),
).Exec().String()
RunPipeline()
allows you to call one pipeline from another. Use it to create reusable pipelines, a bit like shell script functions.
getWordCount := scriptish.NewPipeline(
scriptish.SplitWords(),
scriptish.CountLines(),
)
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.RunPipeline(getWordCount),
).Exec().ParseInt()
Sort()
sorts the contents of the pipeline into ascending alphabetical order.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Sort(),
).Exec().String()
StripExtension()
treats every line in the pipeline as a filepath. It removes the extension from each filepath.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder"),
scriptish.StripExtension(),
).Exec().Strings()
SwapExtensions()
treats every line in the pipeline as a filepath.
It replaces every old extension with the corresponding new one.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder"),
scriptish.SwapExtensions([]string{"txt","yml"}, []string{"text","yaml"}),
).Exec().Strings()
If the second parameter is a string slice of length 1, every old file extension will be replaced by that parameter.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder"),
scriptish.SwapExtensions([]string{"yml","YAML"}, []string{"yaml"}),
).Exec().Strings()
If the first and second parameters are different lengths, SwapExtensions()
will return an scriptish.ErrMismatchedInputs
.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder"),
scriptish.SwapExtensions([]string{"one"}, []string{"1","2"}),
).Exec().Strings()
// err is an ErrMismatchedInputs, and result is empty
Tail()
copies the last N lines from the pipeline's Stdin
to its Stdout
.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Tail(50),
).Exec().String()
Tr()
replaces all occurances of one string with another.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Tr([]string{"one","two"}, []string{"1","2"}),
).Exec().String()
If the second parameter is a string slice of length 1, everything from the first parameter will be replaced by that slice.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Tr([]string{"one","two"}, []string{"numberwang"}),
).Exec().String()
If the first and second parameters are different lengths, Tr()
will return an scriptish.ErrMismatchedInputs
.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Tr([]string{"one","two","three"}, []string{"1","2"}),
).Exec().String()
// err is an ErrMismatchedInputs, and result is empty
TrimSuffix()
removes the given suffix from each line of the pipeline.
Use it to emulate basename(1)
's [suffix]
parameter.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder/"),
scriptish.TrimSuffix(".txt")
).Exec().Strings()
TrimWhitespace()
removes any whitespace from the front and end of each line in the pipeline.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.TrimWhitespace(),
).Exec().String()
Uniq()
removes duplicated lines from the pipeline.
result, err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.Uniq(),
).Exec().Strings()
XargsBasename()
treats each line in the pipeline's Stdin
as a filepath. Any parent elements are stripped from the line, and the results written to the pipeline's Stdout
.
Any blank lines are preserved.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder/*.txt"),
scriptish.XargsBasename()
).Exec().Strings()
XargsCat()
treats each line in the pipeline's Stdin
as a filepath. The contents of each file are written to the pipeline's Stdout
.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder/*.txt"),
scriptish.XargsCat()
).Exec().String()
XargsDirname()
treats each line in the pipeline's Stdin
as a filepath. The last element is stripped from the line, and the results written to the pipeline's Stdout
.
Any blank lines are turned in '.'
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder/*.txt"),
scriptish.XargsDirname()
).Exec().Strings()
XargsRmFile()
treats every line in the pipeline as a filename. It attempts to delete each file.
It stops at the first file that cannot be deleted.
Each successfully-deleted filepath is written to the pipeline's Stdout
, for use by the next command in the pipeline.
err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/files"),
scriptish.XargsRmFile(),
scriptish.ForXIn(
scriptish.Printf("deleted file: $x")
)
).Exec().Error()
XargsTestFilepathExists()
treats each line in the pipeline as a filepath. It checks to see if the given filepath exists. If the filepath exists, it is written to the pipeline's stdout.
It does not care what the filepath points at (file, folder, named pipe, and so on).
// example: find all RAW photo files that also have a corresponding
// JPEG file
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/folder/*.raw"),
scriptish.SwapExtensions(".raw", ".jpeg"),
scriptish.XargsTestFilepathExists()
).Exec().Strings()
XargsTruncatesFiles()
treats each line of the pipeline's Stdin
as a filepath. The contents of each file are truncated. If the file does not exist, it is created.
Each filepath is written to the pipeline's Stdout
, for use by the next command in the pipeline.
result, err := scriptish.NewPipeline(
scriptish.ListFiles("/path/to/files"),
scriptish.XargsTruncateFiles(),
).Exec().Strings()
// result now contains a list of the files that have been truncated
Sinks take the contents of the pipeline's Stdin
, and write it to somewhere outside the pipeline.
A sink should be the last command in your pipeline. You can add more commands afterwards if you really want to. Just be aware that the first command after any sink will be starting with an empty Stdin
.
AppendToFile()
writes the contents of the pipeline's Stdin
to the given file
If the file does not exist, it is created.
err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.AppendToFile("my-app.log"),
).Exec().Error()
Exit()
terminates your Golang program with the given status code. Use with caution.
dieFunc := scriptish.NewList(
scriptish.Echo("*** error: $*"),
scriptish.ToStderr(),
scriptish.Exit(1),
)
scriptish.ExecList(
scriptish.TestFilepathExists("./Dockerfile"),
scriptish.Or(dieFunc("cannot find Dockerfile")),
)
Return()
terminates the pipeline with the given status code.
statusCode := scriptish.NewPipeline(
scriptish.Return(3)
).Exec().StatusCode()
// statusCode will be: 3
ToStdout()
writes the pipeline's Stdin
to the program's stderr
(os.Stderr
in Golang terms).
err := scriptish.NewPipeline(
scriptish.Echo("usage: simpleca <command>"),
scriptish.ToStderr(),
).Exec().Error()
ToStdout()
writes the pipeline's Stdin
to the program's Stdout
(os.Stdout
in Golang terms).
err := scriptish.NewPipeline(
scriptish.Echo("usage: simpleca <command>"),
scriptish.ToStdout(),
).Exec().Error()
WriteToFile()
writes the contents of the pipeline's Stdin
to the given file. The existing contents of the file are replaced.
If the file does not exist, it is created.
err := scriptish.NewPipeline(
scriptish.CatFile("/path/to/file.txt"),
scriptish.WriteToFile("/path/to/new_file.txt"),
).Exec().Error()
Many UNIX shell scripts send the output of individual commands (or even whole pipelines) to somewhere else. We've created equivalent functionality for Scriptish users:
Shell Redirect | Scriptish Equiv | Description |
---|---|---|
>&2 |
RedirectStdoutToStderr | Anything written to stdout goes to stderr instead. |
2>&1 |
RedirectStderrToStdout | Anything written to stderr goes to stdout instead. |
2>/dev/null |
RedirectStderrToDevNull | Anything written to stderr is thrown away. |
> /dev/null |
RedirectStdoutToDevNull | Anything written to stdout is thrown away. |
> <filename> |
OverwriteFilenameWithStdout | Anything written to stdout is written to the given file instead, replacing the file's existing contents. |
2> <filename> |
OverwriteFilenameWithStderr | Anything written to stderr is written to the given file instead, replacing the file's existing contents. |
>> <filename> |
AppendStdoutToFilename | Anything written to stdout is appended to the given file instead. |
2>> <filename> |
AppendStderrToFilename | Anything written to stderr is appended to the given file instead. |
n/a | RedirectStdoutToTextReaderWriter | Anything written to pipe.Stdout is written to the given Golang file instead. |
n/a | RedirectStderrToTextReaderWriter | Anything written to pipe.Stderr is written to the given Golang file instead. |
n/a | AttachOsStdin | Read from the program's os.Stdin . |
To use a redirect, simply pass it as a parameter to any of the sources or filters. For example:
pipeline := scriptish.NewPipeline(
// write text to the pipeline's stdout
// but redirect the pipeline's stdout to /dev/null
scriptish.Echo("this is a test", scriptish.RedirectStdoutToDevNull()),
)
pipeline.Exec()
// output will be empty
output := pipeline.String()
You can pass more than one redirect in, too:
pipeline := NewPipeline(
// write text to the pipeline's stderr
// but redirect the pipeline's stderr to the pipeline's stdout
// and redirect the pipeline's stdout to /dev/null
scriptish.EchoToStderr(
"this is a test",
scriptish.RedirectStdoutToDevNull(),
scriptish.RedirectStderrToStdout(),
)
)
pipeline.Exec()
// output will be empty
output := pipeline.String()
AppendStdoutToFilename()
redirects the pipe's stdout to the given filename.
Anything written to the pipe's stdout is appended to the named file. If the file does not exist, it is created.
It is an emulation of UNIX shell scripting's >> <filename>
.
ExecPipeline(
scriptish.Exec(
[]string{"go", "test"},
scriptish.AppendStdoutToFilename("test.out"),
)
)
AppendStderrToFilename()
redirects the pipe's Stderr to the given filename.
Anything written to the pipe's Stderr is appended to the named file. If the file does not exist, it is created.
It is an emulation of UNIX shell scripting's 2>> <filename>
.
pipeline := scriptish.NewPipeline(
scriptish.Exec(
[]string{"go", "test"},
scriptish.AppendStderrToFilename("test.err"),
)
).Exec()
AttachOsStdin()
sets the pipe's Stdin to read from the program's os.Stdin.
pipeline := scriptish.NewPipeline(
scriptish.Cat(scriptish.AttachOsStdin())
)
input, err := pipeline.Exec().String()
OverwriteFilenameWithStdout()
redirects the pipe's Stdout to the given filename.
If the file does not exist, it is created. If the file does exist, its contents are replaced.
It is an emulation of UNIX shell scripting's > <filename>
.
copyFile := scriptish.NewPipeline(
scriptish.CatFile("$1", scriptish.OverwriteFilenameWithStdout("$2")),
)
copyFile.Exec("some-input-file", "some-output-file")
OverwriteFilenameWithStderr()
redirects the pipe's Stderr to the given filename.
If the file does not exist, it is created. If the file does exist, its contents are replaced.
It is an emulation of UNIX shell scripting's 2> <filename>
.
captureTests := scriptish.NewPipeline(
scriptish.Exec(
[]string{"npm", "test"},
scriptish.OverwriteFilenameWithStderr("$1")
),
)
captureTests.Exec("test.out")
RedirectStderrToStdout()
makes all output to the pipe's Stderr go to the pipe's Stdout instead.
It is an emulation of UNIX shell scripting's 2>&1
.
pipeline := scriptish.NewPipeline(
scriptish.Exec(
[]string{"npm", "test"},
scriptish.RedirectStderrToStdout(),
)
).Exec()
RedirectStderrToDevNull()
replaces the pipe's Stderr with an ioextra.TextDevNull
.
It is an emulation of UNIX shell scripting's 2> /dev/null
.
It's must useful when running external commands that produce output that you're just not interested in. If you redirect the pipeline's stderr using RedirectStderrToDevNull()
, you avoid Golang having to allocate any memory to store that output.
pipeline := scriptish.NewPipeline(
// run an external command, and throw away the output you do not need
scriptish.Exec([]string{"some-noisy-command"}, scriptish.RedirectStderrToDevNull())
).Exec()
RedirectStderrToTextReaderWriter makes all output to the pipe's Stderr go to the given ioextra.TextReaderWriter
instead.
It gives you a Golang-native way to redirect the output to wherever you need it to.
There's no direct equivalent in UNIX shell script programming.
func captureTestErrors(dest *os.File) (string, error) {
pipeline := scriptish.NewPipeline(
scriptish.Exec(
[]string{"npm", "test"},
RedirectStderrToTextReaderWriter(NewTextFile(dest)),
),
)
return pipeline.Exec().String()
}
RedirectStdoutToDevNull()
replaces the pipe's Stdout with an ioextra.TextDevNull before the command runs.
It is an emulation of UNIX shell scripting's > /dev/null
.
It's most useful when running external commands that produce a lot of output that you're just not interested in. You can avoid Golang having to allocate any memory to store that output. The Exec() command will make sure that the external command's writes to stdout are just thrown away.
pipeline := scriptish.NewPipeline(
// write text to the pipeline's stdout
// but redirect the pipeline's stdout to /dev/null
scriptish.Echo("this is a test", scriptish.RedirectStdoutToDevNull()),
)
pipeline.Exec()
// output will be empty
output := pipeline.String()
RedirectStdoutToStderr()
makes all output to the pipe's Stdout go to the pipe's Stderr instead.
It is an emulation of UNIX shell scripting's >&2
. It's mostly been added for completeness.
pipeline := scriptish.NewPipeline(
// write text to the pipeline's Stdout
// but redirect Stdout to Stderr
scriptish.Echo("this is a test", scriptish.RedirectStdoutToStderr()),
)
pipeline.Exec()
// stdout will be empty
stdout := pipeline.Stdout.String()
// stderr will contain "this is a test"
stderr := pipeline.Stderr.String()
RedirectStdoutToTextReaderWriter()
makes all output to the pipe's Stdout go to the given ioextra.TextReaderWriter
instead.
It gives you a Golang-native way to redirect the output to wherever you need it to.
There's no direct equivalent in UNIX shell script programming.
func copyToFile(filename string, dest *os.File) (int, err) {
return ExecPipeline(
scriptish.CatFile(
filename,
RedirectStdoutToTextReaderWriter(ioextra.NewTextFile(dest))
)
).StatusError()
}
Builtins are UNIX shell commands and UNIX CLI utilities that don't fall into the sources, sinks and filters categories:
- their input is a parameter; they ignore the pipeline
- their only output is the status code; they don't write anything new to the pipeline
Chmod()
attempts to change the permissions on the given filepath.
It ignores the contents of the pipeline.
On success, it returns the status code StatusOkay
. On failure, it returns the status code StatusNotOkay
.
result, err := scriptish.NewPipeline(
scriptish.Chmod("/path/to/file", 0644),
).Exec().StatusError()
Mkdir()
creates the named directory, along with any parent folders that are needed.
It ignores the contents of the pipeline.
On success, it returns the status code StatusOkay
. On failure, it returns the status code StatusNotOkay
.
result, err := scriptish.NewPipeline(
scriptish.Mkdir("/path/to/folder", 0755),
).Exec().StatusError()
RmDir()
deletes the given folder, as long as the folder is empty.
It ignores the contents of the pipeline.
It ignores the file's file permissions, because the underlying Golang os.Remove() behaves that way.
err := scriptish.NewPipeline(
scriptish.RmDir("/path/to/folder"),
).Exec().Error()
RmFile()
deletes the given file.
It ignores the contents of the pipeline.
It ignores the file's file permissions, because the underlying Golang os.Remove() behaves that way.
err := scriptish.NewPipeline(
scriptish.RmFile("/path/to/file"),
).Exec().Error()
TestEmpty()
returns StatusOkay
if the (expanded) input is empty; StatusNotOkay
otherwise.
It is the equivalent to if [[ -z $VAR ]]
in a UNIX shell script.
show_usage() {
echo "*** error: $*"
echo
echo "usage: $0 <name-of-arg>"
exit 1
}
if [[ -z $1 ]] ; then
show_usage("missing parameter <name-of-arg>")
fi
Here's the equivalent Scriptish:
showUsage := scriptish.NewList(
scriptish.Echo("*** error: $*"),
scriptish.Echo(""),
scriptish.Echo("usage: $0 <name-of-arg>")
scriptish.Exit(1),
)
checkArgs := scriptish.NewList(
scriptish.If(
scriptish.NewPipeline(scriptish.TestEmpty("$1")),
showUsage("missing argument")
),
)
checkArgs.Exec(os.Args...)
TestFilepathExists()
checks to see if the given filepath exists. If it does, it returns StatusOkay
. If not, it returns StatusNotOkay
.
- It does not care what the filepath points at (file, folder, named pipe, and so on).
- It ignores the contents of the pipeline.
- It follows symbolic links.
fileExists := scriptish.ExecPipeline(
scriptish.TestFilepathExists("/path/to/file")
).Okay()
TestNotEmpty()
returns StatusOkay
if the (expanded) input is not empty; StatusNotOkay
otherwise.
It is the equivalent to if [[ -n $VAR ]]
in a UNIX shell script.
show_usage() {
echo "*** error: $*"
echo
echo "usage: $0 <name-of-arg>"
exit 1
}
[[ -n $1 ]] || show_usage("missing parameter <name-of-arg>")
Here's the equivalent Scriptish:
showUsage := scriptish.NewList(
scriptish.Echo("*** error: $*"),
scriptish.Echo(""),
scriptish.Echo("usage: $0 <name-of-arg>")
scriptish.Exit(1),
)
checkArgs := scriptish.NewList(
scriptish.TestNotEmpty("$1"),
scriptish.Or(showUsage("missing argument")),
)
checkArgs.Exec(os.Args...)
Touch()
creates the named file (if it doesn't exist), or updates its atime and mtime (if it does exist).
It ignores the contents of the pipeline.
On success, it returns the status code StatusOkay
. On failure, it returns the status code StatusNotOkay
.
pipeline := NewPipeline(
Touch("./config.yaml")
)
success, err := pipeline.Exec().StatusError()
TruncateFile()
removes the contents of the given file.
If the file does not exist, it is created.
err := scriptish.NewPipeline(
scriptish.TruncateFile("/tmp/scriptish-test"),
).Exec().Error()
Capture methods available on each Pipeline
. Use them to get the output from the pipeline.
Don't forget to run your pipeline first, if you haven't already!
Bytes()
is the standard interface's io.Reader
Bytes()
method.
- It writes the contents of the pipeline's
Stdout
into the byte slice that you provide. - It returns the number of bytes written.
- It also returns the pipeline's current Golang error value.
Normally, you wouldn't call this yourself.
Error()
returns the pipeline's current Golang error status, which may be nil
.
err := scriptish.ExecPipeline(scriptish.RmFile("/path/to/file")).Error()
Flush()
writes the contents of the pipeline or list out to the given destinations.
Use os.Stdout
and os.Stderr
to send the output to your program's terminal.
pipeline := scriptish.NewPipeline(
scriptish.Echo("usage: simpleca <command>"),
)
pipeline.Exec().Flush(os.Stdout, os.Stderr)
Okay()
returns true|false
depending on the pipeline's current UNIX status code.
success := scriptish.ExecPipeline(scriptish.Exec([]string{"git", "push"})).Okay()
success
is a bool
:
false
if the pipeline'sStatusCode
property is not 0true
otherwise
All Scriptish commands set the pipeline's StatusCode
, so it's safe to use Okay()
to check any pipeline you create.
It's mostly there if you want to call a pipeline in a Golang if
statement:
if !scriptish.ExecPipeline(scriptish.Exec([]string{"git", "push"})).Okay() {
// push failed, do something about it
}
ParseInt()
returns the pipeline's Stdout
as an int
value:
lineCount, err := scriptish.ExecPipeline(
scriptish.CatFile("/path/to/file"),
scriptish.CountLines(),
).ParseInt()
If the pipeline's Stdout
can't be turned into an integer, then it will return 0
and the parsing error from Golang's strconv.ParseInt()
.
If the pipeline didn't execute successfully, it will return 0
and the pipeline's current Golang error status.
String()
returns the pipeline's Stdout
as a single string:
contents, err := scriptish.ExecPipeline(
scriptish.CatFile("/path/to/file")
).String()
The string will be terminated by a linefeed \n
character. String()
is a good choice if you want to get content into your Golang code. If you just want a single-line value, see TrimmedString() below.
If the pipeline's Stdout
is empty, an empty string will be returned.
If the pipeline didn't execute successfully, the contents of the pipeline's Stderr
will be returned. We might change this behaviour in the future.
Strings()
returns the pipeline's Stdout
as an array of strings (aka a string slice):
files, err := scriptish.ExecPipeline(
scriptish.ListFiles("/path/to/folder"),
scriptish.XargsBasename(),
).Strings()
Each string will not be terminated by a linefeed \n
character.
If the pipeline's Stdout
is empty, an empty string slice will be returned.
If the pipeline didn't execute successfully, the contents of the pipeline's Stderr
will be returned. We might change this behaviour in the future.
TrimmedString()
returns the pipeline's Stdout
as a single string, with leading and trailing whitespace removed:
localBranch, err := scriptish.ExecPipeline(
scriptish.Exec([]string{"git", "branch", "--no-color"}),
scriptish.Grep("^[* ]"),
scriptish.Tr([]string{"* "}, []string{""}),
).TrimmedString()
TrimmedString()
is the right choice if you're expecting a single line of text back. This is very useful for getting results back into your Golang code!
If the pipeline's Stdout
is empty, an empty string will be returned.
If the pipeline didn't execute successfully, the contents of the pipeline's Stderr
will be returned. We might change this behaviour in the future.
Most of the time, you will use native Golang code to write if
statements for your code. Use the capture methods to get the results of your pipelines back into your Golang code.
Sometimes, it will be more convenient to use Scriptish's built-in logic support. The classic example is the die()
function commonly created in UNIX shell scripts to handle errors:
die() {
echo "*** error: $*" >&2
exit 1
}
[[ -e ./Dockerfile ]] || die "cannot find Dockerfile"
Here's the equivalent Scriptish:
dieFunc := scriptish.NewList(
scriptish.Echo("*** error: $*"),
scriptish.ToStderr(),
scriptish.Exit(1),
)
scriptish.ExecList(
scriptish.TestFilepathExists("./Dockerfile"),
scriptish.Or(dieFunc("cannot find Dockerfile")),
)
(It also has to be said that implementing logic support in Scriptish was a good test case for proving Scriptish's underlying design.)
Generally, all the implemented logc takes Lists or Pipelines as arguments. (Lists and Pipelines are both aliases for the Sequence
struct, so you can pass either in to suit.) If there are any exceptions to this rule, we'll make sure to point it out in the documentation for the individual logic call.
And()
executes the given sequence only if the previous command did not return any kind of error.
The sequence starts with an empty Stdin. The sequence's output is written back to the Stdout and Stderr of the calling list or pipeline - along with the StatusCode() and Error().
It is an emulation of UNIX shell scripting's command1 && command2
feature.
NOTE that And()
only works when run inside a List:
scriptish.ExecList(
scriptish.Exec([]string{"git", "fetch"}),
// if `git fetch` fails, do not attempt the merge
scriptish.And(
scriptish.NewList(
scriptish.Exec([]string{"git", "merge"})
)
)
)
If you call And()
inside a Pipeline, it'll always run the given sequence. Pipelines terminate whenever a command returns an error, so And()
will only be called if the previous command succeeded.
If()
executes the body
if (and only if) the given expr
does not return any kind of error.
Both expr
and body
start with an empty Stdin
. Their output is written back to the pipeline's Stdout
and Stderr
.
It is an emulation of UNIX shell scripting's if expr ; then body ; fi
feature.
result, err := scriptish.ExecList(
scriptish.If(
// this is the `expr` or expression
scriptish.NewPipeline(
scriptish.TestFilepathExists("/path/to/file"),
),
// this is the `body` that is executed if the `expr` succeeds
scriptish.NewPipeline(
scriptish.CatFile("/path/to/file"),
scriptish.Head(3),
),
)
).String()
You can safely use If()
inside a pipeline, because it doesn't depend upon the result of any previous command.
IfElse()
executes the body if (and only if) the expr completes without an error. Otherwise, it executes the elseBlock instead.
IfElse()
is an emulation of UNIX shell scripting's if expr ; then body ; else elseBlock ; fi
.
result, err := scriptish.ExecList(
scriptish.IfElse(
// this is the `expr` or expression
scriptish.NewPipeline(
scriptish.TestFilepathExists("/path/to/file"),
),
// this is the `body` that is executed if the `expr` succeeds
scriptish.NewPipeline(
scriptish.CatFile("/path/to/file"),
scriptish.Head(3),
),
// and this is the `elseBlock` that is executed if the `expr` fails
scriptish.NewPipeline(
scriptish.Echo("*** error: file not found"),
scriptish.ToStderr(),
)
)
).String()
Or()
executes the given sequence only if the previous command has returned some kind of error.
The sequence starts with an empty Stdin. The sequence's output is written back to the Stdout and Stderr of the calling list or pipeline - along with the StatusCode() and Error().
It is an emulation of UNIX shell scripting's list1 || command
feature.
NOTE that Or()
only works when run inside a List:
statusCode, err := scriptish.NewList(
scriptish.TestFilepathExists("/path/to/file"),
scriptish.Or(dieFunc("cannot find file")),
).Exec().StatusError()
If you call Or()
from inside a Pipeline, it will never work. Pipelines terminate when the first command returns an error. This means that either:
- the pipeline will terminate before
Or()
is reached, or Or()
never executes the given sequence (because there's no previous error)
At the moment, we can't think of a way of detecting any attempt to call Or()
from a pipeline.
ErrMismatchedInputs
is returned whenever two input arrays aren't the same length.
Scriptish is inspired by:
Pipe is a bit more low level, and seems to be aimed predominantly at executing external commands, as a more powerful alternative to Golang's exec
package.
If that's what you need, definitely check it out!
We started out using (and contributing to) Script, but ran into a few things that didn't suit what we were doing:
-
Built for different purposes
script is aimed at doing minimal shell-like operations from a Golang app.
scriptish is more about providing everything necessary to recreate any size UNIX shell script - including powerful ones like the HubFlow extension for Git - to Golang, without having to port everything to Golang.
-
Aimed at different people
We want it to take as little thinking as possible to port UNIX shell scripts over to Golang - especially for casual or lapsed Golang programmers!
That means (amongst other things) using function names that are similar to the UNIX shell command that they emulate, and emulating UNIX behaviour as closely as is practical, including UNIX features like status codes and
stderr
. -
Extensibility
script operations are methods on the
script.Pipe
struct. We found that this makes it very hard to extendscript
with your own methods, because Golang doesn't support inheritance, and the method chaining prevents Golang embedding from working.In contrast, script commands are first-order functions that take the Pipe as a function parameter. You can create your own Scriptish commands, and they can live in your own Golang package.
-
Reusability
There's currently no way to call one pipeline from another using script alone. You can achieve that by writing your own Golang boiler plate code.
scriptish builds first-order pipelines that you can run, pass around as values, and call from other scriptish pipelines.
We were originally attracted to script because of how incredibly easy it is to use. There's a lot to like about it, and we've definitely tried to create the same feel in scriptish too. We've borrowed concepts such as sources, filters and sinks from script, because they're such a great way to describe how different Scriptish commands behave.
You should definitely check script out if you think that scriptish is too much for what you need.