Skip to main content

Parser and transpiler for the Kahmi DSL.

Project description

kahmi-dsl

This is a Python-based configuration language for the Kahmi build system that is heavily inspired by Groovy and Gradle.

Example:

buildscript {
  dependencies = ["kahmi-git", "kahmi-cxx"]
}

let cxx = load("kahmi-cxx")
let git = load("kahmi-git")

name = "myproject"
version = git.version()

cxx.build("main") {
  srcs = glob("src/*.cpp")
  type = "executable"
}

Syntax & Semantics

The Kahmi DSL basically wraps Python code and even extends it's normal syntax with multiline lambdas.

Every Kahmi script is executed in a given "context", that can be an arbitrary Python object which is addressable in Python expressions as "self". There are 3 different types of operations one can perform. Entering a Python expression or multiline lambda reverts the parser back into full Python mode (with the aforementioned multiline lambda support).

  1. Define a local variable with the let Keyword

    Local variables are defined using the let keyword. The variable can then be addressed in Python expressions or as call block targets (see below). The right hand side of the assignment must be a Python expression.

    let my_variable = 42
    
  2. Set a propery on the current context object

    The same syntax but without the let keyword assigns the value to a member of the current context object instead of to a local variable.

    nmae = "my-project"
    version = git.version()
    
  3. Call blocks

    A call block is used to invoke a method or function and optionally enter a new scope with the return value as the new context object. The first name of the call block target is resolved in the local variables, then in the members of the current context object and the parent context objects and finally in the Python built-ins.

    print("Hello, World!")  # Call without body
    
    buildscript {  # Call with body and without arguments
      dependencies = ["kahmi-python"]
    }
    
    cxx.build("main") {  # Call with body and arguments, addressing a member of the `cxx` target.
      srcs = glob("src/*.cpp")
    }
    

    If the new context object that is returned by the called function supports the Python context manager interface, it will be used around the execution of the call block's body.

  4. Multi-line lambdas

    The Kahmi DSL parser injects the ability to define multi-line lambdas in any Python expression. The lambda syntax is inspired by Javascript/Typescript and uses => as the lambda arrow operation to connect the argument definition with the lambda body.

    A lambda with braces requires a return statement, otherwhise the return value of the lambda will be None. Single-statement lambdas are not currently supported with this syntax (although you can always fall back to standard syntax lambda: <expr>).

    let myFunc = () => {
      import random
      return random.random()
    }
    
    print(myFunc())
    

    Nesting lambdas is supported and has the expected semantics except if used in comprehensions (as they introduce a new scope that can not be captured by the function definition that is a multi-line lambda is transpiled to).

Built-ins

Kahmi only provides two additional built-in functions on top of what is provided by Python, and they are only necessary for the execution of Kahmi's generated Python code.

Name Description
self The root context object for the script.
__lookup__(name, locals_, ctx) Helper function to resolve the targets of call blocks.

Under the hood

Kahmi comes with a simple cli that allows you to run any Kahmi script, but given the limited ability to override the root context object it is expected that it does not serve much use outside of debugging and development.

$ python -m kahmi.dsl examples/hello.kmi

Using the -E option, you can retrieve the Python code that a Kahmi file is transpiled to. This is especially useful to understand how Kahmi constructs are converted into Python. Below are some examples:

let msg = (name) => {
  return 'Hello, ' + name
}('World')

print(msg)
def lambda_stdin_1_10(name):
    return 'Hello, ' + name


msg = lambda_stdin_1_10('World')
__lookup__('print', locals(), self)(msg)

buildscript {
  dependencies = ["kahmi-python"]
}
def __call_buildscript(self):
    self.dependencies = ['kahmi-python']


__call_buildscript_self_arg = __lookup__('buildscript', locals(), self)()
if hasattr(__call_buildscript_self_arg, '__enter__'):
    with __call_buildscript_self_arg:
        __call_buildscript(__call_buildscript_self_arg)
else:
    __call_buildscript(__call_buildscript_self_arg)

Copyright © 2021 Niklas Rosenstein

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

kahmi-dsl-0.0.2.tar.gz (13.4 kB view details)

Uploaded Source

Built Distribution

kahmi_dsl-0.0.2-py3-none-any.whl (12.9 kB view details)

Uploaded Python 3

File details

Details for the file kahmi-dsl-0.0.2.tar.gz.

File metadata

  • Download URL: kahmi-dsl-0.0.2.tar.gz
  • Upload date:
  • Size: 13.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.2.0 pkginfo/1.6.1 requests/2.25.1 setuptools/50.3.2 requests-toolbelt/0.9.1 tqdm/4.54.1 CPython/3.7.3

File hashes

Hashes for kahmi-dsl-0.0.2.tar.gz
Algorithm Hash digest
SHA256 6ed45e937ce7d95ce80f33742f94df3d62cb5e7dee0563bd843babfe52a839cf
MD5 6fce6529d5cdf4a966f747bfedd68bdb
BLAKE2b-256 008663be9d24afdeb1e0bfcbea680eb15bb9281a12f688a60eddad8d921486c8

See more details on using hashes here.

File details

Details for the file kahmi_dsl-0.0.2-py3-none-any.whl.

File metadata

  • Download URL: kahmi_dsl-0.0.2-py3-none-any.whl
  • Upload date:
  • Size: 12.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/3.2.0 pkginfo/1.6.1 requests/2.25.1 setuptools/50.3.2 requests-toolbelt/0.9.1 tqdm/4.54.1 CPython/3.7.3

File hashes

Hashes for kahmi_dsl-0.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 1d33ad0301663357339355a3caabb62d211085aa4be712cb8c76ccc546969b62
MD5 c402c5088c230037e3191033b7828aab
BLAKE2b-256 dd2663e06bd1affa975a0c29ed9c1871f7f096f734cb37a7bb0c116d409c3d93

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page