1
0
mirror of https://github.com/CLIUtils/CLI11.git synced 2025-05-07 15:33:51 +00:00

Adding optional, refactor single file

This commit is contained in:
Henry Fredrick Schreiner 2018-04-03 21:37:40 +02:00 committed by Henry Schreiner
parent eed238364b
commit 7a7064df4e
4 changed files with 126 additions and 54 deletions

View File

@ -10,6 +10,8 @@
#include "CLI/Macros.hpp"
#include "CLI/Optional.hpp"
#include "CLI/StringTools.hpp"
#include "CLI/Error.hpp"

41
include/CLI/Optional.hpp Normal file
View 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

View File

@ -5,60 +5,110 @@ from __future__ import print_function, unicode_literals
import os
import re
import argparse
import operator
from copy import copy
from subprocess import check_output, CalledProcessError
from functools import reduce
includes_local = 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
BDIR = os.path.join(os.path.dirname(DIR), 'include') # DIR.parent / 'include'
DIR = os.path.dirname(os.path.abspath(__file__))
print("Git directory:", DIR)
class HeaderFile(object):
TAG = "Unknown git revision"
try:
TAG = check_output(['git', 'describe', '--tags', '--always'], cwd=str(DIR)).decode("utf-8")
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:
def __init__(self, base, inc):
with open(os.path.join(base, inc)) as f:
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
// Distributed under the 3-Clause BSD License. See accompanying
// file LICENSE or https://github.com/CLIUtils/CLI11 for details.
// This file was generated using MakeSingleHeader.py in CLI11/scripts
// from: {tag}
// from: {self.TAG}
// This has the complete CLI library in one file.
{header_list}
{output}'''.format(header_list=header_list, output=output, tag=TAG)
{self.header_str}
{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__':
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()
MakeHeader(args.output)
MakeHeader(args.output, args.main, args.include)

View File

@ -1,28 +1,7 @@
#include <cstdlib>
#include <iostream>
#ifdef __has_include
#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;
}
#if CLI11_OPTIONAL
#include "app_helper.hpp"