mirror of
https://github.com/CLIUtils/CLI11.git
synced 2025-05-09 16:13:54 +00:00
Adding optional, refactor single file
This commit is contained in:
parent
eed238364b
commit
7a7064df4e
@ -10,6 +10,8 @@
|
|||||||
|
|
||||||
#include "CLI/Macros.hpp"
|
#include "CLI/Macros.hpp"
|
||||||
|
|
||||||
|
#include "CLI/Optional.hpp"
|
||||||
|
|
||||||
#include "CLI/StringTools.hpp"
|
#include "CLI/StringTools.hpp"
|
||||||
|
|
||||||
#include "CLI/Error.hpp"
|
#include "CLI/Error.hpp"
|
||||||
|
41
include/CLI/Optional.hpp
Normal file
41
include/CLI/Optional.hpp
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
// Distributed under the 3-Clause BSD License. See accompanying
|
||||||
|
// file LICENSE or https://github.com/CLIUtils/CLI11 for details.
|
||||||
|
|
||||||
|
#include <sstream>
|
||||||
|
|
||||||
|
#include "CLI/Macros.hpp"
|
||||||
|
|
||||||
|
// [CLI11:verbatim]
|
||||||
|
#ifdef __has_include
|
||||||
|
#if defined(CLI11_CPP17) && __has_include(<optional>)
|
||||||
|
#include <optional>
|
||||||
|
#define CLI11_OPTIONAL
|
||||||
|
namespace CLI {
|
||||||
|
using std::experimental::optional;
|
||||||
|
} // namespace CLI
|
||||||
|
#elif defined(CPP11_CPP14) && __has_include(<experimental/optional>)
|
||||||
|
#include <experimental/optional>
|
||||||
|
#define CLI11_OPTIONAL
|
||||||
|
namespace CLI {
|
||||||
|
using std::optional;
|
||||||
|
} // namespace CLI
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
// [CLI11:verbatim]
|
||||||
|
|
||||||
|
namespace CLI {
|
||||||
|
|
||||||
|
#ifdef CLI11_OPTIONAL
|
||||||
|
|
||||||
|
template <typename T> std::istream &operator>>(std::istream &in, optional<T> &val) {
|
||||||
|
T v;
|
||||||
|
in >> v;
|
||||||
|
val = v;
|
||||||
|
return in;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
} // namespace CLI
|
@ -5,60 +5,110 @@ from __future__ import print_function, unicode_literals
|
|||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
import argparse
|
import argparse
|
||||||
|
import operator
|
||||||
|
from copy import copy
|
||||||
from subprocess import check_output, CalledProcessError
|
from subprocess import check_output, CalledProcessError
|
||||||
|
from functools import reduce
|
||||||
|
|
||||||
includes_local = re.compile(r"""^#include "(.*)"$""", re.MULTILINE)
|
includes_local = re.compile(r"""^#include "(.*)"$""", re.MULTILINE)
|
||||||
includes_system = re.compile(r"""^#include \<(.*)\>$""", re.MULTILINE)
|
includes_system = re.compile(r"""^#include \<(.*)\>$""", re.MULTILINE)
|
||||||
|
verbatim_tag_str = r"""
|
||||||
|
^ # Begin of line
|
||||||
|
[^\n^\[]+ # Some characters, not including [ or the end of a line
|
||||||
|
\[ # A literal [
|
||||||
|
[^\]^\n]* # Anything except a closing ]
|
||||||
|
CLI11:verbatim # The tag
|
||||||
|
[^\]^\n]* # Anything except a closing ]
|
||||||
|
\] # A literal ]
|
||||||
|
[^\n]* # Up to end of line
|
||||||
|
$ # End of a line
|
||||||
|
"""
|
||||||
|
verbatim_all = re.compile(verbatim_tag_str + "(.*)" + verbatim_tag_str,
|
||||||
|
re.MULTILINE | re.DOTALL | re.VERBOSE)
|
||||||
|
|
||||||
DIR = os.path.dirname(os.path.abspath(__file__)) # Path(__file__).resolve().parent
|
DIR = os.path.dirname(os.path.abspath(__file__))
|
||||||
BDIR = os.path.join(os.path.dirname(DIR), 'include') # DIR.parent / 'include'
|
|
||||||
|
|
||||||
print("Git directory:", DIR)
|
class HeaderFile(object):
|
||||||
|
TAG = "Unknown git revision"
|
||||||
|
|
||||||
try:
|
def __init__(self, base, inc):
|
||||||
TAG = check_output(['git', 'describe', '--tags', '--always'], cwd=str(DIR)).decode("utf-8")
|
with open(os.path.join(base, inc)) as f:
|
||||||
except CalledProcessError:
|
|
||||||
TAG = "A non-git source"
|
|
||||||
|
|
||||||
def MakeHeader(out):
|
|
||||||
main_header = os.path.join(BDIR, 'CLI', 'CLI.hpp')
|
|
||||||
with open(main_header) as f:
|
|
||||||
header = f.read()
|
|
||||||
|
|
||||||
include_files = includes_local.findall(header)
|
|
||||||
|
|
||||||
headers = set()
|
|
||||||
output = ''
|
|
||||||
for inc in include_files:
|
|
||||||
with open(os.path.join(BDIR, inc)) as f:
|
|
||||||
inner = f.read()
|
inner = f.read()
|
||||||
headers |= set(includes_system.findall(inner))
|
|
||||||
output += '\n// From {inc}\n\n'.format(inc=inc)
|
|
||||||
output += inner[inner.find('namespace'):]
|
|
||||||
|
|
||||||
header_list = '\n'.join('#include <'+h+'>' for h in headers)
|
# add self.verbatim
|
||||||
|
if 'CLI11:verbatim' in inner:
|
||||||
|
self.verbatim = ["\n\n// Verbatim copy from {}".format(inc)]
|
||||||
|
self.verbatim += verbatim_all.findall(inner)
|
||||||
|
inner = verbatim_all.sub("", inner)
|
||||||
|
else:
|
||||||
|
self.verbatim = []
|
||||||
|
|
||||||
output = '''\
|
self.headers = set(includes_system.findall(inner))
|
||||||
|
|
||||||
|
self.body = '\n// From {}\n\n'.format(inc) + inner[inner.find('namespace'):]
|
||||||
|
|
||||||
|
def __add__(self, other):
|
||||||
|
out = copy(self)
|
||||||
|
out.headers |= other.headers
|
||||||
|
out.body += other.body
|
||||||
|
out.verbatim += other.verbatim
|
||||||
|
return out
|
||||||
|
|
||||||
|
@property
|
||||||
|
def header_str(self):
|
||||||
|
return '\n'.join('#include <'+h+'>' for h in sorted(self.headers))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def verbatim_str(self):
|
||||||
|
return '\n'.join(self.verbatim)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return '''\
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
// Distributed under the 3-Clause BSD License. See accompanying
|
// Distributed under the 3-Clause BSD License. See accompanying
|
||||||
// file LICENSE or https://github.com/CLIUtils/CLI11 for details.
|
// file LICENSE or https://github.com/CLIUtils/CLI11 for details.
|
||||||
|
|
||||||
// This file was generated using MakeSingleHeader.py in CLI11/scripts
|
// This file was generated using MakeSingleHeader.py in CLI11/scripts
|
||||||
// from: {tag}
|
// from: {self.TAG}
|
||||||
// This has the complete CLI library in one file.
|
// This has the complete CLI library in one file.
|
||||||
|
|
||||||
{header_list}
|
{self.header_str}
|
||||||
{output}'''.format(header_list=header_list, output=output, tag=TAG)
|
{self.verbatim_str}
|
||||||
|
{self.body}
|
||||||
|
'''.format(self=self)
|
||||||
|
|
||||||
with open(out, 'w') as f:
|
|
||||||
f.write(output)
|
|
||||||
|
|
||||||
print("Created {out}".format(out=out))
|
def MakeHeader(output, main_header, include_dir = '../include'):
|
||||||
|
# Set tag if possible to class variable
|
||||||
|
try:
|
||||||
|
HeaderFile.TAG = check_output(['git', 'describe', '--tags', '--always'], cwd=str(DIR)).decode("utf-8")
|
||||||
|
except CalledProcessError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
base_dir = os.path.abspath(os.path.join(DIR, include_dir))
|
||||||
|
main_header = os.path.join(base_dir, main_header)
|
||||||
|
|
||||||
|
with open(main_header) as f:
|
||||||
|
header = f.read()
|
||||||
|
|
||||||
|
include_files = includes_local.findall(header)
|
||||||
|
|
||||||
|
headers = [HeaderFile(base_dir, inc) for inc in include_files]
|
||||||
|
single_header = reduce(operator.add, headers)
|
||||||
|
|
||||||
|
with open(output, 'w') as f:
|
||||||
|
f.write(str(single_header))
|
||||||
|
|
||||||
|
print("Created", output)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
parser.add_argument("output", nargs='?', default=os.path.join(BDIR, 'CLI11.hpp'))
|
parser.add_argument("output", help="Single header file output")
|
||||||
|
parser.add_argument("--main", default='CLI/CLI.hpp', help="The main include file that defines the other files")
|
||||||
|
parser.add_argument("--include", default='../include')
|
||||||
args = parser.parse_args()
|
args = parser.parse_args()
|
||||||
MakeHeader(args.output)
|
|
||||||
|
MakeHeader(args.output, args.main, args.include)
|
||||||
|
|
||||||
|
@ -1,28 +1,7 @@
|
|||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
#ifdef __has_include
|
#if CLI11_OPTIONAL
|
||||||
#if defined(CLI11_CPP17) && __has_include(<optional>)
|
|
||||||
#include <optional>
|
|
||||||
#define have_optional 1
|
|
||||||
using std::experimental::optional;
|
|
||||||
#elif defined(CPP11_CPP14) && __has_include(<experimental/optional>)
|
|
||||||
#include <experimental/optional>
|
|
||||||
#define have_optional 1
|
|
||||||
using std::optional;
|
|
||||||
#else
|
|
||||||
#define have_optional 0
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if have_optional
|
|
||||||
|
|
||||||
template <typename T> std::istream &operator>>(std::istream &in, optional<T> &val) {
|
|
||||||
T v;
|
|
||||||
in >> v;
|
|
||||||
val = v;
|
|
||||||
return in;
|
|
||||||
}
|
|
||||||
|
|
||||||
#include "app_helper.hpp"
|
#include "app_helper.hpp"
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user