aboutsummaryrefslogtreecommitdiff
import decimal, re, typing
from src.utils import datetime, errors, io

from .spec import *
from .time import duration
from .types import try_int
from . import sed

COMMENT_TYPES = ["#", "//"]
def hashflags(filename: str
        ) -> typing.List[typing.Tuple[str, typing.Optional[str]]]:
    hashflags = [] # type: typing.List[typing.Tuple[str, typing.Optional[str]]]
    with io.open(filename, "r") as f:
        for line in f:
            line = line.strip("\n")
            found = False
            for comment_type in COMMENT_TYPES:
                if line.startswith(comment_type):
                    line = line.replace(comment_type, "", 1).lstrip()
                    found = True
                    break

            if not found:
                break
            elif line.startswith("--"):
                hashflag, sep, value = line[2:].partition(" ")
                hashflags.append((hashflag, (value if sep else None)))
    return hashflags

class Docstring(object):
    def __init__(self, description: str, items: typing.Dict[str, str],
            var_items: typing.Dict[str, typing.List[str]]):
        self.description = description
        self.items = items
        self.var_items = var_items

def docstring(s: str) -> Docstring:
    description = ""
    last_item = None
    last_item_no_space = False
    items = {} # type: typing.Dict[str, str]
    var_items = {} # type: typing.Dict[str, typing.List[str]]
    if s:
        for line in s.split("\n"):
            line = line.strip()

            if line:
                if line[0] == ":":
                    key, _, value = line[1:].partition(": ")
                    last_item = key.lstrip("-")
                    last_item_no_space = key.startswith("-")

                    if key in var_items:
                        var_items[last_item].append(value)
                    elif key in items:
                        var_items[last_item] = [items.pop(last_item), value]
                    else:
                        items[last_item] = value
                else:
                    if last_item:
                        if last_item_no_space:
                            items[last_item] += line
                        else:
                            items[last_item] += " %s" % line
                    else:
                        if description:
                            description += " "
                        description += line
    return Docstring(description, items, var_items)

def keyvalue(s: str, delimiter: str=" "
        ) -> typing.Dict[str, typing.Optional[str]]:
    items = {} # type: typing.Dict[str, typing.Optional[str]]
    pairs = s.split(delimiter)
    for pair in filter(None, pairs):
        key, sep, value = pair.partition("=")
        if sep:
            items[key] = value
        else:
            items[key] = None
    return items

def line_normalise(s: str) -> str:
    lines = list(filter(None, [line.strip() for line in s.split("\n")]))
    return "  ".join(line.replace("  ", " ") for line in lines)

def parse_number(s: str) -> str:
    try:
        decimal.Decimal(s)
        return s
    except:
        pass

    unit = s[-1].lower()
    number_str = s[:-1]
    try:
        number = decimal.Decimal(number_str)
    except:
        raise ValueError("Invalid format '%s' passed to parse_number" %
            number_str)

    if unit == "k":
        number *= decimal.Decimal("1_000")
    elif unit == "m":
        number *= decimal.Decimal("1_000_000")
    elif unit == "b":
        number *= decimal.Decimal("1_000_000_000")
    else:
        raise ValueError("Unknown unit '%s' given to parse_number" % unit)
    return str(number)

def format_tokens(s: str, sigil: str="$"
        ) -> typing.List[typing.Tuple[int, int, str]]:
    i = 0
    max = len(s)-1
    sigil_found = False
    tokens: typing.List[typing.Tuple[int, int, str]] = []

    while i < max:
        if s[i] == sigil:
            i += 1
            if s[i] == "{":
                token_end = s.find("}", i)
                if token_end > i:
                    token = s[i:token_end]
                    tokens.append((i-1, token_end, s[i+1:token_end]))
                    i = token_end
            elif s[i] == sigil:
                tokens.append((i-1, i, sigil))
        i += 1
    return tokens

def format_token_replace(s: str, vars: typing.Dict[str, str],
        sigil: str="$") -> typing.Tuple[typing.List[str], str]:
    vars = vars.copy()
    vars.update({sigil: sigil})

    tokens = format_tokens(s, sigil)
    # reverse sort tokens so replaces don't effect proceeding indexes
    tokens.sort(key=lambda x: x[0])
    tokens.reverse()

    not_found: typing.List[str] = []
    for start, end, token in tokens:
        if token in vars:
            s = s[:start] + vars[token] + s[end+1:]
        else:
            not_found += token
    return not_found, s