1 package org
.argeo
.slc
.rpmfactory
.core
;
4 import java
.io
.IOException
;
5 import java
.text
.DateFormat
;
6 import java
.text
.SimpleDateFormat
;
7 import java
.util
.ArrayList
;
8 import java
.util
.Calendar
;
9 import java
.util
.GregorianCalendar
;
10 import java
.util
.HashMap
;
11 import java
.util
.List
;
14 import javax
.jcr
.Node
;
15 import javax
.jcr
.Repository
;
16 import javax
.jcr
.RepositoryException
;
17 import javax
.jcr
.Session
;
19 import org
.apache
.commons
.io
.FileUtils
;
20 import org
.argeo
.api
.cms
.CmsLog
;
21 import org
.argeo
.jcr
.JcrUtils
;
22 import org
.argeo
.slc
.SlcConstants
;
23 import org
.argeo
.slc
.SlcException
;
24 import org
.argeo
.slc
.repo
.NodeIndexerVisitor
;
25 import org
.argeo
.slc
.rpmfactory
.RpmFactory
;
26 import org
.argeo
.slc
.rpmfactory
.RpmRepository
;
27 import org
.argeo
.slc
.runtime
.tasks
.SystemCall
;
30 * Defines a build environment. This information is typically used by other
31 * components performing the various actions related to RPM build.
33 public class RpmFactoryImpl
implements RpmFactory
{
34 private CmsLog log
= CmsLog
.getLog(RpmFactoryImpl
.class);
36 private Repository rpmRepository
;
37 private Repository distRepository
;
40 private List
<RpmRepository
> repositories
= new ArrayList
<RpmRepository
>();
41 private List
<String
> archs
= new ArrayList
<String
>();
43 private String rpmBase
= "/mnt/slc/repos/rpm";
44 private String distBase
= "/mnt/slc/repos/dist";
45 private String mockVar
= "/var/lib/mock";
46 private String mockEtc
= "/etc/mock";
48 private DateFormat dateFormat
= new SimpleDateFormat("yyyyMMdd_HHmm");
50 private String gitWorkspace
= "git";
52 private String localUrlBase
= "http://localhost:7070/";
53 /** If not null or empty, this is a developer instance. */
54 private String gitDevBaseUrl
= null;
56 private Boolean withTestingRepository
= false;
58 private String yumConfigMainSection
= "cachedir=/var/cache/yum\n"
59 + "debuglevel=1\n" + "reposdir=/dev/null\n"
60 + "logfile=/var/log/yum.log\n" + "retries=20\n" + "obsoletes=1\n"
61 + "gpgcheck=0\n" + "assumeyes=1\n" + "syslog_ident=mock\n"
62 + "syslog_device=\n" + "http_caching=none\n";
64 private String defaultMacroFiles
= "/usr/lib/rpm/macros:"
65 + "/usr/lib/rpm/ia32e-linux/macros:"
66 + "/usr/lib/rpm/redhat/macros:" + "/etc/rpm/macros.*:"
67 + "/etc/rpm/macros:" + "/etc/rpm/ia32e-linux/macros:"
69 private Map
<String
, String
> rpmmacros
= new HashMap
<String
, String
>();
72 private String proxiedReposBase
;
73 private String managedReposBase
;
75 private String stagingWorkspace
;
76 private String testingWorkspace
;
77 private String stableWorkspace
;
79 private File rpmFactoryBaseDir
;
80 private File mockConfDir
;
81 private File yumConfDir
;
85 proxiedReposBase
= localUrlBase
+ "repo/rpm/";
86 managedReposBase
= localUrlBase
+ "data/public/rpm/";
89 rpmFactoryBaseDir
.mkdirs();
90 mockConfDir
= new File(rpmFactoryBaseDir
.getPath() + "/conf/mock");
92 yumConfDir
= new File(rpmFactoryBaseDir
.getPath() + "/conf/yum");
95 // managed repositories
96 stagingWorkspace
= id
+ "-staging";
97 if (withTestingRepository
)
98 testingWorkspace
= id
+ "-testing";
101 initDistWorkspace(stableWorkspace
);
103 initRpmWorkspace(stagingWorkspace
);
104 if (withTestingRepository
)
105 initRpmWorkspace(testingWorkspace
);
106 initRpmWorkspace(stableWorkspace
);
109 protected void initRpmWorkspace(String workspace
) {
110 Session session
= null;
112 session
= JcrUtils
.loginOrCreateWorkspace(rpmRepository
, workspace
);
113 JcrUtils
.addPrivilege(session
, "/", "anonymous", "jcr:read");
114 JcrUtils
.addPrivilege(session
, "/", SlcConstants
.ROLE_SLC
,
117 for (String arch
: archs
) {
118 Node archFolder
= JcrUtils
.mkfolders(session
, "/" + arch
);
120 File workspaceDir
= getWorkspaceDir(workspace
);
122 if (!archFolder
.hasNode("repodata")) {
123 // touch a file in order to make sure this is properly
125 File touch
= new File(workspaceDir
, ".touch");
126 touch
.createNewFile();
129 SystemCall createrepo
= new SystemCall();
130 createrepo
.arg("createrepo");
131 createrepo
.arg("-q");
132 File archDir
= new File(workspaceDir
, arch
);
133 createrepo
.arg(archDir
.getAbsolutePath());
136 } catch (IOException e
) {
137 log
.error(workspaceDir
+ " not properly mounted.", e
);
140 } catch (Exception e
) {
141 throw new SlcException("Cannot initialize workspace " + workspace
,
144 JcrUtils
.logoutQuietly(session
);
148 /** Caller must logout the underlying session. */
149 public Node
newDistribution(String distributionId
) {
150 Session session
= null;
152 session
= JcrUtils
.loginOrCreateWorkspace(rpmRepository
,
154 JcrUtils
.addPrivilege(session
, "/", "anonymous", "jcr:read");
155 JcrUtils
.addPrivilege(session
, "/", SlcConstants
.ROLE_SLC
,
158 Calendar now
= new GregorianCalendar();
159 String folderName
= dateFormat
.format(now
.getTime());
160 return JcrUtils
.mkfolders(session
, "/" + folderName
);
161 } catch (Exception e
) {
162 JcrUtils
.logoutQuietly(session
);
163 throw new SlcException("Cannot initialize distribution workspace "
164 + distributionId
, e
);
168 protected void initGitWorkspace() {
169 Session session
= null;
171 session
= JcrUtils
.loginOrCreateWorkspace(rpmRepository
,
173 JcrUtils
.addPrivilege(session
, "/", "anonymous", "jcr:read");
174 JcrUtils
.addPrivilege(session
, "/", SlcConstants
.ROLE_SLC
,
176 } catch (Exception e
) {
177 throw new SlcException("Cannot initialize workspace "
180 JcrUtils
.logoutQuietly(session
);
184 protected void initDistWorkspace(String workspace
) {
185 Session session
= null;
188 .loginOrCreateWorkspace(distRepository
, workspace
);
189 JcrUtils
.addPrivilege(session
, "/", "anonymous", "jcr:read");
190 } catch (RepositoryException e
) {
191 throw new SlcException("Cannot initialize workspace " + workspace
,
194 JcrUtils
.logoutQuietly(session
);
198 public void destroy() {
202 public String
generateMockConfigFile(String arch
, String branch
) {
203 StringBuffer buf
= new StringBuffer();
205 buf
.append("config_opts['root'] = '" + getIdWithArch(arch
) + "'\n");
206 buf
.append("config_opts['target_arch'] = '" + arch
+ "'\n");
207 buf
.append("config_opts['legal_host_arches'] = ('" + arch
+ "',)\n");
208 buf
.append("config_opts['chroot_setup_cmd'] = 'groupinstall buildsys-build'\n");
209 // buf.append("config_opts['dist'] = 'el6'\n");
210 buf
.append("config_opts['plugin_conf']['yum_cache_enable'] = False\n");
212 buf
.append("config_opts['scm'] = False\n");
213 buf
.append("config_opts['scm_opts']['method'] = 'git'\n");
214 buf
.append("config_opts['scm_opts']['spec'] = 'SCM_PKG.spec'\n");
215 buf
.append("config_opts['scm_opts']['ext_src_dir'] = '"
216 + getSourcesDir().getAbsolutePath() + "'\n");
217 buf
.append("config_opts['scm_opts']['git_timestamps'] = True\n");
220 if (gitDevBaseUrl
!= null && !gitDevBaseUrl
.trim().equals(""))
221 buf
.append("config_opts['scm_opts']['git_get'] = 'git clone "
222 + (branch
!= null ?
"-b " + branch
: "") + " "
223 + gitDevBaseUrl
+ "/SCM_PKG SCM_PKG'\n");
225 buf
.append("config_opts['scm_opts']['git_get'] = 'git clone "
226 + (branch
!= null ?
"-b " + branch
: "") + " "
227 + getGitBaseUrl() + "/SCM_PKG.git SCM_PKG'\n");
229 buf
.append("\nconfig_opts['yum.conf'] = \"\"\"\n");
230 buf
.append(generateYumConfigFile(arch
)).append('\n');
231 buf
.append("\"\"\"\n");
232 return buf
.toString();
235 public String
generateYumConfigFile(String arch
) {
236 StringBuffer buf
= new StringBuffer();
237 buf
.append("[main]\n");
238 buf
.append(yumConfigMainSection
).append('\n');
240 for (RpmRepository repository
: repositories
) {
241 buf
.append('[').append(repository
.getId()).append("]\n");
242 buf
.append("name=").append(repository
.getId()).append('\n');
243 if (repository
instanceof ThirdPartyRpmRepository
) {
244 buf
.append("#baseurl=").append(repository
.getUrl())
245 .append(arch
).append('/').append("\n");
246 buf
.append("baseurl=").append(proxiedReposBase
)
247 .append(repository
.getId()).append('/').append(arch
)
248 .append('/').append("\n");
249 if (((ThirdPartyRpmRepository
) repository
).getYumConf() != null)
251 ((ThirdPartyRpmRepository
) repository
).getYumConf()
252 .trim()).append('\n');
257 addManagedRepository(buf
, stagingWorkspace
, arch
);
258 if (withTestingRepository
)
259 addManagedRepository(buf
, testingWorkspace
, arch
);
260 addManagedRepository(buf
, stableWorkspace
, arch
);
261 return buf
.toString();
264 protected void addManagedRepository(StringBuffer buf
, String workspace
,
266 buf
.append('[').append(workspace
).append("]\n");
267 buf
.append("baseurl=").append(managedReposBase
).append(workspace
)
268 .append('/').append(arch
).append('/').append("\n");
269 buf
.append("gpgcheck=0").append("\n");
272 /** Creates a mock config file. */
273 public File
getMockConfigFile(String arch
, String branch
) {
274 File mockSiteDefaultsFile
= new File(mockConfDir
, "site-defaults.cfg");
275 File mockLoggingFile
= new File(mockConfDir
, "logging.ini");
276 File mockConfigFile
= new File(mockConfDir
, getIdWithArch(arch
)
279 if (!mockSiteDefaultsFile
.exists())
280 mockSiteDefaultsFile
.createNewFile();
281 if (!mockLoggingFile
.exists())
282 FileUtils
.copyFile(new File(mockEtc
+ "/logging.ini"),
285 FileUtils
.writeStringToFile(mockConfigFile
,
286 generateMockConfigFile(arch
, branch
));
287 return mockConfigFile
;
288 } catch (IOException e
) {
289 throw new SlcException("Cannot write mock config file to "
290 + mockConfigFile
, e
);
294 /** Creates a yum config file. */
295 public File
getYumRepoFile(String arch
) {
296 File yumConfigFile
= new File(yumConfDir
, getIdWithArch(arch
) + ".repo");
298 FileUtils
.writeStringToFile(yumConfigFile
,
299 generateYumConfigFile(arch
));
300 return yumConfigFile
;
301 } catch (IOException e
) {
302 throw new SlcException("Cannot write yum config file to "
307 public File
getResultDir(String arch
) {
308 return new File(mockVar
+ "/" + getIdWithArch(arch
) + "/result");
311 public File
getWorkspaceDir(String workspace
) {
312 return new File(rpmBase
+ "/" + workspace
);
315 public File
getSourcesDir() {
316 return new File(distBase
+ "/" + stableWorkspace
);
319 public String
getMockConfig(String arch
) {
320 return getIdWithArch(arch
);
323 public String
getIdWithArch(String arch
) {
324 return id
+ "-" + arch
;
327 public String
getGitBaseUrl() {
328 return managedReposBase
+ gitWorkspace
;
331 public void indexWorkspace(String workspace
) {
332 Session session
= null;
334 session
= rpmRepository
.login(workspace
);
335 session
.getRootNode().accept(
336 new NodeIndexerVisitor(new RpmIndexer()));
337 if (log
.isDebugEnabled())
338 log
.debug("Indexed workspace " + workspace
);
339 } catch (RepositoryException e
) {
340 throw new SlcException("Cannot index workspace " + workspace
, e
);
342 JcrUtils
.logoutQuietly(session
);
346 public Boolean
isDeveloperInstance() {
347 return gitDevBaseUrl
!= null;
350 /** Write (topdir)/rpmmacros and (topdir)/rpmrc */
351 public void writeRpmbuildConfigFiles(File topdir
) {
352 writeRpmbuildConfigFiles(topdir
, new File(topdir
, "rpmmacros"),
353 new File(topdir
, "rpmrc"));
356 public void writeRpmbuildConfigFiles(File topdir
, File rpmmacroFile
,
359 List
<String
> macroLines
= new ArrayList
<String
>();
360 macroLines
.add("%_topdir " + topdir
.getCanonicalPath());
361 for (String macroKey
: rpmmacros
.keySet()) {
362 macroLines
.add(macroKey
+ " " + rpmmacros
.get(macroKey
));
364 FileUtils
.writeLines(rpmmacroFile
, macroLines
);
366 List
<String
> rpmrcLines
= new ArrayList
<String
>();
367 rpmrcLines
.add("include: /usr/lib/rpm/rpmrc");
368 rpmrcLines
.add("macrofiles: " + defaultMacroFiles
+ ":"
369 + rpmmacroFile
.getCanonicalPath());
370 FileUtils
.writeLines(rpmrcFile
, rpmrcLines
);
371 } catch (IOException e
) {
372 throw new SlcException("Cannot write rpmbuild config files", e
);
377 public Map
<String
, String
> getRpmmacros() {
381 public void setRpmmacros(Map
<String
, String
> rpmmacros
) {
382 this.rpmmacros
= rpmmacros
;
385 public String
getDefaultMacroFiles() {
386 return defaultMacroFiles
;
389 public void setDefaultMacroFiles(String defaultMacroFiles
) {
390 this.defaultMacroFiles
= defaultMacroFiles
;
393 public void setArchs(List
<String
> archs
) {
397 public List
<String
> getArchs() {
401 public void setRpmBase(String stagingBase
) {
402 this.rpmBase
= stagingBase
;
405 public void setId(String id
) {
409 public void setMockVar(String mockVar
) {
410 this.mockVar
= mockVar
;
413 public void setRpmRepository(Repository rpmRepository
) {
414 this.rpmRepository
= rpmRepository
;
417 public void setDistRepository(Repository distRepository
) {
418 this.distRepository
= distRepository
;
421 public void setLocalUrlBase(String localUrlBase
) {
422 this.localUrlBase
= localUrlBase
;
425 public void setYumConfigMainSection(String yumConfigMainSection
) {
426 this.yumConfigMainSection
= yumConfigMainSection
;
429 public void setRepositories(List
<RpmRepository
> repositories
) {
430 this.repositories
= repositories
;
433 public void setRpmFactoryBaseDir(File rpmFactoryBaseDir
) {
434 this.rpmFactoryBaseDir
= rpmFactoryBaseDir
;
437 public String
getStagingWorkspace() {
438 return stagingWorkspace
;
441 public String
getTestingWorkspace() {
442 return testingWorkspace
;
445 public String
getStableWorkspace() {
446 return stableWorkspace
;
449 public void setWithTestingRepository(Boolean withTestingRepository
) {
450 this.withTestingRepository
= withTestingRepository
;
453 public void setGitDevBaseUrl(String gitBaseUrl
) {
454 this.gitDevBaseUrl
= gitBaseUrl
;