7 Author : Valentin Kuznetsov <vkuznet@gmail.com>
8 Description: AbstractGenerator class provides basic functionality
9 to generate CMSSW class from given template
11 from __future__
import print_function
20 from FWCore.Skeletons.utils
import parse_word, functor, user_info, tree, template_directory
24 AbstractPkg takes care how to generate code from template/PKG
25 package area. The PKG can be any directory which may include
26 any types of files, e.g. C++ (.cc), python (.py), etc.
27 This class relies on specific logic which we outline here:
29 - each template may use tags defined with double underscores
30 enclosure, e.g. __class__, __record__, etc.
31 - each template may have example tags, such tags should
32 start with @example_. While processing template user may
33 choose to strip them off or keep the code behind those tags
34 - in addition user may specify pure python code which can
35 operate with user defined tags. This code snipped should
36 be enclosed with #python_begin and #python_end lines
37 which declares start and end of python block
50 self.
date = time.strftime(
"%a, %d %b %Y %H:%M:%S GMT", time.gmtime())
55 "Scan template files and return example tags"
57 sdir =
'%s/%s' % (self.
tdir, self.
tmpl)
58 for name
in os.listdir(sdir):
63 fname = os.path.join(sdir, name)
64 with open(fname,
'r')
as stream:
65 for line
in stream.readlines():
66 if line.find(
'@example_') != -1:
67 keys += [k
for k
in line.split()
if \
68 k.find(
'@example_') != -1]
72 "Print out template example tags"
77 "Scan template files and return template tags"
79 sdir =
'%s/%s' % (self.
tdir, self.
tmpl)
80 for name
in os.listdir(sdir):
85 fname = os.path.join(sdir, name)
86 with open(fname,
'r')
as stream:
87 for line
in stream.readlines():
88 if line.find(
'__') != -1:
93 "Print out template keys"
99 Determine either skip or keep given line based on class tags
103 keep_etags = self.
config.get(
'tmpl_etags', [])
104 for tag
in tmpl_etags:
105 for valid_tag
in keep_etags:
106 if line.find(valid_tag) != -1:
107 line = line.replace(valid_tag,
'')
109 if line.find(tag) != -1:
114 def write(self, fname, tmpl_name, kwds):
115 "Create new file from given template name and set of arguments"
118 if os.path.exists(fname):
120 with open(fname,
'w')
as stream:
121 for line
in open(tmpl_name,
'r').readlines():
125 if line.find(
'#python_begin') != -1:
128 if line.find(
'#python_end') != -1:
132 if code
and not read_code:
138 for key, val
in kwds.items():
139 if isinstance(val, str):
140 line = line.replace(key, val)
144 "Return keyword arguments to be used in methods"
145 kwds = {
'__pkgname__': self.
config.get(
'pkgname',
'Package'),
146 '__author__': self.
author,
147 '__date__': self.
date,
148 '__class__': self.
pname,
149 '__class_lowercase__': self.
pname.lower(),
150 '__name__': self.
pname,
151 '__subsys__': self.
config.get(
'subsystem',
'Subsystem')}
152 args = self.
config.get(
'args',
None)
155 print(
"Template tags:")
160 "Generate package templates in a given directory"
166 tmpl_files = self.
config.get(
'tmpl_files',
'all')
173 if os.path.isdir(self.
pname):
174 msg =
"Can't create package '%s'\n" % self.
pname
175 msg +=
"Directory %s is already exists" % self.
pname
178 os.makedirs(self.
pname)
182 sdir = os.path.join(self.
tdir, self.
tmpl)
183 sources = [s
for s
in os.listdir(sdir) \
184 if s !=
'Driver.dir' and s.find(
'~') == -1]
185 driver = os.path.join(sdir,
'Driver.dir')
186 if os.path.isfile(driver):
187 sources = [s.replace(
'\n',
'')
for s
in open(driver,
'r').readlines()]
189 sources.remove(
'CVS')
193 names = set([s.split(
'.')[0]
for s
in sources])
194 if names == set([
'Skeleton']):
196 _, ext = os.path.splitext(self.
pname)
197 sources = [s
for s
in sources
if s.rfind(ext) != -1]
201 msg =
'Unable to find skeleton for extension "%s"' % ext
204 bdir = os.environ.get(
'CMSSW_BASE',
'')
206 ldir = os.getcwd().
split(
'/')[-1]
208 subsys = kwds[
'__subsys__']
209 pkgname = kwds[
'__pkgname__']
210 if sources == [
'Skeleton.cc',
'Skeleton.h']:
211 if ldir ==
'interface' and os.getcwd().
find(bdir) != -1:
212 idir =
'%s/%s/interface/' % (subsys, pkgname)
215 elif sources == [
'Skeleton.cc']
and \
216 len(dirs) == 5
and dirs[0] ==
'' and dirs[1] ==
'src':
217 idir =
'%s/%s/interface/' % (subsys, pkgname)
218 elif sources == [
'Skeleton.h']
and ldir ==
'interface' and \
219 len(dirs) == 5
and dirs[0] ==
'' and dirs[1] ==
'src':
220 idir =
'%s/%s/interface/' % (subsys, pkgname)
221 kwds.update({
'__incdir__': idir})
227 if tmpl_files !=
'all':
228 fname, ext = os.path.splitext(src)
229 if tmpl_files != ext:
234 src = src.split(
'/')[-1]
237 items = src.split(
'/')
241 tmpl_name = os.path.join(sdir, items[-1])
242 if os.path.isfile(tmpl_name):
247 if items[-1] ==
'testBuildFile.xml':
248 name2gen =
'/'.
join(src.split(
'/')[:-1])+
'/BuildFile.xml'
249 if -1 !=tname.split(
'.')[0].
find(self.
tmpl):
250 name2gen = name2gen.replace(self.
tmpl, self.
pname)
251 name2gen = os.path.join(os.getcwd(), name2gen)
253 print(
"Create", name2gen)
255 if not os.path.isdir(name2gen):
256 os.makedirs(name2gen)
258 fdir = os.path.dirname(name2gen)
259 if not os.path.isdir(fdir):
261 self.
write(name2gen, tmpl_name, kwds)
262 gen_files.append(name2gen.split(
'/')[-1])
264 msg =
'New package "%s" of %s type is successfully generated' \
267 msg =
'Generated %s file' %
', '.
join(gen_files)
268 if len(gen_files) > 1:
273 if msg.find(
'New package') != -1: