Skip to content
/ ella Public

a compiler to generate golang and typescript http client and server

License

Notifications You must be signed in to change notification settings

ella-to/ella

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

18 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

███████╗██╗░░░░░██╗░░░░░░█████╗░
██╔════╝██║░░░░░██║░░░░░██╔══██╗
█████╗░░██║░░░░░██║░░░░░███████║
██╔══╝░░██║░░░░░██║░░░░░██╔══██║
███████╗███████╗███████╗██║░░██║
╚══════╝╚══════╝╚══════╝╚═╝░░╚═╝ v0.0.7

Ella, is yet another compiler to produce Go and Typescript code based on simple and easy-to-read schema IDL. There are many tools like gRPC, Twirp or event WebRPC to generate codes, but this little compiler is designed based on my views of 10+ years of developing backend and APIs. I wanted to simplify the tooling and produce almost perfect optimized, handcrafted code that can be read and understood, not like the output of gRPC.

Ella's schema went through several iterations to make it easier for extension and backward compatibility in future releases.

NOTE:

Ella's code generated has been used in couple of production projects, it has some designs that might not fit your needs, but I think it might solve a large number of projects. Also Ella's only emit Go, as a server and client, and Typescript as only client. This is intentinal as it servers my needs. However it can be extetended to produce other languages code by traversing the AST code.

Installation

to install Ella's compiler, simply use the go install command

go install ella.to/ella@latest

Usage

Simplicity applies to the CLI command as well, it looks for all files that need to be compiled and outputs the result to the designated file. The extension of the output file tells the compiler whether you want to produce the typescript or golang code. That's pretty much of it.

For example, the following command, will generate api.gen.go in /api folder with the package name api and will read all the ella files inside ./schema folder.

ella gen api /api/api.gen.go ./schema/*.ella

Also, we can format the schema as well to have a consistent look by running the following command

ella fmt ./schema/*.ella

The full CLI documentation can be accessed by running Ella command without any arguments

███████╗██╗░░░░░██╗░░░░░░█████╗░
██╔════╝██║░░░░░██║░░░░░██╔══██╗
█████╗░░██║░░░░░██║░░░░░███████║
██╔══╝░░██║░░░░░██║░░░░░██╔══██║
███████╗███████╗███████╗██║░░██║
╚══════╝╚══════╝╚══════╝╚═╝░░╚═╝ v0.0.7

Usage: ella [command]

Commands:
  - fmt Format one or many files in place using glob pattern
        ella fmt <glob path>

  - gen Generate code from a folder to a file and currently
        supports .go and .ts extensions
        ella gen <pkg> <output path to file> <search glob paths...>

  - ver Print the version of ella

example:
  ella fmt ./path/to/*.ella
  ella gen rpc ./path/to/output.go ./path/to/*.ella
  ella gen rpc ./path/to/output.ts ./path/to/*.ella ./path/to/other/*.ella

Schema

There is a simple role when writing a schema in Ella's IDL, all identifier needs to be Pascal Case. This eliminates colliding with reserved keywords.

const

A constant is a value that is immutable and can be used in field and method options.

int

const BuildNumber = 10

float

const Pi = 3.14

string

const DoubleQuote = "1.0.0"

const SingleQuote = 'Hello World'

const MultiLine = `Hello
  This is really cool

  bye
`

bool

const Debug = true

byte size

This is a very helpful time as it makes it easier to write values in bytes type. The following postfixes can be used: b, kb, mb, gb, tb, pb, eb

Note: the number has to be integer. No floating is premitted. In order to represent 1.1kb, use the lower postfix b and represent the number as follows: 1100b.

const MaxFileUploadSize = 100mb

duration

Note: duration type is also the same as byte size type. Only interger value is permitted.

This is a very helpful time as it makes it easier to write values in duration type. The following postfixes can be used: ns, us, ms, s, m, h

const MaxWaitTime = 5h

enum

enum is a way to define a series of const values under the same category. In Golang, there is no such thing as enum and usually, people use a custom type and assign values to it. Ella's compiler does the heavy lifting of that and generates the most optimized version of the Go representation that supports both yaml and json marshal and unmarshal operations. It also supports ignoring value using _ keyword.

Note: Ella's enum type doesn't have a type, because behind the scene it will generate approporate type which is most memory efficient and optimize.

enum UserType {
  _
  Normal
  Guest
  Root
}

enum UserStatus {
  _
  Active = 10
  Deactive
  Deleted = 65
}

model

model is a way to define a series of variables under the same category, similar to struct in Go.

model User {
  Firstname: string
  Lastname: string
  Age: int8
  LocationMap: []string
  Parents: []User
  ComplexMap: map<string, []User>
  CreatedAt: timestamp
}

Note: Model's field type can be any of the default types such as byte, bool, int8, int16, int32, int64, uint8, uint16, uint32, uint64, float32, float64, timestamp, string, and complex types such as map<key, value> and array []type, or it can be any other model's or enum's name type.

field options

field options is a way to customize and assign values to each field of the model. Currently, there are the following predefined field options available.

  • JsonOmitEmpty

is a boolean value that adds omitempty inside Go generated struct tag.

model User {
  Username: string
  Password: string {
    JsonOmitEmpty = true
  }
}

So basically the above model generates a Go struct as follows:

type User struct {
  Username string `json:"username"`
  Password string `json:"password,omitempty"`
}
  • Json

Json option fields are a way to either not marshal and unmarshal the value or renaming the field during marshal and unmarshal operations.

model User {
  Firstname: string {
    Json = "firstName"
  }
}

model AnotherUser {
  Username: string
  Password: string {
    Json = false
  }
}

The above model will be converted to the following Go code

type User struct {
  Firstname string `json:"firstName"`
}

type AnotherUser struct {
  Username string `json:"username"`
  Password string `json:"-"`
}
  • Yaml

is the same as Json. Currently, this is the only option available for Yaml.

model User {
  Firstname: string {
    Yaml = "firstName"
  }`
}

model User {`
  Username: string
  Password: string {
    Yaml = false
  }
}

service

http

stream

file upload

rpc

method options

error

Defining custom errors

error ErrUserNotFound { Code = 1000 HttpStatus = NotFound Msg = "user not found" }

http status can be one of the following values

  • Continue: 100
  • SwitchingProtocols: 101
  • Processing: 102
  • EarlyHints: 103
  • OK: 200
  • Created: 201
  • Accepted: 202
  • NonAuthoritativeInfo: 203
  • NoContent: 204
  • ResetContent: 205
  • PartialContent: 206
  • MultiStatus: 207
  • AlreadyReported: 208
  • IMUsed: 226
  • MultipleChoices: 300
  • MovedPermanently: 301
  • Found: 302
  • SeeOther: 303
  • NotModified: 304
  • UseProxy: 305
  • TemporaryRedirect: 307
  • PermanentRedirect: 308
  • BadRequest: 400
  • Unauthorized: 401
  • PaymentRequired: 402
  • Forbidden: 403
  • NotFound: 404
  • MethodNotAllowed: 405
  • NotAcceptable: 406
  • ProxyAuthRequired: 407
  • RequestTimeout: 408
  • Conflict: 409
  • Gone: 410
  • LengthRequired: 411
  • PreconditionFailed: 412
  • RequestEntityTooLarge: 413
  • RequestURITooLong: 414
  • UnsupportedMediaType: 415
  • RequestedRangeNotSatisfiable: 416
  • ExpectationFailed: 417
  • Teapot: 418
  • MisdirectedRequest: 421
  • UnprocessableEntity: 422
  • Locked: 423
  • FailedDependency: 424
  • TooEarly: 425
  • UpgradeRequired: 426
  • PreconditionRequired: 428
  • TooManyRequests: 429
  • RequestHeaderFieldsTooLarge: 431
  • UnavailableForLegalReasons: 451
  • InternalServerError: 500
  • NotImplemented: 501
  • BadGateway: 502
  • ServiceUnavailable: 503
  • GatewayTimeout: 504
  • HTTPVersionNotSupported: 505
  • VariantAlsoNegotiates: 506
  • InsufficientStorage: 507
  • LoopDetected: 508
  • NotExtended: 510
  • NetworkAuthenticationRequired: 511

References

  • Here is the list of reserved keywords:

    • const
    • enum
    • model
    • http
    • rpc
    • service
    • byte
    • bool
    • int8
    • int16
    • int32
    • int64
    • uint8
    • uint16
    • uint32
    • uint64
    • float32
    • float64
    • timestamp
    • string
    • map
    • any
    • file
    • stream
    • error
  • The logo was generated here


About

a compiler to generate golang and typescript http client and server

Resources

License

Stars

Watchers

Forks

Packages

No packages published