Skip to content

Commit

Permalink
Updated the documentation file to explain how to use Expressive in C#
Browse files Browse the repository at this point in the history
  • Loading branch information
bijington committed Jun 20, 2016
1 parent 609ccbb commit 7ce42b4
Showing 1 changed file with 175 additions and 43 deletions.
218 changes: 175 additions & 43 deletions Documentation/Expressive.md
Original file line number Diff line number Diff line change
@@ -1,12 +1,143 @@
# Expressive expression parser
# Expressive

## Supported languages
Expressive is a multi-platform expression parsing and evaluation framework.

C#
Swift
Java
## Design and Implementation
There are 3 main stages of the parsing/evaluation of an Expression.

## Operators
1. Tokenising - this deals with walking through the characters in the expression and matching the items against the known set of `Operators`, `Functions` and `Values`.
2. Compiling - once the `Expression` has been tokenised the set of tokens is then used to build a tree of internal `IExpression` nodes.
3. Evaluation - each individual `IExpression` node is then evaluated to determine it's result and eventually the overall result of the top level `Expression`.

### Lazy Parameter evaluation
Each individual `parameter` will only be evaluated at the point of being used within its `IExpression` node. This will prevent parts of an expression being evaluated should the result already be determined (e.g. when the left hand side of an && (logical AND) operator being `false`).

### Error handling

The `Expression` will terminate early in the following situations:
1. Any part of the tokenising process fails (i.e. unrecognised tokens).
2. If any of the `Function`s are supplied with an incorrect number of parameters.

The supported languages are:

- C#
- Java (TBA)
- Swift (TBA)

## Usage Examples

### C#

#### Simple Evaluation
```c
var expression = new Expression("1+2");
var result = expression.Evaluate();
```

#### Custom options
```c
public enum ExpressiveOptions
{
/// <summary>
/// Specifies that no options are set.
/// </summary>
None = 1,
/// <summary>
/// Specifies case-insensitive matching.
/// </summary>
IgnoreCase = 2,
/// <summary>
/// No-cache mode. Ignores any pre-compiled expression in the cache.
/// </summary>
NoCache = 4,
/// <summary>
/// When using Round(), if a number is halfway between two others, it is rounded toward the nearest number that is away from zero.
/// </summary>
RoundAwayFromZero = 8,
/// <summary>
/// All options are used.
/// </summary>
All = IgnoreCase | NoCache | RoundAwayFromZero,
}

// Use in any combination.
var expression = new Expression("1+2", ExpressiveOptions.IgnoreCase | ExpressiveOptions.NoCache);
var result = expression.Evaluate();
```

#### Variables
```c#
var expression = new Expression("1 * [variable]");
var result = expression.Evaluate(new Dictionary<string, object> { ["variable"] = 2);
```

#### Built-in Functions
Expressive provides a standard set of functions (see Functions).
```c#
var expression = new Expression("sum(1,2,3,4)");
var result = expression.Evaluate();
```

#### Custom Functions
Custom functions allow you to register your own set of funcitonality in order to extend the current set. There are 2 methods of registering a custom function:

* `IFunction` implementation.

```c#
internal class AbsFunction : IFunction
{
#region IFunction Members

public IDictionary<string, object> Variables { get; set; }

public string Name { get { return "Asin"; } }

public object Evaluate(IExpression[] parameters)
{
return Math.Asin(Convert.ToDouble(parameters[0].Evaluate(Variables)));
}

#endregion
}

var expression = new Expression("Abs(1)");
expression.RegisterFunction(new AbsFunction());
var result = expression.Evaluate();
```

* Lambda callback.

```c#
var expression = new Expression("myfunc(1)");
expression.RegisterFunction("myfunc", (p, v) =>
{
// p = parameters
// v = variables
return p[0].Evaluate(v);
});
var result = expression.Evaluate();
```

#### Retrieve referenced variables
Compiles the expression and returns an array of the variable names contained within the expression.
```c#
var expression = new Expression("[abc] + [def] * [ghi]");
var variables = expression.ReferencedVariables;
```

#### Asynchronous evaluation
Useful for when an evaluation might take a long time (i.e. a function that deals with a long running background tasks such as loading from the database).
```c
var expression = new Expression("1+2");
expression.EvaluateAsync((r) =>
{
var result = r;
});
```

## General functionality

### Operators

| Name | Operator | Usage |
| ---------------------- | ----------------- |-------------------------|
Expand All @@ -32,50 +163,51 @@ Java
| Left shift | << | 2 << 2 |
| Right shift | >> | 2 >> 2 |

## Functions

| Function | Usage | Description |
| ------------------ | ----------------------------- | :----------------------------:|
| Abs | Abs(-1) | |
| Acos | Acos(1) | |
| Asin | Asin(0) | |
| Atan | Atan(0) | |
| Avg | Avg(1,2,3,4) | |
| Ceiling | Ceiling(1.5) | |
| Cos | Cos(0) | |
| EndsWith | EndsWith('end') | |
| Exp | Exp(0) | |
| Floor | Floor(1.5) | |
| IEEERemainder | IEEERemainder(3, 2) | |
| If | If(1 == 1, 'true', 'false') | |
| In | In(1, 1, 2, 3, 4) | |
| Log | Log(1, 10) | |
| Log10 | Log10(1) | |
| Max | Max(1, 2) | |
| Min | Min(1, 2) | |
| Pow | Pow(3, 2) | |
| Random | Random() | |
| Round | Round(4.5234, 2) | |
| Sign | Sign(-10) | |
| Sin | Sin(0) | |
| Sqrt | Sqrt(4) | |
| StartsWith | StartsWith('start') | |
| Sum | Sum(0, 2, 3, 4) | |
| Tan | Tan(0) | |
| Truncate | Truncate(1.7) | |
### Functions

| Function | Usage | Parameters |
| ------------------ | ----------------------------- | ------------------------------ |
| Abs | Abs(-1) | Expects **exactly 1** parameter. |
| Acos | Acos(1) | Expects **exactly 1** parameter. |
| Asin | Asin(0) | Expects **exactly 1** parameter. |
| Atan | Atan(0) | Expects **exactly 1** parameter. |
| Average | Average(1,2,3,4) | Expects **at least 1** parameter. |
| Ceiling | Ceiling(1.5) | Expects **exactly 1** parameter. |
| Cos | Cos(0) | Expects **exactly 1** parameter. |
| EndsWith | EndsWith('end') | Expects **exactly 1** parameter. |
| Exp | Exp(0) | Expects **exactly 1** parameter. |
| Floor | Floor(1.5) | Expects **exactly 1** parameter. |
| IEEERemainder | IEEERemainder(3, 2) | Expects **exactly 2** parameters. |
| If | If(1 == 1, 'true', 'false') | Expects **exactly 3** parameters. |
| In | In(1, 1, 2, 3, 4) | Expects **at least 2** parameters. |
| Log | Log(1, 10) | Expects **exactly 2** parameters. |
| Log10 | Log10(1) | Expects **exactly 1** parameter. |
| Max | Max(1, 2) | Expects **at least 1** parameter. |
| Mean | Mean(1, 2) | Expects **exactly 2** parameters. |
| Median | Median(1, 2) | Expects **exactly 2** parameters. |
| Min | Min(1, 2) | Expects **at least 1** parameter. |
| Mode | Mode(1, 2) | Expects **exactly 2** parameters. |
| PadLeft | PadLeft([number], 3, '0') | Expects **exactly 3** parameters. |
| Pow | Pow(3, 2) | Expects **exactly 2** parameters. |
| Random | Random() | Expects **no** parameters. |
| Round | Round(4.5234, 2) | Expects **exactly 2** parameters. |
| Sign | Sign(-10) | Expects **exactly 1** parameter. |
| Sin | Sin(0) | Expects **exactly 1** parameter. |
| Sqrt | Sqrt(4) | Expects **exactly 1** parameter. |
| StartsWith | StartsWith('start') | Expects **exactly 1** parameter. |
| Sum | Sum(0, 2, 3, 4) | Expects **at least 1** parameter. |
| Tan | Tan(0) | Expects **exactly 1** parameter. |
| Truncate | Truncate(1.7) | Expects **exactly 1** parameter. |

### Values

| Name | Usage |
| ------------------- | ----------------------- |
| Variable | [parametername] |
| String | "abc", 'abc' |
| DateTime | #2015/12/25# |
| DateTime | #2015/12/25#, #Today#, #Now# |
| Integer | 123 |
| Boolean | true, false |
| Floating point | 123.456 |
| Scientific notation | 1.22e2 |
| Boolean | true, false |


# Testing Plan

| null | null |

0 comments on commit 7ce42b4

Please sign in to comment.