aboutsummaryrefslogtreecommitdiff
path: root/wqflask/wqflask/parser.py
blob: 7711942afc6d3cc6ef1d6e3d8f58718599b4a285 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
"""
Parses search terms input by user

Searches take two primary forms:
- search term by itself (ex. "shh" or "brain")
- key / separator / value(s) (ex. "LRS=(9 99 Chr4 122 155)" or "GO:342533")

In the example of "LRS=(9 99 Chr4 122 155)", the key is "LRS", the separator is "=" and the value
is everything within the parentheses.

Both "=" and ":" can be used as separators; in the future, it would also be good to allow no
separator at all (ex. "cisLRS(9 999 10)")

Both square brackets and parentheses can be used interchangeably. Both can also be used to
encapsulate a single value; "cisLRS=[9 999 10)" would
be acceptable.]

NEED TO DEAL WITH WILDCARD CHARACTER '*'

"""

from __future__ import print_function, division

import re

from pprint import pformat as pf

def parse(pstring):
    pstring = re.split(r"""(?:(\w+\s*=\s*[\(\[][^)]*[\)\]])  |  # LRS=(1 2 3), cisLRS=[4 5 6], etc
                       (\w+\s*[=:]\w+)  |  # wiki=bar, GO:foobar, etc
                       (\w+))  # shh, brain, etc """, pstring,
                                                    flags=re.VERBOSE)
    pstring = [item.strip() for item in pstring if item and item.strip()]
    print(pstring)

    items = []

    for item in pstring:
        if ":" in item:
            key, seperator, value = item.partition(':')
        elif "=" in item:
            key, seperator, value = item.partition('=')
        else:
            seperator = None

        if seperator:
            if '(' in value or '[' in value:
                assert value.startswith(("(", "[")), "Invalid token"
                assert value.endswith((")", "]")), "Invalid token"
                value = value[1:-1] # Get rid of the parenthesis
                values = re.split(r"""\s+|,""", value)
                value = [value.strip() for value in values if value.strip()]
            term = dict(key=key,
                        seperator=seperator,
                        search_term=value)
        else:
            term = dict(key=None,
                        seperator=None,
                        search_term = item)

        items.append(term)
    print(pf(items))
    return(items)

if __name__ == '__main__':
    parse("foo=[3 2 1]")
    parse("foo=[3 2 1)")
    parse("foo=(3 2 1)")
    parse("shh")
    parse("shh grep")
    parse("LRS=(9 99 Chr4 122 155) cisLRS=(9 999 10)")
    parse("sal1 LRS=(9 99 Chr4 122 155) sal2 cisLRS=(9 999 10)")
    parse("sal1 sal3 LRS=(9 99 Chr4 122 155) wiki=bar sal2 go:foobar cisLRS=(9 999 10)")
    parse("sal1 LRS=(9 99 Chr4 122 155) wiki=bar sal2 go:foobar cisLRS=(9, 999, 10)")