Metadata-Version: 2.1
Name: viper-lang
Version: 0.0.2
Summary: a simple, easy to understand language with easy integration capabilities.
Home-page: https://github.com/IAmTomahawkx/viper-lang
Author: IAmTomahawkx
Author-email: iamtomahawkx@gmail.com
License: MIT
Project-URL: Issue tracker, https://github.com/IAmTomahawkx/viper-lang/issues
Platform: UNKNOWN
Classifier: Development Status :: 4 - Beta
Classifier: License :: OSI Approved :: MIT License
Classifier: Intended Audience :: Developers
Classifier: Natural Language :: English
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3.6
Classifier: Programming Language :: Python :: 3.7
Classifier: Programming Language :: Python :: 3.8
Classifier: Topic :: Internet
Classifier: Topic :: Software Development :: Libraries
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: Topic :: Utilities
Requires-Python: >=3.6.0
Description-Content-Type: text/x-rst

.. image:: https://travis-ci.org/IAmTomahawkx/viper-lang.svg?branch=master
    :target: https://travis-ci.org/IAmTomahawkx/viper-lang
    :alt: Build Status
.. image:: https://discord.com/api/guilds/561043858402836482/embed.png
   :target: https://discord.gg/cEAxG8A
   :alt: Discord server invite
.. image:: https://img.shields.io/pypi/v/viper-lang.svg
   :target: https://pypi.python.org/pypi/viper-lang
   :alt: PyPI version info
.. image:: https://img.shields.io/pypi/pyversions/viper-lang.svg
   :target: https://pypi.python.org/pypi/viper-lang
   :alt: PyPI supported Python versions

Viper
=====
Viper is a simple, easy to understand language with easy integration capabilities.

Install
--------
| Viper is available for python 3.6+.
| Viper is now on pypi! It can be installed via

.. code:: sh

    python -m pip install viper-lang

The dev version can be installed via

.. code:: sh

    pip install -U git+https://github.com/IAmTomahawkx/viper-lang

And can be imported into your project

.. code:: py

    import viper

Python Usage
-------------
To use Viper in your application, make use of the two eval methods, viper.eval and viper.eval_file functions. These functions
are asynchronous, and must be run using asyncio, whether that be through the `await` keyword, or something such as `asyncio.run`. \
The asyncio docs can be found `here <https://docs.python.org/3/library/asyncio.html#module-asyncio/>`_.

.. code-block:: python

    import viper
    import asyncio
    code = '$myvar = "hi"'
    asyncio.run(viper.eval(code))

or

.. code-block:: python

    import asyncio
    import viper
    asyncio.run(viper.eval_file("myfile.vp"))

you can also pass defaults to be injected into the namespace, as such

.. code-block:: python

    import asyncio
    import viper
    asyncio.run(viper.eval("say($myvar)", {"myvar": "blue"}))

another way to pass defaults is to pass a Viper namespace to the eval

.. code-block:: python

    import asyncio
    import viper
    namespace = viper.VPNamespace()
    namespace['myvar'] = "blue"

    # creating a static variable
    namespace['mystaticvar'] = "red", True

    asyncio.run(viper.eval("say($myvar)", namespace=namespace))

you can disable "unsafe" builtins such as file reading/writing by passing the `safe` keyword to `viper.eval`/`viper.eval_file`.

.. code-block:: python

    import asyncio
    import viper

    asyncio.run(viper.eval('$myvar = read("names.txt")', safe=True)) # raises VP_NameError,
    # as the variable `read` doesnt exist due to safe mode

Speaking of errors, Viper stack traces are now available. They can be accessed by printing out `error.format_stack()` on any VP_Error.

.. code-block:: python

    import asyncio
    import viper

    try:
        asyncio.run(viper.eval("blah"))
    except viper.VP_Error as e:
        print(e.format_stack())

will print out:

.. code-block:: python

    File <string>, top-level:
        blah

    SyntaxError: blah

Syntax
---------

Variables
~~~~~~~~~~
variables are set like in python, but with a dollar sign ($) in front of the name. variables are retrieved in the same way,
the name with a dollar sign ($) in front. variables can be marked as `static` by putting `static` in front of the variable
name. static variables cannot be changed by anything other than intervention in python code

.. code-block::

    $myvar = "red"

    static $mystaticvar = "blue"

    $mystaticvar = "hello"  <-- StaticError

functions
~~~~~~~~~~

functions are created either in python and passed to the namespace, or in viper. functions created in viper follow this syntax

.. code-block::

    func myfunc() {
        return
    }

quite similar to python, with a few key differences. you may put `static ` in front of the `func` keyword to mark the function as static,
preventing it from being reassigned.

.. code-block::

    static func myfunc() {
        return
    }

arguments look like the following

.. code-block::

    func myfunc(argument1, argument2) {
        return
    }

an argument can be made optional by inserting a question mark (?) in front of the argument name, E.x.

.. code-block::

    func myfunc(argument1, ?optional_arg1) {
        return
    }

optional arguments that are not given will be passed as a `none` object (note that this is not the same as a python `None`)

functions are called the same as in python:

.. code-block::

    func myfunc() {
        return
    }
    myfunc()

builtins
~~~~~~~~~
there are several built in functions that will be available inside of viper. They can be seen in the `viper/builtins.py` file.
there are a couple builtin not defined in this file, the `namespace` variable, which points back to the global namespace.
there is also `true` / `false`, which are the viper booleans (AKA python booleans).

a full example
----------------

.. code-block::

    static $globalvar = "hi"

    func name(arg, ?arg1) {
        $var = 1
        if ($var is 1) {
            $var += 1
        }
        else if ($var is not 1) {
            $var = "stuff"
        }
        else {
            $var = none
        }
    }

    func main() {
        say("hi")
        name("hello")
    }
    main()

Customizing viper
-----------------
most of viper can be edited by editing `viper/keywords.py` file. Most of the options are pretty self explanatory. \
These can also be changed at runtime, by importing the keywords file and changing the dictionaries

.. code:: py

    import viper.keywords
    viper.keywords.VIPER_KEYWORDS['VIPER_VARMARKER'] = "%"
    # variables will now be accessed with % instead of $

Discord.py integration
-----------------------
to make things easier, the `viper.exts.discord` module makes it easy to pass safe objects, with limited accessibility, to viper,
making it easy to pass discord.py models (indirectly) to your users, without fear of leaking your token and/or other sensitive data. \
Simply pass a discord.py model to its respective `exts.discord` counterpart, and pass that to your viper namespace

.. code:: py

    import viper
    from viper.exts import discord as viper_discord

    async def on_message(message):
        namespace = viper.VPNamespace()
        safe_message = viper_discord.SafeAccessMessage(message)
        namespace['msg'] = safe_message

        await viper.eval('say($msg.channel.send("hi"))', namespace=namespace, safe=True)


