File size: 2,436 Bytes
762d748 | 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 78 79 80 81 82 83 | from __future__ import annotations
from .exceptions import ParseError
from typing import NamedTuple
COMMENTCHARS = "#;"
class _ParsedLine(NamedTuple):
lineno: int
section: str | None
name: str | None
value: str | None
def parse_lines(path: str, line_iter: list[str]) -> list[_ParsedLine]:
result: list[_ParsedLine] = []
section = None
for lineno, line in enumerate(line_iter):
name, data = _parseline(path, line, lineno)
# new value
if name is not None and data is not None:
result.append(_ParsedLine(lineno, section, name, data))
# new section
elif name is not None and data is None:
if not name:
raise ParseError(path, lineno, "empty section name")
section = name
result.append(_ParsedLine(lineno, section, None, None))
# continuation
elif name is None and data is not None:
if not result:
raise ParseError(path, lineno, "unexpected value continuation")
last = result.pop()
if last.name is None:
raise ParseError(path, lineno, "unexpected value continuation")
if last.value:
last = last._replace(value=f"{last.value}\n{data}")
else:
last = last._replace(value=data)
result.append(last)
return result
def _parseline(path: str, line: str, lineno: int) -> tuple[str | None, str | None]:
# blank lines
if iscommentline(line):
line = ""
else:
line = line.rstrip()
if not line:
return None, None
# section
if line[0] == "[":
realline = line
for c in COMMENTCHARS:
line = line.split(c)[0].rstrip()
if line[-1] == "]":
return line[1:-1], None
return None, realline.strip()
# value
elif not line[0].isspace():
try:
name, value = line.split("=", 1)
if ":" in name:
raise ValueError()
except ValueError:
try:
name, value = line.split(":", 1)
except ValueError:
raise ParseError(path, lineno, "unexpected line: %r" % line)
return name.strip(), value.strip()
# continuation
else:
return None, line.strip()
def iscommentline(line: str) -> bool:
c = line.lstrip()[:1]
return c in COMMENTCHARS
|