Ghee: a command line tool for working with Linux extended attributes (xattrs)

The tastiest way to work with Linux extended attributes (xattrs) written in pure Rust and made of delicious, fibrous Open Source code.

Ghee provides tools for manipulating xattrs on individual files as well as for working with the filesystem as a document database where the filesystem paths act as primary keys and extended attributes act as fields.

History

Ghee was developed in conjunction with the Audiotater audio annotation tool.

License

This software is licensed under GPL version 3 only.

Conventions

Extended attribute names are parsed in a consistent manner by Ghee. Any xattr not preceded by the trusted, security, system, or user namespace will have the user namespace by default. For example, xattr trusted.uptime remains as is, while uptime would become user.uptime.

Extended attribute values are parsed as f64 numbers if possible; otherwise, they are interpeted as strings.

REPL

Running ghee with no arguments will enter a read-eval-print-loop (REPL), allowing for fluent command input:

``` $ ghee Ghee 0.5.1

ghee$ set ./test -s test=1 ``` etc.

Subcommands

Ghee operates through a set of subcommands, each with a primary function. Run ghee --help to list them, and ghee $SUBCMD --help to get usage information for each subcommand.

Examples of each subcommand follow:

Move

Moves xattr values from one path to another.

Copy

Copies xattr values from one path to another.

Remove

Removes xattr values, recursively by default.

Set

Sets xattr values, recursively by default.

Get

Recursively get and print xattr values for one or more paths.

By default, the get subcommand outputs a tab-separated table with a column order of path, field, value. The value bytes are written to stdout as-is without decoding.

This excludes the user.ghee prefix unless -a --all is passed.

To opt out of the recursive default, use --flat.

The get command can also output JSON - in which case values are decoded as UTF-8, filling in a default codepoint when decoding fails:

By adding --where (or -w), SQL WHERE-style clauses can be provided to select which files to include in the output. For example, ghee get -w age >= 65 ./patients will select all files under directory ./patients whose user.age attribute is 65 or greater.

Nested indices are always ignored in get output, though they will be used as appropriate to shortcut traversal when WHERE-style predicates are specified.

Init

Initializes a directory as a table with a specified primary key, optionally inserting records from JSON where each line is parsed independently---see people.json in the repository for an example.

Examples: * ghee init -k name ./people: marks the ./people directory as a table with primary key of name * ghee init -k state -k id ./people-by-state-and-id: marks the ./people-by-state-and-id directory as a table with a compound primary key of [state, id]. * ghee init -k sauce ./pizza < ./pizzas.json: marks the ./pizza directory as a table with primary key sauce, importing records from ./pizzas.json

Create

Exactly like init, but creates the directory first, or errors if it already exists.

Insert

Inserts JSON-formatted records into a table.

Records are read one per line from stdin.

Delete

Deletes records from a table.

They are unlinked from all table indices.

The records to be deleted are specified by providing either the components of the primary key or SQL-style WHERE clauses.

Index

Indexes a table.

When Ghee acts on a directory as if it were a database table, each file acts as a relational "record" with the primary key inferred from its subpath under the table directory.

Each file's extended attributes act as the relational attributes.

Table directories created by Ghee also contain a special xattr user.ghee.tableinfo which stores the primary key and related indices (including itself) of a table.

If no index location is provided, it will be placed in a default path under the table being indexed.

Examples:

List

Like the ls command, lists directory contents, but annotated from Ghee's point of view.

Each path is marked as either a table or a record. For tables, the primary key is given.