Add help text/docs for generate-flag-types script
This commit is contained in:
parent
0d01711bcf
commit
3a8ad862a6
2
cli.go
2
cli.go
@ -18,4 +18,4 @@
|
|||||||
// }
|
// }
|
||||||
package cli
|
package cli
|
||||||
|
|
||||||
//go:generate python ./generate-flag-types flag_generated.go flag-types.json
|
//go:generate python ./generate-flag-types -i flag-types.json -o flag_generated.go
|
||||||
|
@ -1,44 +1,106 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
|
"""
|
||||||
|
The flag types that ship with the cli library have many things in common, and
|
||||||
|
so we can take advantage of the `go generate` command to create much of the
|
||||||
|
source code from a list of definitions. These definitions attempt to cover
|
||||||
|
the parts that vary between flag types, and should evolve as needed.
|
||||||
|
|
||||||
|
An example of the minimum definition needed is:
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "SomeType",
|
||||||
|
"type": "sometype",
|
||||||
|
"context_default": "nil"
|
||||||
|
}
|
||||||
|
|
||||||
|
In this example, the generated code will include a type named `SomeTypeFlag`
|
||||||
|
that is expected to wrap a value of type `sometype`. Fetching values by name
|
||||||
|
via `*cli.Context` will default to a value of `nil`.
|
||||||
|
|
||||||
|
A more complete, albeit somewhat redundant, example showing all available
|
||||||
|
definition keys is:
|
||||||
|
|
||||||
|
{
|
||||||
|
"name": "VeryMuchType",
|
||||||
|
"type": "*VeryMuchType",
|
||||||
|
"value": true,
|
||||||
|
"dest": false,
|
||||||
|
"doctail": " which really only wraps a []float64, oh well!",
|
||||||
|
"context_type": "[]float64",
|
||||||
|
"context_default": "nil"
|
||||||
|
}
|
||||||
|
|
||||||
|
The meaning of each field is as follows:
|
||||||
|
|
||||||
|
name (string) - The type "name", which will be suffixed with
|
||||||
|
`Flag` when generating the type definition
|
||||||
|
type (string) - The type that the generated `Flag` type is
|
||||||
|
expected to "contain" as its `.Value` member
|
||||||
|
value (bool) - Should the generated type have a `Value` member?
|
||||||
|
dest (bool) - Should the generated type support a destination
|
||||||
|
pointer?
|
||||||
|
doctail (string) - Additional docs for the flag type comment
|
||||||
|
context_type (string) - The literal type used in the `*cli.Context`
|
||||||
|
reader func signature
|
||||||
|
context_default (string) - The literal value used as the default by the
|
||||||
|
`*cli.Context` reader funcs when no value is
|
||||||
|
present
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
from __future__ import print_function, unicode_literals
|
from __future__ import print_function, unicode_literals
|
||||||
|
|
||||||
import io
|
import argparse
|
||||||
import json
|
import json
|
||||||
import os
|
import os
|
||||||
import subprocess
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
|
import tempfile
|
||||||
import textwrap
|
import textwrap
|
||||||
|
|
||||||
|
|
||||||
|
class _FancyFormatter(argparse.ArgumentDefaultsHelpFormatter,
|
||||||
|
argparse.RawDescriptionHelpFormatter):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
def main(sysargs=sys.argv[:]):
|
def main(sysargs=sys.argv[:]):
|
||||||
_generate_flag_types(sys.argv[1], sys.argv[2])
|
parser = argparse.ArgumentParser(
|
||||||
|
description='Generate flag type code!',
|
||||||
|
formatter_class=_FancyFormatter)
|
||||||
|
parser.add_argument(
|
||||||
|
'-i', '--in-json',
|
||||||
|
type=argparse.FileType('r'),
|
||||||
|
default=sys.stdin,
|
||||||
|
help='Input JSON file which defines each type to be generated'
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'-o', '--out-go',
|
||||||
|
type=argparse.FileType('w'),
|
||||||
|
default=sys.stdout,
|
||||||
|
help='Output file/stream to which generated source will be written'
|
||||||
|
)
|
||||||
|
parser.epilog = __doc__
|
||||||
|
|
||||||
|
args = parser.parse_args(sysargs[1:])
|
||||||
|
_generate_flag_types(args.out_go, args.in_json)
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
|
|
||||||
def _generate_flag_types(output_filename, types_filename):
|
def _generate_flag_types(output_go, input_json):
|
||||||
try:
|
types = json.load(input_json)
|
||||||
types = _load_types(types_filename)
|
|
||||||
if os.path.exists(output_filename):
|
|
||||||
os.chmod(output_filename, 0644)
|
|
||||||
|
|
||||||
with io.open(output_filename, 'w', encoding='utf-8') as outfile:
|
tmp = tempfile.NamedTemporaryFile(suffix='.go', delete=False)
|
||||||
_write_flag_types(outfile, types)
|
_write_flag_types(tmp, types)
|
||||||
|
tmp.close()
|
||||||
|
|
||||||
new_content = subprocess.check_output(
|
new_content = subprocess.check_output(
|
||||||
['goimports', output_filename]
|
['goimports', tmp.name]
|
||||||
).decode('utf-8')
|
).decode('utf-8')
|
||||||
|
|
||||||
with io.open(output_filename, 'w', encoding='utf-8') as outfile:
|
print(new_content, file=output_go, end='')
|
||||||
print(new_content, file=outfile, end='')
|
output_go.flush()
|
||||||
|
os.remove(tmp.name)
|
||||||
finally:
|
|
||||||
if os.path.exists(output_filename):
|
|
||||||
os.chmod(output_filename, 0444)
|
|
||||||
|
|
||||||
|
|
||||||
def _load_types(types_filename):
|
|
||||||
with io.open(types_filename, encoding='utf-8') as infile:
|
|
||||||
return json.load(infile)
|
|
||||||
|
|
||||||
|
|
||||||
def _write_flag_types(outfile, types):
|
def _write_flag_types(outfile, types):
|
||||||
@ -52,31 +114,29 @@ def _write_flag_types(outfile, types):
|
|||||||
for typedef in types:
|
for typedef in types:
|
||||||
typedef.setdefault('doctail', '')
|
typedef.setdefault('doctail', '')
|
||||||
typedef.setdefault('context_type', typedef['type'])
|
typedef.setdefault('context_type', typedef['type'])
|
||||||
typedef.setdefault('struct', True)
|
|
||||||
typedef.setdefault('dest', True)
|
typedef.setdefault('dest', True)
|
||||||
typedef.setdefault('value', True)
|
typedef.setdefault('value', True)
|
||||||
|
|
||||||
if typedef['struct']:
|
_fwrite(outfile, """\
|
||||||
|
// {name}Flag is a flag with type {type}{doctail}
|
||||||
|
type {name}Flag struct {{
|
||||||
|
Name string
|
||||||
|
Usage string
|
||||||
|
EnvVar string
|
||||||
|
Hidden bool
|
||||||
|
""".format(**typedef))
|
||||||
|
|
||||||
|
if typedef['value']:
|
||||||
_fwrite(outfile, """\
|
_fwrite(outfile, """\
|
||||||
// {name}Flag is a flag with type {type}{doctail}
|
Value {type}
|
||||||
type {name}Flag struct {{
|
|
||||||
Name string
|
|
||||||
Usage string
|
|
||||||
EnvVar string
|
|
||||||
Hidden bool
|
|
||||||
""".format(**typedef))
|
""".format(**typedef))
|
||||||
|
|
||||||
if typedef['value']:
|
if typedef['dest']:
|
||||||
_fwrite(outfile, """\
|
_fwrite(outfile, """\
|
||||||
Value {type}
|
Destination *{type}
|
||||||
""".format(**typedef))
|
""".format(**typedef))
|
||||||
|
|
||||||
if typedef['dest']:
|
_fwrite(outfile, "\n}\n\n")
|
||||||
_fwrite(outfile, """\
|
|
||||||
Destination *{type}
|
|
||||||
""".format(**typedef))
|
|
||||||
|
|
||||||
_fwrite(outfile, "\n}\n\n")
|
|
||||||
|
|
||||||
_fwrite(outfile, """\
|
_fwrite(outfile, """\
|
||||||
// String returns a readable representation of this value
|
// String returns a readable representation of this value
|
||||||
|
4
runtests
4
runtests
@ -66,7 +66,7 @@ def _vet():
|
|||||||
|
|
||||||
def _toc():
|
def _toc():
|
||||||
_run('node_modules/.bin/markdown-toc -i README.md')
|
_run('node_modules/.bin/markdown-toc -i README.md')
|
||||||
_run('git diff --quiet')
|
_run('git diff --exit-code')
|
||||||
|
|
||||||
|
|
||||||
def _gen():
|
def _gen():
|
||||||
@ -76,7 +76,7 @@ def _gen():
|
|||||||
return
|
return
|
||||||
|
|
||||||
_run('go generate .')
|
_run('go generate .')
|
||||||
_run('git diff --quiet')
|
_run('git diff --exit-code')
|
||||||
|
|
||||||
|
|
||||||
def _run(command):
|
def _run(command):
|
||||||
|
Loading…
Reference in New Issue
Block a user