-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathkinds.py
77 lines (57 loc) · 1.63 KB
/
kinds.py
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
75
76
77
from abc import ABCMeta, abstractmethod
class Kind(metaclass=ABCMeta):
@abstractmethod
def __repr__(self) -> str:
...
@abstractmethod
def __str__(self) -> str:
...
@abstractmethod
def apply(self, subst: dict):
...
@abstractmethod
def fkv(self) -> set:
...
class KindConstant(Kind):
def __init__(self, name):
self.name = name
def __repr__(self):
if self is Star:
return 'Star'
return f'KindConstant({self.name!r})'
def __str__(self):
return self.name
def apply(self, subst):
return self
def fkv(self):
return set()
Star = KindConstant('★')
class KindVariable(Kind):
def __init__(self, name):
self.name = name
def __repr__(self):
return f'KindVariable({self.name!r})'
def __str__(self):
return self.name
def apply(self, subst):
return subst.get(self.name, self)
def fkv(self):
return {self.name}
class ArrowKind(Kind):
def __init__(self, ks, k):
self.ks = ks
self.k = k
def __repr__(self):
return f'ArrowKind({self.ks!r}, {self.k!r})'
def __str__(self):
ks = ', '.join(map(str, self.ks))
if len(self.ks) == 1:
return f'({ks} → {self.k})'
return f'(({ks}) → {self.k})'
def apply(self, subst):
ks = [k.apply(subst) for k in self.ks]
return ArrowKind(ks, self.k.apply(subst))
def fkv(self):
return set.union(self.k.fkv(), *[k.fkv() for k in self.ks])
UnaryKind = ArrowKind((Star,), Star)
TernaryKind = ArrowKind((Star, Star, Star), Star)