-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathadd_image.py
228 lines (205 loc) · 9.07 KB
/
add_image.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
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
import os
import yaml
def get_image_details():
"""Prompts the user for image name and target folder."""
image_name = input("Enter the name of the image (e.g., my-new-image): ")
target_folder = input("Enter the target folder (leave empty for current directory): ").strip()
return image_name, target_folder
def setup_folders(image_name, target_folder):
"""Creates the image folder and determines base folder."""
if target_folder:
base_folder = os.path.join(os.getcwd(), target_folder, image_name)
else:
base_folder = os.path.join(os.getcwd(), image_name)
os.makedirs(base_folder, exist_ok=True)
print(f"Created base folder: {base_folder}")
return base_folder
def handle_entrypoint(base_folder):
"""Handles entrypoint.sh creation or usage of existing one."""
entrypoint_path = os.path.join(base_folder, "entrypoint.sh")
if os.path.exists(entrypoint_path):
print(f"Using existing entrypoint.sh: {entrypoint_path}")
return True
default_entrypoint = """#!/usr/bin/env bash
cd /home/container
# Make internal Docker IP address available to processes
export INTERNAL_IP=`ip route get 1 | awk '{print $(NF-2);exit}'`
# Evaluate startup variables.
MODIFIED_STARTUP=$(eval "echo \"$(echo ${STARTUP} | sed -e 's/{{/${{/g' -e 's/}}/}}/g')\"")
echo "customer@apollopanel:~# ${MODIFIED_STARTUP}"
eval "${MODIFIED_STARTUP}"
"""
with open(entrypoint_path, "w") as f:
f.write(default_entrypoint)
os.chmod(entrypoint_path, 0o755) # Make executable
print(f"Created entrypoint.sh: {entrypoint_path}")
return False
def create_dockerfile(version_folder):
"""Creates a placeholder Dockerfile."""
dockerfile_path = os.path.join(version_folder, "Dockerfile")
if not os.path.exists(dockerfile_path):
with open(dockerfile_path, "w") as f:
f.write("# Placeholder Dockerfile - Please configure!\n")
print(f"Created placeholder Dockerfile: {dockerfile_path}")
return dockerfile_path
def get_matrix_input():
"""Prompts user for matrix configuration."""
use_matrix = input("Do you want to create a matrix-based workflow? (yes/no): ").lower() == "yes"
if use_matrix:
matrix_values = input("Enter comma-separated values for the matrix (e.g., 16,17,18): ").split(",")
return use_matrix, matrix_values
return use_matrix, None
def create_workflow_data(image_name, use_matrix, matrix_values, base_folder, use_existing_entrypoint):
"""Generates the workflow data dictionary."""
base_folder_rel = os.path.relpath(base_folder, os.getcwd())
if use_matrix:
image_name_base = image_name.split("-")[0] # Remove version from image name
workflow_data = {
"name": f"Build {image_name_base} images",
"on": {
"workflow_dispatch": {},
"push": {
"branches": ["main"],
"paths": [
f"{base_folder_rel}/**/*",
f"{base_folder_rel}/entrypoint.sh",
f".github/workflows/{image_name_base}.yml"
]
}
},
"jobs": {
"build": {
"name": f"{image_name_base} ${{ matrix.version }}",
"runs-on": "ubuntu-latest",
"strategy": {
"fail-fast": False,
"matrix": {
"version": [v.strip() for v in matrix_values]
}
},
"steps": [
{
"uses": "actions/checkout@v3"
},
{
"uses": "docker/setup-buildx-action@v2",
"with": {
"version": "v0.9.1",
"buildkitd-flags": "--debug"
}
},
{
"uses": "docker/login-action@v2",
"with": {
"registry": "ghcr.io",
"username": "${{ github.repository_owner }}",
"password": "${{ secrets.GITHUB_TOKEN }}"
}
},
{
"uses": "docker/build-push-action@v3",
"with": {
"context": f"./{base_folder_rel}/{image_name_base}-${{ matrix.version }}",
"file": f"./{base_folder_rel}/{image_name_base}-${{ matrix.version }}/Dockerfile",
"platforms": "linux/amd64",
"push": True,
"tags": f"ghcr.io/sparkedhost/images:{image_name_base}-${{ matrix.version }}"
}
}
]
}
}
}
else:
workflow_data = {
"name": f"Build {image_name} image",
"on": {
"workflow_dispatch": {},
"push": {
"branches": ["main"],
"paths": [
f"{base_folder_rel}/*",
f".github/workflows/{image_name}.yml"
]
}
},
"jobs": {
"push": {
"name": f"{image_name}",
"runs-on": "ubuntu-latest",
"steps": [
{
"uses": "actions/checkout@v3"
},
{
"uses": "docker/setup-buildx-action@v2",
"with": {
"version": "v0.9.1",
"buildkitd-flags": "--debug"
}
},
{
"uses": "docker/login-action@v2",
"with": {
"registry": "ghcr.io",
"username": "${{ github.repository_owner }}",
"password": "${{ secrets.GITHUB_TOKEN }}"
}
},
{
"uses": "docker/build-push-action@v3",
"with": {
"context": f"./{base_folder_rel}",
"file": f"./{base_folder_rel}/Dockerfile",
"platforms": "linux/amd64",
"push": True,
"tags": f"ghcr.io/sparkedhost/images:{image_name}"
}
}
]
}
}
}
return workflow_data
def create_workflow_file(image_name, workflow_data):
"""Creates the workflow file."""
workflows_dir = os.path.join(os.getcwd(), ".github", "workflows")
os.makedirs(workflows_dir, exist_ok=True)
workflow_file_path = os.path.join(workflows_dir, f"{image_name}.yml")
if os.path.exists(workflow_file_path):
print(f"Workflow file already exists: {workflow_file_path}")
return
# Dump the YAML data to a string
yaml_str = yaml.dump(workflow_data, indent=2, sort_keys=False, default_flow_style=False)
# Replace problematic parts
yaml_str = yaml_str.replace("workflow_dispatch: {}", "workflow_dispatch:")
yaml_str = yaml_str.replace("'on':", "on:")
# Write the modified YAML string to the file
with open(workflow_file_path, "w") as f:
f.write(yaml_str)
print(f"Created workflow file: {workflow_file_path}")
def create_image_wizard():
"""
Guides the user through the process of creating a new image, generating
the necessary files and directory structure.
"""
print("!!! Use with caution - untested !!!\n")
print("This script can be useful for starting a new image, but always verify what is created.\n")
image_name, target_folder = get_image_details()
base_folder = setup_folders(image_name, target_folder)
use_existing_entrypoint = handle_entrypoint(base_folder)
use_matrix, matrix_values = get_matrix_input()
if use_matrix:
image_name_base = image_name.split("-")[0] # Remove version from image name
for version in matrix_values:
version_folder = os.path.join(base_folder, f"{image_name_base}-{version.strip()}")
os.makedirs(version_folder, exist_ok=True)
create_dockerfile(version_folder)
else:
create_dockerfile(base_folder)
workflow_data = create_workflow_data(image_name, use_matrix, matrix_values, base_folder, use_existing_entrypoint)
create_workflow_file(image_name, workflow_data)
print("\nImage creation process complete!")
print(f"Remember to configure workflow, entrypoint and Dockerfile and push to GitHub.")
if __name__ == "__main__":
create_image_wizard()