# # Copyright (c) 2012-2017 The ANTLR Project. All rights reserved. # Use of this file is governed by the BSD 3-clause license that # can be found in the LICENSE.txt file in the project root. # from io import StringIO from antlr4.Token import Token # need forward declarations IntervalSet = None class IntervalSet(object): __slots__ = ('intervals', 'readonly') def __init__(self): self.intervals = None self.readonly = False def __iter__(self): if self.intervals is not None: for i in self.intervals: for c in i: yield c def __getitem__(self, item): i = 0 for k in self: if i==item: return k else: i += 1 return Token.INVALID_TYPE def addOne(self, v:int): self.addRange(range(v, v+1)) def addRange(self, v:range): if self.intervals is None: self.intervals = list() self.intervals.append(v) else: # find insert pos k = 0 for i in self.intervals: # distinct range -> insert if v.stop adjust elif v.stop==i.start: self.intervals[k] = range(v.start, i.stop) return # overlapping range -> adjust and reduce elif v.start<=i.stop: self.intervals[k] = range(min(i.start,v.start), max(i.stop,v.stop)) self.reduce(k) return k += 1 # greater than any existing self.intervals.append(v) def addSet(self, other:IntervalSet): if other.intervals is not None: for i in other.intervals: self.addRange(i) return self def reduce(self, k:int): # only need to reduce if k is not the last if k= r.stop: self.intervals.pop(k+1) self.reduce(k) elif l.stop >= r.start: self.intervals[k] = range(l.start, r.stop) self.intervals.pop(k+1) def complement(self, start, stop): result = IntervalSet() result.addRange(range(start,stop+1)) for i in self.intervals: result.removeRange(i) return result def __contains__(self, item): if self.intervals is None: return False else: return any(item in i for i in self.intervals) def __len__(self): return sum(len(i) for i in self.intervals) def removeRange(self, v): if v.start==v.stop-1: self.removeOne(v.start) elif self.intervals is not None: k = 0 for i in self.intervals: # intervals are ordered if v.stop<=i.start: return # check for including range, split it elif v.start>i.start and v.stop=i.stop: self.intervals.pop(k) k -= 1 # need another pass # check for lower boundary elif v.start1: buf.write("{") first = True for i in self.intervals: for j in i: if not first: buf.write(", ") buf.write(self.elementName(literalNames, symbolicNames, j)) first = False if len(self)>1: buf.write("}") return buf.getvalue() def elementName(self, literalNames:list, symbolicNames:list, a:int): if a==Token.EOF: return "" elif a==Token.EPSILON: return "" else: if a"