forked from opensearch-project/opensearch-build
-
Notifications
You must be signed in to change notification settings - Fork 0
/
input_manifests.py
157 lines (130 loc) · 6.48 KB
/
input_manifests.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
# SPDX-License-Identifier: Apache-2.0
#
# The OpenSearch Contributors require contributions made to
# this file be licensed under the Apache-2.0 license or a
# compatible open source license.
import glob
import logging
import os
import re
from abc import abstractmethod
from typing import Dict, List, Type, Union
from manifests.input_manifest import InputManifest
from manifests.manifests import Manifests
from manifests_workflow.component_opensearch import ComponentOpenSearch
from manifests_workflow.component_opensearch_dashboards_min import ComponentOpenSearchDashboardsMin
from manifests_workflow.component_opensearch_min import ComponentOpenSearchMin
from system.temporary_directory import TemporaryDirectory
class InputManifests(Manifests):
def __init__(self, name: str) -> None:
self.name = name
self.prefix = name.lower().replace(" ", "-")
super().__init__(InputManifest, InputManifests.files(self.prefix))
@classmethod
def manifests_path(self) -> str:
return os.path.realpath(os.path.join(os.path.dirname(__file__), "..", "..", "manifests"))
@classmethod
def jenkins_path(self) -> str:
return os.path.realpath(os.path.join(os.path.dirname(__file__), "..", "..", "jenkins"))
@classmethod
def cron_jenkinsfile(self) -> str:
return os.path.join(self.jenkins_path(), "check-for-build.jenkinsfile")
@classmethod
def files(self, name: str) -> List:
results = []
for filename in glob.glob(os.path.join(self.manifests_path(), f"**/{name}-*.yml")):
# avoids the -maven manifest
match = re.search(rf"^{name}-([0-9.]*).yml$", os.path.basename(filename))
if match:
results.append(filename)
return results
@abstractmethod
def update(self, min_klass: Union[Type[ComponentOpenSearchMin], Type[ComponentOpenSearchDashboardsMin]], component_klass: Type[ComponentOpenSearch], keep: bool = False) -> None:
known_versions = self.versions
logging.info(f"Known versions: {known_versions}")
main_versions: Dict = {}
with TemporaryDirectory(keep=keep, chdir=True) as work_dir:
logging.info(f"Checking out components into {work_dir.name}")
# check out and build #main, 1.x, etc.
branches = min_klass.branches()
logging.info(f"Checking {self.name} {branches} branches")
for branch in branches:
c = min_klass.checkout(
path=os.path.join(work_dir.name, self.name.replace(" ", ""), branch),
branch=branch,
)
version = c.version
logging.info(f"{self.name}#{branch} is version {version}")
if version not in main_versions.keys():
main_versions[version] = [c]
if component_klass is not None:
# components can increment their own version first without incrementing min
manifest = self.latest
logging.info(f"Examining components in the latest manifest of {manifest.build.name} ({manifest.build.version})")
for component in manifest.components.values():
if component.name == self.name:
continue
logging.info(f"Checking out {component.name}#main")
component = component_klass.checkout(
name=component.name,
path=os.path.join(work_dir.name, component.name),
opensearch_version=manifest.build.version,
branch="main",
)
component_version = component.version
if component_version:
release_version = ".".join(component_version.split(".")[:3])
if release_version not in main_versions.keys():
main_versions[release_version] = []
main_versions[release_version].append(component)
logging.info(f"{component.name}#main is version {release_version} (from {component_version})")
# summarize
logging.info("Found versions on main:")
for main_version in main_versions.keys():
for component in main_versions[main_version]:
logging.info(f" {component.name}={main_version}")
# generate new manifests
for release_version in sorted(main_versions.keys() - known_versions):
self.write_manifest(release_version, main_versions[release_version])
self.add_to_cron(release_version)
def create_manifest(self, version: str, components: List = []) -> InputManifest:
data: Dict = {
"schema-version": "1.0",
"build": {
"name": self.name,
"version": version
},
"ci": {
"image": {
"name": "opensearchstaging/ci-runner:centos7-x64-arm64-jdkmulti-node10.24.1-cypress6.9.1-20211028"
}
},
"components": [],
}
for component in components:
logging.info(f" Adding {component.name}")
data["components"].append(component.to_dict())
return InputManifest(data)
def write_manifest(self, version: str, components: List = []) -> None:
logging.info(f"Creating new version: {version}")
manifest = self.create_manifest(version, components)
manifest_dir = os.path.join(self.manifests_path(), version)
os.makedirs(manifest_dir, exist_ok=True)
manifest_path = os.path.join(manifest_dir, f"{self.prefix}-{version}.yml")
manifest.to_file(manifest_path)
logging.info(f"Wrote {manifest_path}")
def add_to_cron(self, version: str) -> None:
logging.info(f"Adding new version to cron: {version}")
jenkinsfile = self.cron_jenkinsfile()
with open(jenkinsfile, "r") as f:
data = f.read()
cron_entry = f"H/10 * * * * %INPUT_MANIFEST={version}/{self.prefix}-{version}.yml;TARGET_JOB_NAME=distribution-build-{self.prefix}\n"
if cron_entry in data:
raise ValueError(f"{jenkinsfile} already contains an entry for {self.prefix} {version}")
data = data.replace(
"parameterizedCron '''\n",
f"parameterizedCron '''\n{' ' * 12}{cron_entry}"
)
with open(jenkinsfile, "w") as f:
f.write(data)
logging.info(f"Wrote {jenkinsfile}")