Skip to content

Latest commit

 

History

History
230 lines (177 loc) · 8.56 KB

README.md

File metadata and controls

230 lines (177 loc) · 8.56 KB

edict: Eric's Dictionary Transformer

A simple interpreted language for specifying transformations to dictionaries (also called records or key-value-stores).

An edict program describes a series of rules and assignments that are applied to a set of records on a record-by-record basis. All variables refer to keys in the current record. Variables values are always stored as strings but certain operations (like math) interpret the variable values as other types. Uninitialized variables are empty strings.

The syntax of edict is loosely based on Python and Bash. Whitespace is ignored.

This is in alpha stage. Changes may be backwards incompatible without warning.

Example

The following is a simple script for categorizing transactions.

# This is a comment
@case_insensitive              # ignore case when matching
@default_field("description")  # use the key "description" for implicit matches

# Use the 'read_date' builtin function to convert DD/MM/YYYY to the standard
# YYYY-MM-DD. The format codes are those used by Python's strptime format
# https://docs.python.org/3/library/datetime.html#strftime-strptime-behavior
date = read_date(date, "%d/%m/%Y")

# Check if "moe's tavern" is a substring of the "description" variable
if "moe's tavern" then
    account = "Expenses:Drinks"

# Check if amount is greater than 0.
# Numerical comparison because one side is a number.
elif amount > 0 then
    account = "Income:Uncategorized"
else
    account = "Expenses:Uncategorized"
fi

See the .edt files in tests/files for more examples.

Install

pip install .

Specification

Variables

Variables in Edict name fields (keys) in the current record. A variable is identified with either an unquoted sequence of word characters (letters, digits, and underscores) that does not start with a digit (e.g. SomeVar_42) or an arbitrary string quoted with braces { and } (e.g. {Some Var 42}) where \ escapes the following character.

Variables do not need to be initialized or defined before use. All variables are stored as strings and uninitialized variables are empty strings.

Literals

Literals are used for expressing a value in an expression. There are four literal types: String, Boolean, Number, and Regular Expression.

  • String: A quoted UTF-8 string: "some string"
  • Boolean: true or false
  • Number: An high-precision decimal number with exact arithmetic. Not floating-point. Uses the Python Decimal type.
  • Regular Expression: A Python regular expression used only in match statements. Quoted with slashes (/): /some regex *\d+/.

Type System

Expressions and literals are strongly typed in Edict. The possible types are STRING, NUMBER, BOOLEAN, REGEX, or NONE for expressions with no value. Implicit casting is never performed with these types and it is an error to use a type where it is not expected.

Variables in Edict are weakly typed. All variables are stored as strings and have the type INDETERMINANT_STRING, which is implicitly cast to STRING or NUMBER depending on context. For example, var == "0.0" interprets var as a string while var == 0.0 interprets var as a number. The inferred type of a variable does not persist between uses in a program, even within the same statement, so the expression var == 0 or var == "1" is legal.

When the context is ambiguous, like in var1 == var2, variables are interpreted as STRING. The as_number function can be used to explicitly cast variables as NUMBER.

Operators

The following operators are available:

  • Comparison: ==, !=, <, <=, >, >=. Compare numbers or strings. The types of both sides must be the same.

  • Match: string ~ pattern. Check if pattern matches a substring of string. If pattern is a string then uses string equalty. If pattern is a regular expression then performs regular expression matching (use ^ and $ to match the entire string rather than a substring). In either case, the matching is case sensitive unless the @case_insensitive directive has been defined.

  • Implicit Match: pattern. A pattern on its own expands to default_field ~ pattern, where default_field is the variable set by the @default_field directive.

  • Math: * (multiplication), / (division), % (mod), + (addition), - (subtraction), - (unary minus). Both operands must be numbers.

  • String: . (string concatenation). Both operands must be strings.

  • Assignment: identifier = value. Assign value to identifier. The value is stored as a string.

Builtin Functions

The available functions are:

  • as_number(x) -> NUMBER Interpret the value x as a number. This is useful when comparing two variables to force numeric comparisons. For example, a == b does string comparison while as_number(a) == b does numeric comparison.

  • input_protocol() -> STRING The name of the input protocol in the current program execution.

  • log(arg1, arg2, ...) Log all arguments to standard error when executed. Takes any number of arguments.

  • output_protocol() -> STRING The name of the output protocol in the current program execution.

  • read_date(date_string: STRING, format_string: STRING) -> STRING Read a date a format as an ISO 8601 string. The format string is the same used by Python strptime

  • record_str() -> STRING Format the current record as a string. Used for debugging with log(record_str()).

  • substring(string: STRING, start: NUMBER[, end: NUMBER]) -> STRING The sub-string of string from start (inclusive) to end (exclusive). Negative indices count from the end. The last character has index -1. Equivalent to the Python expression string[start:end].

  • replace(string: STRING, old: STRING, new: STRING[, count: NUMBER]) -> STRING Replace up to count occurrences of old with new in string. Replaces all by default. Equivalent to python str.replace.

Directives

Directives are special function calls that can be evaluated before running. They may affect the structure of the edict program.

Header Directives

Header directives appear at the top of the file and establish some context for the program.

  • @case_insensitive Ignore case for string matching and comparisons.

  • @default_field(field) Use the field field with implicit match statements.

  • @output_fields(field1, field2, ...) Restrict the output fields. Only the matching field names will be included in the output (for outputs with configurable output fields). Currently, only applies to CSV output.

  • @reverse Reverse the order of the input records.

Inline Directives

Inline directives can go anywhere a regular statement goes.

  • @@import("path/to/script.edt") Import another edict script and apply it at this location. Either an absolute or a relative path.

Protocols

Protocols define input and output formats for the data stream. The default protocol for both reading and writing is csv.

The available protocols are:

  • csv - Comma-separated values. The first line must be a header containing column names. Individual rows may have fewer columns than the header but not more.

  • pattern - Format according to a pattern string (write only). The pattern is specified with the --pattern argument and is formatted according to Python's str.format given the record dictionary as arguments.

  • hledger - The hledger journal format (write only). Looks for specific named fields to construct the output transactions. The field names similar to those used by the hledger CSV conversion; see the implementation in hledger.py for details.

  • beancount - The beancount journal format (write only). Currently supports transaction, open, close, and price directives. See the implementation in beancount.py for details.

Development

Editable Install

python setup.py develop [--user]

Re-run this command to refresh the version number (based on git tags).

Versioning

Uses Semantic Versioning.

Versions are set exclusively via git tags:

git -a v0.1.2 -m "Version 0.1.2"