omlish 0.0.0.dev134__py3-none-any.whl → 0.0.0.dev137__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- omlish/.manifests.json +0 -12
- omlish/__about__.py +2 -2
- omlish/cached.py +2 -2
- omlish/collections/mappings.py +1 -1
- omlish/diag/_pycharm/runhack.py +3 -0
- omlish/formats/json/stream/lex.py +1 -1
- omlish/formats/json/stream/parse.py +1 -1
- omlish/{genmachine.py → funcs/genmachine.py} +4 -2
- omlish/{matchfns.py → funcs/match.py} +1 -1
- omlish/{fnpairs.py → funcs/pairs.py} +3 -3
- omlish/http/sessions.py +1 -1
- omlish/io/compress/__init__.py +0 -0
- omlish/io/compress/abc.py +104 -0
- omlish/io/compress/adapters.py +147 -0
- omlish/io/compress/bz2.py +41 -0
- omlish/io/compress/gzip.py +301 -0
- omlish/io/compress/lzma.py +31 -0
- omlish/io/compress/types.py +29 -0
- omlish/io/generators.py +50 -0
- omlish/lang/__init__.py +8 -1
- omlish/lang/generators.py +182 -0
- omlish/lang/iterables.py +28 -51
- omlish/lang/maybes.py +4 -4
- omlish/lite/fdio/corohttp.py +5 -1
- omlish/lite/marshal.py +9 -6
- omlish/marshal/base.py +1 -1
- omlish/marshal/factories.py +1 -1
- omlish/marshal/forbidden.py +1 -1
- omlish/marshal/iterables.py +1 -1
- omlish/marshal/mappings.py +1 -1
- omlish/marshal/maybes.py +1 -1
- omlish/marshal/standard.py +1 -1
- omlish/marshal/unions.py +1 -1
- omlish/secrets/pwhash.py +1 -1
- omlish/secrets/subprocesses.py +3 -1
- omlish/specs/jsonrpc/marshal.py +1 -1
- omlish/specs/openapi/marshal.py +1 -1
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/METADATA +1 -1
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/RECORD +49 -47
- omlish/formats/json/cli/__main__.py +0 -11
- omlish/formats/json/cli/cli.py +0 -298
- omlish/formats/json/cli/formats.py +0 -71
- omlish/formats/json/cli/io.py +0 -74
- omlish/formats/json/cli/parsing.py +0 -82
- omlish/formats/json/cli/processing.py +0 -48
- omlish/formats/json/cli/rendering.py +0 -92
- /omlish/collections/{_abc.py → abc.py} +0 -0
- /omlish/{formats/json/cli → funcs}/__init__.py +0 -0
- /omlish/{fnpipes.py → funcs/pipes.py} +0 -0
- /omlish/io/{_abc.py → abc.py} +0 -0
- /omlish/logs/{_abc.py → abc.py} +0 -0
- /omlish/sql/{_abc.py → abc.py} +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/WHEEL +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/top_level.txt +0 -0
    
        omlish/formats/json/cli/cli.py
    DELETED
    
    | @@ -1,298 +0,0 @@ | |
| 1 | 
            -
            """
         | 
| 2 | 
            -
            TODO:
         | 
| 3 | 
            -
             - read from http
         | 
| 4 | 
            -
             - jmespath output flat, unquoted strs like jq '.[]'
         | 
| 5 | 
            -
             | 
| 6 | 
            -
            ==
         | 
| 7 | 
            -
             | 
| 8 | 
            -
            jq Command options:
         | 
| 9 | 
            -
              -n, --null-input          use `null` as the single input value;
         | 
| 10 | 
            -
              -R, --raw-input           read each line as string instead of JSON;
         | 
| 11 | 
            -
              -s, --slurp               read all inputs into an array and use it as the single input value;
         | 
| 12 | 
            -
              -c, --compact-output      compact instead of pretty-printed output;
         | 
| 13 | 
            -
              -r, --raw-output          output strings without escapes and quotes;
         | 
| 14 | 
            -
                  --raw-output0         implies -r and output NUL after each output;
         | 
| 15 | 
            -
              -j, --join-output         implies -r and output without newline after each output;
         | 
| 16 | 
            -
              -a, --ascii-output        output strings by only ASCII characters using escape sequences;
         | 
| 17 | 
            -
              -S, --sort-keys           sort keys of each object on output;
         | 
| 18 | 
            -
              -C, --color-output        colorize JSON output;
         | 
| 19 | 
            -
              -M, --monochrome-output   disable colored output;
         | 
| 20 | 
            -
                  --tab                 use tabs for indentation;
         | 
| 21 | 
            -
                  --indent n            use n spaces for indentation (max 7 spaces);
         | 
| 22 | 
            -
                  --unbuffered          flush output stream after each output;
         | 
| 23 | 
            -
                  --stream              parse the input value in streaming fashion;
         | 
| 24 | 
            -
                  --stream-errors       implies --stream and report parse error as an array;
         | 
| 25 | 
            -
                  --seq                 parse input/output as application/json-seq;
         | 
| 26 | 
            -
              -f, --from-file file      load filter from the file;
         | 
| 27 | 
            -
              -L directory              search modules from the directory;
         | 
| 28 | 
            -
                  --arg name value      set $name to the string value;
         | 
| 29 | 
            -
                  --argjson name value  set $name to the JSON value;
         | 
| 30 | 
            -
                  --slurpfile name file set $name to an array of JSON values read from the file;
         | 
| 31 | 
            -
                  --rawfile name file   set $name to string contents of file;
         | 
| 32 | 
            -
                  --args                consume remaining arguments as positional string values;
         | 
| 33 | 
            -
                  --jsonargs            consume remaining arguments as positional JSON values;
         | 
| 34 | 
            -
              -e, --exit-status         set exit status code based on the output;
         | 
| 35 | 
            -
              -V, --version             show the version;
         | 
| 36 | 
            -
              --build-configuration     show jq's build configuration;
         | 
| 37 | 
            -
              -h, --help                show the help;
         | 
| 38 | 
            -
              --                        terminates argument processing;
         | 
| 39 | 
            -
            """
         | 
| 40 | 
            -
            import argparse
         | 
| 41 | 
            -
            import contextlib
         | 
| 42 | 
            -
            import dataclasses as dc
         | 
| 43 | 
            -
            import io
         | 
| 44 | 
            -
            import os
         | 
| 45 | 
            -
            import subprocess
         | 
| 46 | 
            -
            import sys
         | 
| 47 | 
            -
            import typing as ta
         | 
| 48 | 
            -
             | 
| 49 | 
            -
            from .... import check
         | 
| 50 | 
            -
            from .... import fnpipes as fp
         | 
| 51 | 
            -
            from .... import lang
         | 
| 52 | 
            -
            from .formats import FORMATS_BY_NAME
         | 
| 53 | 
            -
            from .formats import Format
         | 
| 54 | 
            -
            from .formats import Formats
         | 
| 55 | 
            -
            from .parsing import DelimitingParser
         | 
| 56 | 
            -
            from .parsing import EagerParser
         | 
| 57 | 
            -
            from .parsing import StreamBuilder
         | 
| 58 | 
            -
            from .parsing import StreamParser
         | 
| 59 | 
            -
            from .processing import ProcessingOptions
         | 
| 60 | 
            -
            from .processing import Processor
         | 
| 61 | 
            -
            from .rendering import EagerRenderer
         | 
| 62 | 
            -
            from .rendering import RenderingOptions
         | 
| 63 | 
            -
            from .rendering import StreamRenderer
         | 
| 64 | 
            -
             | 
| 65 | 
            -
             | 
| 66 | 
            -
            T = ta.TypeVar('T')
         | 
| 67 | 
            -
            U = ta.TypeVar('U')
         | 
| 68 | 
            -
             | 
| 69 | 
            -
             | 
| 70 | 
            -
            def _build_args_parser() -> argparse.ArgumentParser:
         | 
| 71 | 
            -
                parser = argparse.ArgumentParser()
         | 
| 72 | 
            -
             | 
| 73 | 
            -
                parser.add_argument('file', nargs='?')
         | 
| 74 | 
            -
             | 
| 75 | 
            -
                parser.add_argument('--stream', action='store_true')
         | 
| 76 | 
            -
                parser.add_argument('--stream-build', action='store_true')
         | 
| 77 | 
            -
             | 
| 78 | 
            -
                parser.add_argument('-l', '--lines', action='store_true')
         | 
| 79 | 
            -
             | 
| 80 | 
            -
                parser.add_argument('--read-buffer-size', type=int, default=0x4000)
         | 
| 81 | 
            -
             | 
| 82 | 
            -
                parser.add_argument('-f', '--format')
         | 
| 83 | 
            -
             | 
| 84 | 
            -
                parser.add_argument('-x', '--jmespath-expr')
         | 
| 85 | 
            -
                parser.add_argument('-F', '--flat', action='store_true')
         | 
| 86 | 
            -
             | 
| 87 | 
            -
                parser.add_argument('-z', '--compact', action='store_true')
         | 
| 88 | 
            -
                parser.add_argument('-p', '--pretty', action='store_true')
         | 
| 89 | 
            -
                parser.add_argument('-i', '--indent')
         | 
| 90 | 
            -
                parser.add_argument('-s', '--sort-keys', action='store_true')
         | 
| 91 | 
            -
                parser.add_argument('-R', '--raw', action='store_true')
         | 
| 92 | 
            -
                parser.add_argument('-U', '--unicode', action='store_true')
         | 
| 93 | 
            -
                parser.add_argument('-c', '--color', action='store_true')
         | 
| 94 | 
            -
             | 
| 95 | 
            -
                parser.add_argument('-L', '--less', action='store_true')
         | 
| 96 | 
            -
             | 
| 97 | 
            -
                return parser
         | 
| 98 | 
            -
             | 
| 99 | 
            -
             | 
| 100 | 
            -
            def _parse_args(args: ta.Any = None) -> ta.Any:
         | 
| 101 | 
            -
                return _build_args_parser().parse_args(args)
         | 
| 102 | 
            -
             | 
| 103 | 
            -
             | 
| 104 | 
            -
            @dc.dataclass(frozen=True, kw_only=True)
         | 
| 105 | 
            -
            class RunConfiguration:
         | 
| 106 | 
            -
                format: Format
         | 
| 107 | 
            -
                processing: ProcessingOptions
         | 
| 108 | 
            -
                rendering: RenderingOptions
         | 
| 109 | 
            -
             | 
| 110 | 
            -
             | 
| 111 | 
            -
            def _process_args(args: ta.Any) -> RunConfiguration:
         | 
| 112 | 
            -
                fmt_name = args.format
         | 
| 113 | 
            -
                if fmt_name is None:
         | 
| 114 | 
            -
                    if args.file is not None:
         | 
| 115 | 
            -
                        ext = args.file.rpartition('.')[2]
         | 
| 116 | 
            -
                        if ext in FORMATS_BY_NAME:
         | 
| 117 | 
            -
                            fmt_name = ext
         | 
| 118 | 
            -
                if fmt_name is None:
         | 
| 119 | 
            -
                    fmt_name = 'json'
         | 
| 120 | 
            -
                format = FORMATS_BY_NAME[fmt_name]  # noqa
         | 
| 121 | 
            -
             | 
| 122 | 
            -
                if args.stream:
         | 
| 123 | 
            -
                    check.arg(format is Formats.JSON.value)
         | 
| 124 | 
            -
             | 
| 125 | 
            -
                #
         | 
| 126 | 
            -
             | 
| 127 | 
            -
                processing = ProcessingOptions(
         | 
| 128 | 
            -
                    jmespath_expr=args.jmespath_expr,
         | 
| 129 | 
            -
                    flat=args.flat,
         | 
| 130 | 
            -
                )
         | 
| 131 | 
            -
             | 
| 132 | 
            -
                #
         | 
| 133 | 
            -
             | 
| 134 | 
            -
                separators = None
         | 
| 135 | 
            -
                if args.compact:
         | 
| 136 | 
            -
                    separators = (',', ':')
         | 
| 137 | 
            -
             | 
| 138 | 
            -
                indent = None
         | 
| 139 | 
            -
                if args.pretty:
         | 
| 140 | 
            -
                    indent = 2
         | 
| 141 | 
            -
                if args.indent:
         | 
| 142 | 
            -
                    try:
         | 
| 143 | 
            -
                        indent = int(args.indent)
         | 
| 144 | 
            -
                    except ValueError:
         | 
| 145 | 
            -
                        indent = args.indent
         | 
| 146 | 
            -
             | 
| 147 | 
            -
                rendering = RenderingOptions(
         | 
| 148 | 
            -
                    indent=indent,
         | 
| 149 | 
            -
                    separators=separators,
         | 
| 150 | 
            -
                    sort_keys=args.sort_keys,
         | 
| 151 | 
            -
                    raw=args.raw,
         | 
| 152 | 
            -
                    unicode=args.unicode,
         | 
| 153 | 
            -
                    color=args.color,
         | 
| 154 | 
            -
                )
         | 
| 155 | 
            -
             | 
| 156 | 
            -
                #
         | 
| 157 | 
            -
             | 
| 158 | 
            -
                return RunConfiguration(
         | 
| 159 | 
            -
                    format=format,
         | 
| 160 | 
            -
                    processing=processing,
         | 
| 161 | 
            -
                    rendering=rendering,
         | 
| 162 | 
            -
                )
         | 
| 163 | 
            -
             | 
| 164 | 
            -
             | 
| 165 | 
            -
            def _main() -> None:
         | 
| 166 | 
            -
                args = _parse_args()
         | 
| 167 | 
            -
             | 
| 168 | 
            -
                #
         | 
| 169 | 
            -
             | 
| 170 | 
            -
                cfg = _process_args(args)
         | 
| 171 | 
            -
             | 
| 172 | 
            -
                #
         | 
| 173 | 
            -
             | 
| 174 | 
            -
                with contextlib.ExitStack() as es:
         | 
| 175 | 
            -
                    if args.file is None:
         | 
| 176 | 
            -
                        in_file = sys.stdin.buffer
         | 
| 177 | 
            -
             | 
| 178 | 
            -
                    else:
         | 
| 179 | 
            -
                        in_file = es.enter_context(open(args.file, 'rb'))
         | 
| 180 | 
            -
             | 
| 181 | 
            -
                    def yield_input() -> ta.Generator[bytes, None, None]:
         | 
| 182 | 
            -
                        fd = check.isinstance(in_file.fileno(), int)
         | 
| 183 | 
            -
             | 
| 184 | 
            -
                        while True:
         | 
| 185 | 
            -
                            buf = os.read(fd, args.read_buffer_size)
         | 
| 186 | 
            -
             | 
| 187 | 
            -
                            yield buf
         | 
| 188 | 
            -
             | 
| 189 | 
            -
                            if not buf:
         | 
| 190 | 
            -
                                break
         | 
| 191 | 
            -
             | 
| 192 | 
            -
                    #
         | 
| 193 | 
            -
             | 
| 194 | 
            -
                    if args.less:
         | 
| 195 | 
            -
                        less = subprocess.Popen(
         | 
| 196 | 
            -
                            [
         | 
| 197 | 
            -
                                'less',
         | 
| 198 | 
            -
                                *(['-R'] if cfg.rendering.color else []),
         | 
| 199 | 
            -
                            ],
         | 
| 200 | 
            -
                            stdin=subprocess.PIPE,
         | 
| 201 | 
            -
                            encoding='utf-8',
         | 
| 202 | 
            -
                        )
         | 
| 203 | 
            -
                        out = check.not_none(less.stdin)
         | 
| 204 | 
            -
             | 
| 205 | 
            -
                        def close_less() -> None:
         | 
| 206 | 
            -
                            out.close()
         | 
| 207 | 
            -
                            less.wait()
         | 
| 208 | 
            -
             | 
| 209 | 
            -
                        es.enter_context(lang.defer(close_less))  # noqa
         | 
| 210 | 
            -
             | 
| 211 | 
            -
                    else:
         | 
| 212 | 
            -
                        out = sys.stdout
         | 
| 213 | 
            -
             | 
| 214 | 
            -
                    #
         | 
| 215 | 
            -
             | 
| 216 | 
            -
                    parser: ta.Any
         | 
| 217 | 
            -
                    renderer: ta.Any
         | 
| 218 | 
            -
             | 
| 219 | 
            -
                    if args.stream:
         | 
| 220 | 
            -
                        with contextlib.ExitStack() as es2:
         | 
| 221 | 
            -
                            parser = es2.enter_context(StreamParser())
         | 
| 222 | 
            -
             | 
| 223 | 
            -
                            def flush_output(
         | 
| 224 | 
            -
                                    fn: ta.Callable[[T], ta.Iterable[U]],
         | 
| 225 | 
            -
                                    i: T,
         | 
| 226 | 
            -
                            ) -> ta.Generator[U, None, None]:
         | 
| 227 | 
            -
                                n = 0
         | 
| 228 | 
            -
                                for o in fn(i):
         | 
| 229 | 
            -
                                    yield o
         | 
| 230 | 
            -
                                    n += 1
         | 
| 231 | 
            -
                                if n:
         | 
| 232 | 
            -
                                    out.flush()
         | 
| 233 | 
            -
             | 
| 234 | 
            -
                            pipeline: ta.Any
         | 
| 235 | 
            -
             | 
| 236 | 
            -
                            if args.stream_build:
         | 
| 237 | 
            -
                                builder: StreamBuilder = es2.enter_context(StreamBuilder())
         | 
| 238 | 
            -
                                processor = Processor(cfg.processing)
         | 
| 239 | 
            -
                                renderer = EagerRenderer(cfg.rendering)
         | 
| 240 | 
            -
                                trailing_newline = False
         | 
| 241 | 
            -
             | 
| 242 | 
            -
                                def append_newlines(
         | 
| 243 | 
            -
                                        fn: ta.Callable[[T], ta.Iterable[str]],
         | 
| 244 | 
            -
                                        i: T,
         | 
| 245 | 
            -
                                ) -> ta.Generator[str, None, None]:
         | 
| 246 | 
            -
                                    yield from fn(i)
         | 
| 247 | 
            -
                                    yield '\n'
         | 
| 248 | 
            -
             | 
| 249 | 
            -
                                pipeline = lambda v: (renderer.render(v),)  # Any -> [str]  # noqa
         | 
| 250 | 
            -
                                pipeline = fp.bind(append_newlines, pipeline)  # Any -> [str]
         | 
| 251 | 
            -
                                pipeline = fp.bind(lang.flatmap, pipeline)  # [Any] -> [str]
         | 
| 252 | 
            -
                                pipeline = fp.pipe(fp.bind(lang.flatmap, processor.process), pipeline)  # [Any] -> [str]
         | 
| 253 | 
            -
                                pipeline = fp.pipe(fp.bind(lang.flatmap, builder.build), pipeline)  # [JsonStreamParserEvent] -> [str]  # noqa
         | 
| 254 | 
            -
                                pipeline = fp.pipe(parser.parse, pipeline)  # bytes -> [str]
         | 
| 255 | 
            -
             | 
| 256 | 
            -
                            else:
         | 
| 257 | 
            -
                                renderer = StreamRenderer(cfg.rendering)
         | 
| 258 | 
            -
                                trailing_newline = True
         | 
| 259 | 
            -
             | 
| 260 | 
            -
                                pipeline = renderer.render  # JsonStreamParserEvent -> [str]
         | 
| 261 | 
            -
                                pipeline = fp.bind(lang.flatmap, pipeline)  # [JsonStreamParserEvent] -> [str]
         | 
| 262 | 
            -
                                pipeline = fp.pipe(parser.parse, pipeline)  # bytes -> [str]
         | 
| 263 | 
            -
             | 
| 264 | 
            -
                            pipeline = fp.bind(flush_output, pipeline)  # bytes -> [str]
         | 
| 265 | 
            -
             | 
| 266 | 
            -
                            for buf in yield_input():
         | 
| 267 | 
            -
                                for s in pipeline(buf):
         | 
| 268 | 
            -
                                    print(s, file=out, end='')
         | 
| 269 | 
            -
             | 
| 270 | 
            -
                            if trailing_newline:
         | 
| 271 | 
            -
                                print(file=out)
         | 
| 272 | 
            -
             | 
| 273 | 
            -
                    elif args.lines:
         | 
| 274 | 
            -
                        parser = DelimitingParser(cfg.format)
         | 
| 275 | 
            -
                        processor = Processor(cfg.processing)
         | 
| 276 | 
            -
                        renderer = EagerRenderer(cfg.rendering)
         | 
| 277 | 
            -
             | 
| 278 | 
            -
                        for buf in yield_input():
         | 
| 279 | 
            -
                            for v in parser.parse(buf):
         | 
| 280 | 
            -
                                for e in processor.process(v):
         | 
| 281 | 
            -
                                    s = renderer.render(e)
         | 
| 282 | 
            -
                                    print(s, file=out)
         | 
| 283 | 
            -
             | 
| 284 | 
            -
                    else:
         | 
| 285 | 
            -
                        parser = EagerParser(cfg.format)
         | 
| 286 | 
            -
                        processor = Processor(cfg.processing)
         | 
| 287 | 
            -
                        renderer = EagerRenderer(cfg.rendering)
         | 
| 288 | 
            -
             | 
| 289 | 
            -
                        with io.TextIOWrapper(in_file) as tf:
         | 
| 290 | 
            -
                            v = parser.parse(tf)
         | 
| 291 | 
            -
             | 
| 292 | 
            -
                        for e in processor.process(v):
         | 
| 293 | 
            -
                            s = renderer.render(e)
         | 
| 294 | 
            -
                            print(s, file=out)
         | 
| 295 | 
            -
             | 
| 296 | 
            -
             | 
| 297 | 
            -
            if __name__ == '__main__':
         | 
| 298 | 
            -
                _main()
         | 
| @@ -1,71 +0,0 @@ | |
| 1 | 
            -
            """
         | 
| 2 | 
            -
            TODO:
         | 
| 3 | 
            -
             - options lol - csv header, newline, etc
         | 
| 4 | 
            -
            """
         | 
| 5 | 
            -
            import dataclasses as dc
         | 
| 6 | 
            -
            import enum
         | 
| 7 | 
            -
            import json
         | 
| 8 | 
            -
            import typing as ta
         | 
| 9 | 
            -
             | 
| 10 | 
            -
            from .... import lang
         | 
| 11 | 
            -
             | 
| 12 | 
            -
             | 
| 13 | 
            -
            if ta.TYPE_CHECKING:
         | 
| 14 | 
            -
                import ast
         | 
| 15 | 
            -
                import csv
         | 
| 16 | 
            -
                import tomllib
         | 
| 17 | 
            -
             | 
| 18 | 
            -
                import yaml
         | 
| 19 | 
            -
             | 
| 20 | 
            -
                from ... import dotenv
         | 
| 21 | 
            -
                from ... import props
         | 
| 22 | 
            -
                from ... import xml
         | 
| 23 | 
            -
             | 
| 24 | 
            -
            else:
         | 
| 25 | 
            -
                ast = lang.proxy_import('ast')
         | 
| 26 | 
            -
                csv = lang.proxy_import('csv')
         | 
| 27 | 
            -
                tomllib = lang.proxy_import('tomllib')
         | 
| 28 | 
            -
             | 
| 29 | 
            -
                yaml = lang.proxy_import('yaml')
         | 
| 30 | 
            -
             | 
| 31 | 
            -
                dotenv = lang.proxy_import('...dotenv', __package__)
         | 
| 32 | 
            -
                props = lang.proxy_import('...props', __package__)
         | 
| 33 | 
            -
                xml = lang.proxy_import('...xml', __package__)
         | 
| 34 | 
            -
             | 
| 35 | 
            -
             | 
| 36 | 
            -
            ##
         | 
| 37 | 
            -
             | 
| 38 | 
            -
             | 
| 39 | 
            -
            @dc.dataclass(frozen=True)
         | 
| 40 | 
            -
            class Format:
         | 
| 41 | 
            -
                names: ta.Sequence[str]
         | 
| 42 | 
            -
                load: ta.Callable[[ta.TextIO], ta.Any]
         | 
| 43 | 
            -
             | 
| 44 | 
            -
             | 
| 45 | 
            -
            class Formats(enum.Enum):
         | 
| 46 | 
            -
                JSON = Format(['json'], json.load)
         | 
| 47 | 
            -
             | 
| 48 | 
            -
                YAML = Format(['yaml', 'yml'], lambda f: yaml.safe_load(f))
         | 
| 49 | 
            -
             | 
| 50 | 
            -
                TOML = Format(['toml'], lambda f: tomllib.loads(f.read()))
         | 
| 51 | 
            -
             | 
| 52 | 
            -
                ENV = Format(['env', 'dotenv'], lambda f: dotenv.dotenv_values(stream=f))
         | 
| 53 | 
            -
             | 
| 54 | 
            -
                PROPS = Format(['properties', 'props'], lambda f: dict(props.Properties().load(f.read())))
         | 
| 55 | 
            -
             | 
| 56 | 
            -
                PY = Format(['py', 'python', 'repr'], lambda f: ast.literal_eval(f.read()))
         | 
| 57 | 
            -
             | 
| 58 | 
            -
                XML = Format(['xml'], lambda f: xml.build_simple_element(xml.parse_tree(f.read()).getroot()).as_dict())
         | 
| 59 | 
            -
             | 
| 60 | 
            -
                CSV = Format(['csv'], lambda f: list(csv.DictReader(f)))
         | 
| 61 | 
            -
                TSV = Format(['tsv'], lambda f: list(csv.DictReader(f, delimiter='\t')))
         | 
| 62 | 
            -
                FLAT_CSV = Format(['fcsv'], lambda f: list(csv.reader(f)))
         | 
| 63 | 
            -
                FLAT_TSV = Format(['ftsv'], lambda f: list(csv.reader(f, delimiter='\t')))
         | 
| 64 | 
            -
             | 
| 65 | 
            -
             | 
| 66 | 
            -
            FORMATS_BY_NAME: ta.Mapping[str, Format] = {
         | 
| 67 | 
            -
                n: f
         | 
| 68 | 
            -
                for e in Formats
         | 
| 69 | 
            -
                for f in [e.value]
         | 
| 70 | 
            -
                for n in f.names
         | 
| 71 | 
            -
            }
         | 
    
        omlish/formats/json/cli/io.py
    DELETED
    
    | @@ -1,74 +0,0 @@ | |
| 1 | 
            -
            # """
         | 
| 2 | 
            -
            # TODO:
         | 
| 3 | 
            -
            #  - -I/-Ogz, lz4, etc
         | 
| 4 | 
            -
            #   - fnpairs? or not yet just do it
         | 
| 5 | 
            -
            # """
         | 
| 6 | 
            -
            # import abc
         | 
| 7 | 
            -
            # import contextlib
         | 
| 8 | 
            -
            # import dataclasses as dc
         | 
| 9 | 
            -
            # import gzip
         | 
| 10 | 
            -
            # import os
         | 
| 11 | 
            -
            # import typing as ta
         | 
| 12 | 
            -
            #
         | 
| 13 | 
            -
            # from .... import lang
         | 
| 14 | 
            -
            #
         | 
| 15 | 
            -
            #
         | 
| 16 | 
            -
            # if ta.TYPE_CHECKING:
         | 
| 17 | 
            -
            #     import bz2 as _bz2
         | 
| 18 | 
            -
            #     import gzip as _gzip
         | 
| 19 | 
            -
            #     import lzma as _lzma
         | 
| 20 | 
            -
            # else:
         | 
| 21 | 
            -
            #     _bz2 = lang.proxy_import('bz2')
         | 
| 22 | 
            -
            #     _gzip = lang.proxy_import('gzip')
         | 
| 23 | 
            -
            #     _lzma = lang.proxy_import('lzma')
         | 
| 24 | 
            -
            #
         | 
| 25 | 
            -
            #
         | 
| 26 | 
            -
            # ##
         | 
| 27 | 
            -
            #
         | 
| 28 | 
            -
            #
         | 
| 29 | 
            -
            # class BaseIo(lang.Abstract):
         | 
| 30 | 
            -
            #     def close(self) -> None:
         | 
| 31 | 
            -
            #         raise NotImplementedError
         | 
| 32 | 
            -
            #
         | 
| 33 | 
            -
            #     def fileno(self) -> int | None:
         | 
| 34 | 
            -
            #         return None
         | 
| 35 | 
            -
            #
         | 
| 36 | 
            -
            #
         | 
| 37 | 
            -
            # class Input(BaseIo):
         | 
| 38 | 
            -
            #     @abc.abstractmethod
         | 
| 39 | 
            -
            #     def read(self, sz: int | None = None) -> bytes:
         | 
| 40 | 
            -
            #         raise NotImplementedError
         | 
| 41 | 
            -
            #
         | 
| 42 | 
            -
            #
         | 
| 43 | 
            -
            # class Output(BaseIo):
         | 
| 44 | 
            -
            #     @abc.abstractmethod
         | 
| 45 | 
            -
            #     def write(self, data: bytes) -> int:
         | 
| 46 | 
            -
            #         raise NotImplementedError
         | 
| 47 | 
            -
            #
         | 
| 48 | 
            -
            #
         | 
| 49 | 
            -
            # #
         | 
| 50 | 
            -
            #
         | 
| 51 | 
            -
            #
         | 
| 52 | 
            -
            # DEFAULT_READ_SZ = 0x4000
         | 
| 53 | 
            -
            #
         | 
| 54 | 
            -
            #
         | 
| 55 | 
            -
            # @dc.dataclass(frozen=True)
         | 
| 56 | 
            -
            # class FdIo(Input, Output):
         | 
| 57 | 
            -
            #     fd: int
         | 
| 58 | 
            -
            #
         | 
| 59 | 
            -
            #     default_read_sz: int = DEFAULT_READ_SZ
         | 
| 60 | 
            -
            #
         | 
| 61 | 
            -
            #     def read(self, sz: int | None = None) -> bytes:
         | 
| 62 | 
            -
            #         return os.read(self.fd, sz or self.default_read_sz)
         | 
| 63 | 
            -
            #
         | 
| 64 | 
            -
            #     def write(self, data: bytes) -> int:
         | 
| 65 | 
            -
            #         return os.write(self.fd, data)
         | 
| 66 | 
            -
            #
         | 
| 67 | 
            -
            #
         | 
| 68 | 
            -
            # ##
         | 
| 69 | 
            -
            #
         | 
| 70 | 
            -
            #
         | 
| 71 | 
            -
            # @contextlib.contextmanager
         | 
| 72 | 
            -
            # def gzip_io_codec(f: ta.IO, mode: str) -> ta.ContextManager[ta.IO]:
         | 
| 73 | 
            -
            #     with gzip.open(f, mode) as o:
         | 
| 74 | 
            -
            #         yield o
         | 
| @@ -1,82 +0,0 @@ | |
| 1 | 
            -
            import codecs
         | 
| 2 | 
            -
            import io
         | 
| 3 | 
            -
            import typing as ta
         | 
| 4 | 
            -
             | 
| 5 | 
            -
            from .... import check
         | 
| 6 | 
            -
            from .... import lang
         | 
| 7 | 
            -
            from ....lite.io import DelimitingBuffer
         | 
| 8 | 
            -
            from ..stream.build import JsonObjectBuilder
         | 
| 9 | 
            -
            from ..stream.lex import JsonStreamLexer
         | 
| 10 | 
            -
            from ..stream.parse import JsonStreamParser
         | 
| 11 | 
            -
            from ..stream.parse import JsonStreamParserEvent
         | 
| 12 | 
            -
            from .formats import Format
         | 
| 13 | 
            -
             | 
| 14 | 
            -
             | 
| 15 | 
            -
            ##
         | 
| 16 | 
            -
             | 
| 17 | 
            -
             | 
| 18 | 
            -
            class EagerParser:
         | 
| 19 | 
            -
                def __init__(self, fmt: Format) -> None:
         | 
| 20 | 
            -
                    super().__init__()
         | 
| 21 | 
            -
             | 
| 22 | 
            -
                    self._fmt = fmt
         | 
| 23 | 
            -
             | 
| 24 | 
            -
                def parse(self, f: ta.TextIO) -> ta.Generator[ta.Any, None, None]:
         | 
| 25 | 
            -
                    return self._fmt.load(f)
         | 
| 26 | 
            -
             | 
| 27 | 
            -
             | 
| 28 | 
            -
            ##
         | 
| 29 | 
            -
             | 
| 30 | 
            -
             | 
| 31 | 
            -
            class DelimitingParser:
         | 
| 32 | 
            -
                def __init__(
         | 
| 33 | 
            -
                        self,
         | 
| 34 | 
            -
                        fmt: Format,
         | 
| 35 | 
            -
                        *,
         | 
| 36 | 
            -
                        delimiters: ta.Iterable[int] = b'\n',
         | 
| 37 | 
            -
                ) -> None:
         | 
| 38 | 
            -
                    super().__init__()
         | 
| 39 | 
            -
             | 
| 40 | 
            -
                    self._fmt = fmt
         | 
| 41 | 
            -
             | 
| 42 | 
            -
                    self._db = DelimitingBuffer(delimiters)
         | 
| 43 | 
            -
             | 
| 44 | 
            -
                def parse(self, b: bytes) -> ta.Generator[ta.Any, None, None]:
         | 
| 45 | 
            -
                    for chunk in self._db.feed(b):
         | 
| 46 | 
            -
                        s = check.isinstance(chunk, bytes).decode('utf-8')
         | 
| 47 | 
            -
                        v = self._fmt.load(io.StringIO(s))
         | 
| 48 | 
            -
                        yield v
         | 
| 49 | 
            -
             | 
| 50 | 
            -
             | 
| 51 | 
            -
            ##
         | 
| 52 | 
            -
             | 
| 53 | 
            -
             | 
| 54 | 
            -
            class StreamBuilder(lang.ExitStacked):
         | 
| 55 | 
            -
                _builder: JsonObjectBuilder | None = None
         | 
| 56 | 
            -
             | 
| 57 | 
            -
                def __enter__(self) -> ta.Self:
         | 
| 58 | 
            -
                    super().__enter__()
         | 
| 59 | 
            -
                    self._builder = self._enter_context(JsonObjectBuilder())
         | 
| 60 | 
            -
                    return self
         | 
| 61 | 
            -
             | 
| 62 | 
            -
                def build(self, e: JsonStreamParserEvent) -> ta.Generator[ta.Any, None, None]:
         | 
| 63 | 
            -
                    yield from check.not_none(self._builder)(e)
         | 
| 64 | 
            -
             | 
| 65 | 
            -
             | 
| 66 | 
            -
            class StreamParser(lang.ExitStacked):
         | 
| 67 | 
            -
                _decoder: codecs.IncrementalDecoder
         | 
| 68 | 
            -
                _lex: JsonStreamLexer
         | 
| 69 | 
            -
                _parse: JsonStreamParser
         | 
| 70 | 
            -
             | 
| 71 | 
            -
                def __enter__(self) -> ta.Self:
         | 
| 72 | 
            -
                    super().__enter__()
         | 
| 73 | 
            -
                    self._decoder = codecs.getincrementaldecoder('utf-8')()
         | 
| 74 | 
            -
                    self._lex = self._enter_context(JsonStreamLexer())
         | 
| 75 | 
            -
                    self._parse = self._enter_context(JsonStreamParser())
         | 
| 76 | 
            -
                    return self
         | 
| 77 | 
            -
             | 
| 78 | 
            -
                def parse(self, b: bytes) -> ta.Generator[JsonStreamParserEvent, None, None]:
         | 
| 79 | 
            -
                    for s in self._decoder.decode(b, not b):
         | 
| 80 | 
            -
                        for c in s:
         | 
| 81 | 
            -
                            for t in self._lex(c):
         | 
| 82 | 
            -
                                yield from self._parse(t)
         | 
| @@ -1,48 +0,0 @@ | |
| 1 | 
            -
            import dataclasses as dc
         | 
| 2 | 
            -
            import typing as ta
         | 
| 3 | 
            -
             | 
| 4 | 
            -
            from .... import lang
         | 
| 5 | 
            -
             | 
| 6 | 
            -
             | 
| 7 | 
            -
            if ta.TYPE_CHECKING:
         | 
| 8 | 
            -
                from ....specs import jmespath
         | 
| 9 | 
            -
            else:
         | 
| 10 | 
            -
                jmespath = lang.proxy_import('....specs.jmespath', __package__)
         | 
| 11 | 
            -
             | 
| 12 | 
            -
             | 
| 13 | 
            -
            ##
         | 
| 14 | 
            -
             | 
| 15 | 
            -
             | 
| 16 | 
            -
            @dc.dataclass(frozen=True, kw_only=True)
         | 
| 17 | 
            -
            class ProcessingOptions:
         | 
| 18 | 
            -
                jmespath_expr: ta.Any | None = None
         | 
| 19 | 
            -
                flat: bool = False
         | 
| 20 | 
            -
             | 
| 21 | 
            -
             | 
| 22 | 
            -
            class Processor:
         | 
| 23 | 
            -
                def __init__(self, opts: ProcessingOptions) -> None:
         | 
| 24 | 
            -
                    super().__init__()
         | 
| 25 | 
            -
             | 
| 26 | 
            -
                    self._opts = opts
         | 
| 27 | 
            -
             | 
| 28 | 
            -
                    jmespath_expr = opts.jmespath_expr
         | 
| 29 | 
            -
                    if isinstance(jmespath_expr, str):
         | 
| 30 | 
            -
                        jmespath_expr = jmespath.compile(jmespath_expr)
         | 
| 31 | 
            -
                    self._jmespath_expr: ta.Any | None = jmespath_expr
         | 
| 32 | 
            -
             | 
| 33 | 
            -
                def process(self, v: ta.Any) -> ta.Iterable[ta.Any]:
         | 
| 34 | 
            -
                    if self._jmespath_expr is not None:
         | 
| 35 | 
            -
                        v = self._jmespath_expr.search(v)
         | 
| 36 | 
            -
             | 
| 37 | 
            -
                    if self._opts.flat:
         | 
| 38 | 
            -
                        if (
         | 
| 39 | 
            -
                                not isinstance(v, ta.Iterable) or  # noqa
         | 
| 40 | 
            -
                                isinstance(v, ta.Mapping) or
         | 
| 41 | 
            -
                                isinstance(v, lang.BUILTIN_SCALAR_ITERABLE_TYPES)
         | 
| 42 | 
            -
                        ):
         | 
| 43 | 
            -
                            raise TypeError(f'Flat output must be flat collections, got {type(v)}', v)
         | 
| 44 | 
            -
             | 
| 45 | 
            -
                        yield from v
         | 
| 46 | 
            -
             | 
| 47 | 
            -
                    else:
         | 
| 48 | 
            -
                        yield v
         | 
| @@ -1,92 +0,0 @@ | |
| 1 | 
            -
            import dataclasses as dc
         | 
| 2 | 
            -
            import json
         | 
| 3 | 
            -
            import typing as ta
         | 
| 4 | 
            -
             | 
| 5 | 
            -
            from .... import lang
         | 
| 6 | 
            -
            from .... import term
         | 
| 7 | 
            -
            from ..render import JsonRenderer
         | 
| 8 | 
            -
            from ..stream.parse import JsonStreamParserEvent
         | 
| 9 | 
            -
            from ..stream.render import StreamJsonRenderer
         | 
| 10 | 
            -
             | 
| 11 | 
            -
             | 
| 12 | 
            -
            ##
         | 
| 13 | 
            -
             | 
| 14 | 
            -
             | 
| 15 | 
            -
            @dc.dataclass(frozen=True, kw_only=True)
         | 
| 16 | 
            -
            class RenderingOptions:
         | 
| 17 | 
            -
                indent: int | str | None = None
         | 
| 18 | 
            -
                separators: tuple[str, str] | None = None
         | 
| 19 | 
            -
                sort_keys: bool = False
         | 
| 20 | 
            -
                raw: bool = False
         | 
| 21 | 
            -
                unicode: bool = False
         | 
| 22 | 
            -
                color: bool = False
         | 
| 23 | 
            -
             | 
| 24 | 
            -
             | 
| 25 | 
            -
            def make_render_kwargs(opts: RenderingOptions) -> ta.Mapping[str, ta.Any]:
         | 
| 26 | 
            -
                return dict(
         | 
| 27 | 
            -
                    indent=opts.indent,
         | 
| 28 | 
            -
                    separators=opts.separators,
         | 
| 29 | 
            -
                    sort_keys=opts.sort_keys,
         | 
| 30 | 
            -
                    ensure_ascii=not opts.unicode,
         | 
| 31 | 
            -
                )
         | 
| 32 | 
            -
             | 
| 33 | 
            -
             | 
| 34 | 
            -
            class Renderer(lang.Abstract):
         | 
| 35 | 
            -
                def __init__(self, opts: RenderingOptions) -> None:
         | 
| 36 | 
            -
                    super().__init__()
         | 
| 37 | 
            -
                    self._opts = opts
         | 
| 38 | 
            -
                    self._kw = make_render_kwargs(opts)
         | 
| 39 | 
            -
             | 
| 40 | 
            -
             | 
| 41 | 
            -
            ##
         | 
| 42 | 
            -
             | 
| 43 | 
            -
             | 
| 44 | 
            -
            def term_color(o: ta.Any, state: JsonRenderer.State) -> tuple[str, str]:
         | 
| 45 | 
            -
                if state is JsonRenderer.State.KEY:
         | 
| 46 | 
            -
                    return term.SGR(term.SGRs.FG.BRIGHT_BLUE), term.SGR(term.SGRs.RESET)
         | 
| 47 | 
            -
                elif isinstance(o, str):
         | 
| 48 | 
            -
                    return term.SGR(term.SGRs.FG.GREEN), term.SGR(term.SGRs.RESET)
         | 
| 49 | 
            -
                else:
         | 
| 50 | 
            -
                    return '', ''
         | 
| 51 | 
            -
             | 
| 52 | 
            -
             | 
| 53 | 
            -
            ##
         | 
| 54 | 
            -
             | 
| 55 | 
            -
             | 
| 56 | 
            -
            class EagerRenderer(Renderer):
         | 
| 57 | 
            -
                def render(self, v: ta.Any) -> str:
         | 
| 58 | 
            -
                    if self._opts.raw:
         | 
| 59 | 
            -
                        if not isinstance(v, str):
         | 
| 60 | 
            -
                            raise TypeError(f'Raw output must be strings, got {type(v)}', v)
         | 
| 61 | 
            -
             | 
| 62 | 
            -
                        return v
         | 
| 63 | 
            -
             | 
| 64 | 
            -
                    elif self._opts.color:
         | 
| 65 | 
            -
                        return JsonRenderer.render_str(
         | 
| 66 | 
            -
                            v,
         | 
| 67 | 
            -
                            **self._kw,
         | 
| 68 | 
            -
                            style=term_color,
         | 
| 69 | 
            -
                        )
         | 
| 70 | 
            -
             | 
| 71 | 
            -
                    else:
         | 
| 72 | 
            -
                        return json.dumps(
         | 
| 73 | 
            -
                            v,
         | 
| 74 | 
            -
                            **self._kw,
         | 
| 75 | 
            -
                        )
         | 
| 76 | 
            -
             | 
| 77 | 
            -
             | 
| 78 | 
            -
            ##
         | 
| 79 | 
            -
             | 
| 80 | 
            -
             | 
| 81 | 
            -
            class StreamRenderer(Renderer):
         | 
| 82 | 
            -
                def __init__(self, opts: RenderingOptions) -> None:
         | 
| 83 | 
            -
                    super().__init__(opts)
         | 
| 84 | 
            -
             | 
| 85 | 
            -
                    self._renderer = StreamJsonRenderer(
         | 
| 86 | 
            -
                        style=term_color if self._opts.color else None,
         | 
| 87 | 
            -
                        delimiter='\n',
         | 
| 88 | 
            -
                        **self._kw,
         | 
| 89 | 
            -
                    )
         | 
| 90 | 
            -
             | 
| 91 | 
            -
                def render(self, e: JsonStreamParserEvent) -> ta.Generator[str, None, None]:
         | 
| 92 | 
            -
                    return self._renderer.render((e,))
         | 
| 
            File without changes
         | 
| 
            File without changes
         | 
| 
            File without changes
         | 
    
        /omlish/io/{_abc.py → abc.py}
    RENAMED
    
    | 
            File without changes
         | 
    
        /omlish/logs/{_abc.py → abc.py}
    RENAMED
    
    | 
            File without changes
         | 
    
        /omlish/sql/{_abc.py → abc.py}
    RENAMED
    
    | 
            File without changes
         | 
| 
            File without changes
         | 
| 
            File without changes
         | 
| 
            File without changes
         | 
| 
            File without changes
         |