forked from exaloop/codon
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathgetopt.codon
More file actions
146 lines (134 loc) · 5.91 KB
/
getopt.codon
File metadata and controls
146 lines (134 loc) · 5.91 KB
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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
# Copyright (C) 2022-2026 Exaloop Inc. <https://exaloop.io>
# Parts of this file: (c) 2022 Python Software Foundation. All right reserved.
# License:
# 1. This LICENSE AGREEMENT is between the Python Software Foundation ("PSF"), and
# the Individual or Organization ("Licensee") accessing and otherwise using Python
# 3.10.2 software in source or binary form and its associated documentation.
#
# 2. Subject to the terms and conditions of this License Agreement, PSF hereby
# grants Licensee a nonexclusive, royalty-free, world-wide license to reproduce,
# analyze, test, perform and/or display publicly, prepare derivative works,
# distribute, and otherwise use Python 3.10.2 alone or in any derivative
# version, provided, however, that PSF's License Agreement and PSF's notice of
# copyright, i.e., "Copyright © 2001-2022 Python Software Foundation; All Rights
# Reserved" are retained in Python 3.10.2 alone or in any derivative version
# prepared by Licensee.
#
# 3. In the event Licensee prepares a derivative work that is based on or
# incorporates Python 3.10.2 or any part thereof, and wants to make the
# derivative work available to others as provided herein, then Licensee hereby
# agrees to include in any such work a brief summary of the changes made to Python
# 3.10.2.
#
# 4. PSF is making Python 3.10.2 available to Licensee on an "AS IS" basis.
# PSF MAKES NO REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED. BY WAY OF
# EXAMPLE, BUT NOT LIMITATION, PSF MAKES NO AND DISCLAIMS ANY REPRESENTATION OR
# WARRANTY OF MERCHANTABILITY OR FITNESS FOR ANY PARTICULAR PURPOSE OR THAT THE
# USE OF PYTHON 3.10.2 WILL NOT INFRINGE ANY THIRD PARTY RIGHTS.
#
# 5. PSF SHALL NOT BE LIABLE TO LICENSEE OR ANY OTHER USERS OF PYTHON 3.10.2
# FOR ANY INCIDENTAL, SPECIAL, OR CONSEQUENTIAL DAMAGES OR LOSS AS A RESULT OF
# MODIFYING, DISTRIBUTING, OR OTHERWISE USING PYTHON 3.10.2, OR ANY DERIVATIVE
# THEREOF, EVEN IF ADVISED OF THE POSSIBILITY THEREOF.
#
# 6. This License Agreement will automatically terminate upon a material breach of
# its terms and conditions.
#
# 7. Nothing in this License Agreement shall be deemed to create any relationship
# of agency, partnership, or joint venture between PSF and Licensee. This License
# Agreement does not grant permission to use PSF trademarks or trade name in a
# trademark sense to endorse or promote products or services of Licensee, or any
# third party.
#
# 8. By copying, installing or otherwise using Python 3.10.2, Licensee agrees
# to be bound by the terms and conditions of this License Agreement.
import os
class GetoptError(Exception):
def __init__(self, message: str = ""):
super().__init__(message)
def long_has_args(opt: str, longopts: List[str]) -> Tuple[bool, str]:
possibilities = [o for o in longopts if o.startswith(opt)]
if not possibilities:
raise GetoptError(f"option --{opt} not recognized")
# Is there an exact match?
if opt in possibilities:
return False, opt
elif opt + "=" in possibilities:
return True, opt
# No exact match, so better be unique.
if len(possibilities) > 1:
# XXX since possibilities contains all valid continuations, might be
# nice to work them into the error msg
raise GetoptError(f"option --{opt} not a unique prefix")
assert len(possibilities) == 1
unique_match = possibilities[0]
has_arg = unique_match.endswith("=")
if has_arg:
unique_match = unique_match[:-1]
return has_arg, unique_match
def do_longs(
opts: List[Tuple[str, str]], opt: str, longopts: List[str], args: List[str]
) -> Tuple[List[Tuple[str, str]], List[str]]:
optarg = ""
try:
i = opt.index("=")
opt, optarg = opt[:i], opt[i + 1 :]
except ValueError:
pass
has_arg, opt = long_has_args(opt, longopts)
if has_arg:
if optarg == "":
if not args:
raise GetoptError(f"option --{opt} requires argument")
optarg, args = args[0], args[1:]
elif optarg != "":
raise GetoptError(f"option --{opt} must not have an argument")
opts.append((f"--{opt}", optarg))
return opts, args
def short_has_arg(opt: str, shortopts: str) -> bool:
for i in range(len(shortopts)):
if opt == shortopts[i] != ":":
return shortopts.startswith(":", i + 1)
raise GetoptError(f"option -{opt} not recognized")
def do_shorts(
opts: List[Tuple[str, str]], optstring: str, shortopts: str, args: List[str]
) -> Tuple[List[Tuple[str, str]], List[str]]:
while optstring != "":
opt, optstring = optstring[0], optstring[1:]
optarg = ""
if short_has_arg(opt, shortopts):
if optstring == "":
if not args:
raise GetoptError(f"option -{opt} requires argument")
optstring, args = args[0], args[1:]
optarg, optstring = optstring, ""
opts.append((f"-{opt}", optarg))
return opts, args
def getopt(
args: List[str], shortopts: str, longopts: List[str] = []
) -> Tuple[List[Tuple[str, str]], List[str]]:
opts = []
prog_args = []
# Allow options after non-option arguments?
all_options_first = False
if shortopts.startswith("+"):
shortopts = shortopts[1:]
all_options_first = True
elif "POSIXLY_CORRECT" in os.environ:
all_options_first = True
while args:
if args[0] == "--":
prog_args += args[1:]
break
if args[0][:2] == "--":
opts, args = do_longs(opts, args[0][2:], longopts, args[1:])
elif args[0][:1] == "-" and args[0] != "-":
opts, args = do_shorts(opts, args[0][1:], shortopts, args[1:])
else:
if all_options_first:
prog_args += args
break
else:
prog_args.append(args[0])
args = args[1:]
return opts, prog_args