Skip to content

Commit 3bd8def

Browse files
committed
config: Consolidate the version information into a single configuration file
1 parent 5416cfa commit 3bd8def

File tree

10 files changed

+258
-93
lines changed

10 files changed

+258
-93
lines changed

config/rtems-version.ini

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
#
2+
# RTEMS Tools Project (http://www.rtems.org/)
3+
#
4+
# RTEMS Version
5+
#
6+
7+
[version]
8+
revision = 5.0.not_released

rtemstoolkit/check.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -168,7 +168,7 @@ def run():
168168
try:
169169
_opts = options.command_line(argv = sys.argv)
170170
options.load(_opts)
171-
log.notice('RTEMS Source Builder - Check, v%s' % (version.str()))
171+
log.notice('RTEMS Source Builder - Check, v%s' % (version.string()))
172172
if host_setup(_opts):
173173
print('Environment is ok')
174174
else:

rtemstoolkit/options.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -289,7 +289,7 @@ def _help_indent(self):
289289

290290
def help(self):
291291
print('%s: [options] [args]' % (self.command_name))
292-
print('RTEMS Tools Project (c) 2012-2015 Chris Johns')
292+
print('RTEMS Tools Project, %s' % (version.string()))
293293
print('Options and arguments:')
294294
opts = list(self.long_opts_help.keys())
295295
if self.optargs:
@@ -559,7 +559,7 @@ def run(args):
559559
long_opts = long_opts,
560560
command_path = '.')
561561
load(opts)
562-
log.notice('RTEMS Tools Project - Defaults, v%s' % (version.str()))
562+
log.notice('RTEMS Tools Project - Defaults, v%s' % (version.string()))
563563
opts.log_info()
564564
log.notice('Options:')
565565
log.notice(str(opts))

rtemstoolkit/rtems.py

+51-8
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
#
22
# RTEMS Tools Project (http://www.rtems.org/)
3-
# Copyright 20162018 Chris Johns ([email protected])
3+
# Copyright 2016-2018 Chris Johns ([email protected])
44
# All rights reserved.
55
#
66
# This file is part of the RTEMS Tools package in 'rtems-tools'.
@@ -33,26 +33,69 @@
3333
import copy
3434
import os
3535
import re
36+
import sys
3637
import textwrap
3738

3839
from rtemstoolkit import configuration as configuration_
3940
from rtemstoolkit import error
41+
from rtemstoolkit import path
4042
from rtemstoolkit import textbox
41-
from rtemstoolkit import version
4243

44+
#
45+
# The default path we install RTEMS under
46+
#
47+
_prefix_path = '/opt/rtems'
48+
49+
def default_prefix():
50+
from rtemstoolkit import version
51+
return path.join(_prefix_path, version.version())
4352

4453
def clean_windows_path():
45-
#
46-
# On Windows MSYS2 prepends a path to itself to the environment
47-
# path. This means the RTEMS specific automake is not found and which
48-
# breaks the bootstrap. We need to remove the prepended path. Also
49-
# remove any ACLOCAL paths from the environment.
50-
#
54+
'''On Windows MSYS2 prepends a path to itself to the environment path. This
55+
means the RTEMS specific automake is not found and which breaks the
56+
bootstrap. We need to remove the prepended path. Also remove any ACLOCAL
57+
paths from the environment.
58+
59+
'''
5160
if os.name == 'nt':
5261
cspath = os.environ['PATH'].split(os.pathsep)
5362
if 'msys' in cspath[0] and cspath[0].endswith('bin'):
5463
os.environ['PATH'] = os.pathsep.join(cspath[1:])
5564

65+
def configuration_path():
66+
'''Return the path the configuration data path for RTEMS. The path is relative
67+
to the installed executable. Mangage the installed package and the in source
68+
tree when running from within the rtems-tools repo.
69+
70+
'''
71+
exec_name = os.path.abspath(sys.argv[0])
72+
for top in [os.path.dirname(exec_name),
73+
os.path.dirname(os.path.dirname(exec_name))]:
74+
config_path = path.join(top, 'share', 'rtems', 'config')
75+
if path.exists(config_path):
76+
break
77+
config_path = path.join(top, 'config')
78+
if path.exists(config_path):
79+
break
80+
config_path = None
81+
return config_path
82+
83+
def configuration_file(config):
84+
'''Return the path to a configuration file for RTEMS. The path is relative to
85+
the installed executable or we are testing and running from within the
86+
rtems-tools repo.
87+
88+
'''
89+
return path.join(configuration_path(), config)
90+
91+
def bsp_configuration_file():
92+
'''Return the path to the BSP configuration file for RTEMS. The path is
93+
relative to the installed executable or we are testing and running from
94+
within the rtems-tools repo.
95+
96+
'''
97+
return configuration_file('rtems-bsps.ini')
98+
5699
class configuration:
57100

58101
def __init__(self):

rtemstoolkit/version.py

+171-25
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
#
22
# RTEMS Tools Project (http://www.rtems.org/)
3-
# Copyright 2010-2016 Chris Johns ([email protected])
3+
# Copyright 2010-2018 Chris Johns ([email protected])
44
# All rights reserved.
55
#
66
# This file is part of the RTEMS Tools package in 'rtems-tools'.
@@ -29,83 +29,229 @@
2929
#
3030

3131
#
32-
# To release RTEMS Tools create a git archive and then add a suitable VERSION
33-
# file to the top directory.
32+
# Releasing RTEMS Tools
33+
# ---------------------
34+
#
35+
# Format:
36+
#
37+
# The format is INI. The file requires a `[version`] section and a `revision`
38+
# option:
39+
#
40+
# [version]
41+
# revision = <version-string>
42+
#
43+
# The `<version-string>` has the `version` and `revision` delimited by a
44+
# single `.`. An example file is:
45+
#
46+
# [version]
47+
# revision = 5.0.not_released
48+
#
49+
# where the `version` is `5` and the revision is `0` and the package is not
50+
# released. The label `not_released` is reversed to mean the package is not
51+
# released. A revision string can contain extra characters after the
52+
# `revision` number for example `5.0-rc1` or is deploying a package
53+
# `5.0-nasa-cfs`
54+
#
55+
# Packages can optionally add specialised sections to a version configuration
56+
# files. These can be accessed via the:
57+
#
58+
# load_release_settings: Return the items in a section
59+
# load_release_setting: Return an item from a section
60+
#
61+
# User deployment:
62+
#
63+
# Create a git archive and then add a suitable VERSION file to the top
64+
# directory of the package. The package assumes your python executable is
65+
# location in `bin` directory which is one below the top of the package's
66+
# install prefix.
67+
#
68+
# RTEMS Release:
69+
#
70+
# Set the values in the `rtems-version.ini` file. This is a shared file so
71+
# packages and encouraged to add specific settings to other configuration
72+
# files.
73+
#
74+
# Notes:
75+
#
76+
# This module uses os.apth for paths and assumes all paths are in the host
77+
# format.
3478
#
3579

3680
from __future__ import print_function
3781

82+
import itertools
83+
import os
3884
import sys
3985

86+
try:
87+
import configparser
88+
except ImportError:
89+
import ConfigParser as configparser
90+
4091
#
41-
# Support to handle use in a package and as a unit test.
92+
# Support to handle importing when installed in a package and as a unit test.
4293
# If there is a better way to let us know.
4394
#
4495
try:
4596
from . import error
4697
from . import git
47-
from . import path
98+
from . import rtems
4899
except (ValueError, SystemError):
49100
import error
50101
import git
51102
import path
103+
import rtems
52104

53105
#
54106
# Default to an internal string.
55107
#
56-
_version = '5'
108+
_version = 'undefined'
57109
_revision = 'not_released'
58110
_version_str = '%s.%s' % (_version, _revision)
59111
_released = False
60112
_git = False
113+
_is_loaded = False
114+
115+
def _top():
116+
top = os.path.dirname(sys.argv[0])
117+
if len(top) == 0:
118+
top = '.'
119+
return top
61120

62-
def _at():
63-
return path.dirname(__file__)
121+
def _load_released_version_config():
122+
'''Local worker to load a configuration file.'''
123+
top = _top()
124+
for ver in [os.path.join(top, 'VERSION'),
125+
os.path.join('..', 'VERSION'),
126+
rtems.configuration_file('rtems-version.ini')]:
127+
if os.path.exists(os.path.join(ver)):
128+
v = configparser.SafeConfigParser()
129+
try:
130+
v.read(ver)
131+
except Exception as e:
132+
raise error.general('Invalid version config format: %s: %s' % (ver,
133+
e))
134+
return ver, v
135+
return None, None
64136

65137
def _load_released_version():
138+
'''Load the release data if present. If not found the package is not released.
139+
140+
A release can be made by adding a file called `VERSION` to the top level
141+
directory of a package. This is useful for user deploying a package and
142+
making custom releases.
143+
144+
The RTEMS project reserves the `rtems-version.ini` file for it's
145+
releases. This is the base release and should not be touched by users
146+
deploying a package.
147+
148+
'''
149+
global _version
150+
global _revision
66151
global _released
67152
global _version_str
68-
at = _at()
69-
for ver in [at, path.join(at, '..')]:
70-
if path.exists(path.join(ver, 'VERSION')):
153+
global _is_loaded
154+
155+
if not _is_loaded:
156+
vc, v = _load_released_version_config()
157+
if v is not None:
71158
try:
72-
import configparser
73-
except ImportError:
74-
import ConfigParser as configparser
75-
v = configparser.SafeConfigParser()
76-
v.read(path.join(ver, 'VERSION'))
77-
_version_str = v.get('version', 'release')
78-
_released = True
159+
ver_str = v.get('version', 'revision')
160+
except Exception as e:
161+
raise error.general('Invalid version file: %s: %s' % (vc, e))
162+
ver_split = ver_str.split('.')
163+
if len(ver_split) < 2:
164+
raise error.general('Invalid version release value: %s: %s' % (vc,
165+
ver_str))
166+
ver = ver_split[0]
167+
rev = '.'.join(ver_split[1:])
168+
try:
169+
_version = int(ver)
170+
except:
171+
raise error.general('Invalid version config value: %s: %s' % (vc,
172+
ver))
173+
try:
174+
_revision = int(''.join(itertools.takewhile(str.isdigit, rev)))
175+
except Exception as e:
176+
raise error.general('Invalid revision config value: %s: %s: %s' % (vc,
177+
rev,
178+
e))
179+
if not 'not_released' in ver:
180+
_released = True
181+
_version_str = ver_str
182+
_is_loaded = True
79183
return _released
80184

81185
def _load_git_version():
186+
global _version
187+
global _revision
82188
global _git
83189
global _version_str
84-
repo = git.repo(_at())
190+
repo = git.repo(_top())
85191
if repo.valid():
86192
head = repo.head()
87193
if repo.dirty():
88-
modified = ' modified'
194+
modified = 'modified'
195+
sep = ' '
89196
else:
90197
modified = ''
91-
_version_str = '%s (%s%s)' % (_version, head[0:12], modified)
198+
sep = ''
199+
_revision = '%s-%s' % (head[0:12], modified)
200+
_version_str = '%s (%s%s%s)' % (_version, head[0:12], sep, modified)
92201
_git = True
93202
return _git
94203

204+
def load_release_settings(section, error = False):
205+
vc, v = _load_released_version_config()
206+
items = []
207+
if v is not None:
208+
try:
209+
items = v.items(section)
210+
except Exception as e:
211+
if not isinstance(error, bool):
212+
error(e)
213+
elif error:
214+
raise error.general('Invalid config section: %s: %s: %s' % (vc,
215+
section,
216+
e))
217+
return items
218+
219+
def load_release_setting(section, option, raw = False, error = False):
220+
vc, v = _load_released_version_config()
221+
value = None
222+
if v is not None:
223+
try:
224+
value = v.get(section, option, raw = raw)
225+
except Exception as e:
226+
if not isinstance(error, bool):
227+
error(e)
228+
elif error:
229+
raise error.general('Invalid config section: %s: %s: %s.%s' % (vc,
230+
section,
231+
option,
232+
e))
233+
return value
234+
95235
def released():
96236
return _load_released_version()
97237

98238
def version_control():
99239
return _load_git_version()
100240

101-
def str():
102-
if not _released and not _git:
103-
if not _load_released_version():
104-
_load_git_version()
241+
def string():
242+
_load_released_version()
243+
_load_git_version()
105244
return _version_str
106245

107246
def version():
247+
_load_released_version()
248+
_load_git_version()
108249
return _version
109250

251+
def revision():
252+
_load_released_version()
253+
_load_git_version()
254+
return _revision
255+
110256
if __name__ == '__main__':
111257
print('Version: %s' % (str()))

0 commit comments

Comments
 (0)