Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/xyproto/rangetype
:1234: Mini-language/DSL for defining and dealing with ranges of numbers
https://github.com/xyproto/rangetype
ada dsl foreach go mini-language ranges
Last synced: 14 days ago
JSON representation
:1234: Mini-language/DSL for defining and dealing with ranges of numbers
- Host: GitHub
- URL: https://github.com/xyproto/rangetype
- Owner: xyproto
- License: mit
- Created: 2017-11-27T14:32:19.000Z (about 7 years ago)
- Default Branch: master
- Last Pushed: 2018-09-17T16:09:20.000Z (over 6 years ago)
- Last Synced: 2024-12-07T01:16:27.755Z (about 2 months ago)
- Topics: ada, dsl, foreach, go, mini-language, ranges
- Language: Go
- Homepage:
- Size: 703 KB
- Stars: 4
- Watchers: 3
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Rangetype
[![Build Status](https://travis-ci.org/xyproto/rangetype.svg?branch=master)](https://travis-ci.org/xyproto/rangetype) [![GoDoc](https://godoc.org/github.com/xyproto/rangetype?status.svg)](http://godoc.org/github.com/xyproto/rangetype) [![License](http://img.shields.io/badge/license-MIT-red.svg?style=flat)](https://raw.githubusercontent.com/xyproto/rangetype/master/LICENSE) [![Report Card](https://img.shields.io/badge/go_report-A+-brightgreen.svg?style=flat)](http://goreportcard.com/report/xyproto/rangetype)
A mini-language for defining numeric types by defining ranges.
The idea is to provide a DSL for defining and validating numeric types for implementations of programming languages.
It can also be used for iterating over ranges, generating lists of numbers or slicing a given slice (like slices in Python).
## ForEach
Looping over a range can be done by providing a function that takes a `float64`:
```go
r.New("1..10").ForEach(func(x float64) {
fmt.Println(int(x))
})
```## Join
Collecting integers to a comma separated string can be done with `Join`:
```go
r.New("1..10").Join(", ", 0)
```Or for floats, with 2 digits after the period, separated by semicolons:
```go
r.New("1..3 step 0.5").Join(";", 2)
```## Syntax
Expressions can optionally start with:
* `[` for including the first value in the range, or
* `(` for excluding the first value in the rangeAnd can end with:
* `]` for including the last value in the range, or
* `)` for excluding the last value in the rangeNumbers can be suffixed by:
* `~` for subtracting 1 from the preceding number. Any number of `~` is possible.
The ranges can be Python-style:
`[0:10]`
Python-style with a step:
`[1:20:-1]`
Ruby-style:
`1..10`
Ruby-style with a step:
`1..10 step 2`
Math-style:
`[1,5)`
Math-style with a step:
`[1,5) step 2`
Math-style with negative steps:
`(5,1] step -0.1`
Here's an expression for specifying the range for a 16-bit unsigned integer:
`0..65535`
This can also be defined by dropping the initial `0` and using `~` for `-1`:
`..65536~`
This can be made clearer (when defining many numbers of many bit-sizes) by using `2**16` instead of `65536`:
`..2**16~`
This can be used for validating if a given number fits a 16-bit unsigned type:
```go
IntType := r.New("..2**16~") // from 0 up to and including 65536-1
IntType.Valid(42) // true
```## Examples
An int with a range from 1 to 3 that includes both 1, 2 and 3:
`1,3`
A float with a range from 1 to 3 that includes 1.0, 1.1 etc up to 3.0:
`1,3 step 0.1`
Inclusivity is specified with square brackets:
`[1,3]`
Exclusivity is specified with parenthesis:
`(1,3)`
Ranges inspired by Ruby also work:
`1..3`
These are inclusive, unless parenthesis are used.
Ruby-style range which will exclude `1` and `3` and only keep `2`:
`(1..3)`
Python style ranges are also supported, where the start value is inclusive and the end value is exclusive:
`1:3`
Adding square brackets makes the range inclusive:
`[1:3]`
Brackets and parenthesis does not have to be balanced. This works too:
`1:3]`
Adding an iteration step is also possible:
`1..5 step 2`
This is a range with the numbers `1`, `3` and `5`.
The Python-style syntax also supports steps:
`[3:1:-1]`
This is `3`, `2`, `1`.
Steps does not have to be integers:
`[3:1:-0.1]`
This steps from 3 (inclusive) down to 1 (inclusive) in step sizes of 0.1.
## More Examples
### Defining a SmallInt type and checking if a given number is valid
```go
package mainimport (
"fmt"r "github.com/xyproto/rangetype"
)func main() {
// Define a new type that can hold numbers from 0 up to and including 99
SmallInt := r.New("0..99")// Another way to define a number type from 0 up to and excluding 100
//SmallInt := r.New("[0,100)")// Another way to define a number type from 0 up to and excluding 100
//SmallInt := r.New("..10**2~")// Is 42 a valid SmallInt?
fmt.Println("0 is a valid SmallInt value:", SmallInt.Valid(0))
fmt.Println("2 is a valid SmallInt value:", SmallInt.Valid(2))
fmt.Println("-1 is a valid SmallInt value:", SmallInt.Valid(-1))
fmt.Println("99 is a valid SmallInt value:", SmallInt.Valid(99))
fmt.Println("100 is a valid SmallInt value:", SmallInt.Valid(100))// How many integers are there room for?
fmt.Printf("SmallInt can hold %d different numbers.\n", SmallInt.Len())
fmt.Printf("Storage required for SmallInt: a %d-bit int\n", SmallInt.Bits())// All possible SmallInt values, comma separated:
fmt.Println("All possible values for SmallInt:\n" + SmallInt.Join(",", 0))
}
```## Example 2
### Slicing slices and looping with the ForEach method
```go
package mainimport (
"fmt"r "github.com/xyproto/rangetype"
)func main() {
// Outputs 1 to 10, with 0 digits after "."
fmt.Println(r.New("1..10").Join(", ", 0))// Outputs 2 and 4
for _, x := range r.Slice([]float64{1.0, 2.0, 3.0, 4.0}, "1..3 step 2") {
fmt.Print(x, " ")
}
fmt.Println()// Also outputs 2 and 4
r.New("(0:6:2)").ForEach(func(x float64) {
fmt.Print(x, " ")
})
fmt.Println()
}
```There are more examples in the `range_test.go` file.
## Features and Limitations
* Can handle very large ranges without storing the actual numbers in the ranges, but iterating over large ranges may be slow.
* Only `**` and `~` are supported for manipulating numbers in the range expressions.
* It's not a general language, it's only a DSL for expressing ranges of integers or floating point numbers, with an optional step size.## Error Handling
* `New2` and `Slice2` will return both a value and an error (if the expression failed to evaluate) and are the recommended functions to use.
* `New` and `Slice` are fine to use for expressions that are already known to evaluate, but may panic if there are errors in the given expression.## General Info
* License: MIT
* Version: 0.1
* Author: Alexander F Rødseth <[email protected]>