Add project-tools #142

Merged
Francesco Siddi merged 26 commits from ZedDB/blender-studio-pipeline:project-helper-tools into main 2023-08-31 20:33:04 +02:00
Showing only changes of commit 1bea5c53cd - Show all commits

99
scripts/project-helper-tools/update_blender.py Normal file → Executable file
View File

@ -1,15 +1,39 @@
#!/usr/bin/env python3
import pathlib import pathlib
import os import os
import requests import requests
import re import re
import shutil import shutil
import hashlib import hashlib
import glob
import email.utils
HOMEPAGE = "https://builder.blender.org/download/" HOMEPAGE = "https://builder.blender.org/download/"
BLENDER_BRANCH = "main" BLENDER_BRANCH = "main"
DOWNLOAD_DIR = "../../shared/artifacts/blender" DOWNLOAD_DIR = "../../shared/artifacts/blender"
def download_file(url, out_folder):
print("Downloading: " + url)
local_filename = out_folder / url.split('/')[-1]
# NOTE the stream=True parameter below
with requests.get(url, stream=True) as r:
r.raise_for_status()
with open(local_filename, 'wb') as f:
for chunk in r.iter_content(chunk_size=None):
if chunk:
f.write(chunk)
return local_filename
def shasum_matches(file, sha_sum):
with open(file, "rb") as f:
digest = hashlib.file_digest(f, "sha256")
return digest.hexdigest() == sha_sum
current_file_folder_path = pathlib.Path(__file__).parent current_file_folder_path = pathlib.Path(__file__).parent
download_folder_path = (current_file_folder_path / DOWNLOAD_DIR).resolve() download_folder_path = (current_file_folder_path / DOWNLOAD_DIR).resolve()
backup_folder_path = download_folder_path / "previous/current_snapshot" backup_folder_path = download_folder_path / "previous/current_snapshot"
@ -25,37 +49,86 @@ for f in os.listdir(download_folder_path):
shutil.copy(path_to_file, backup_folder_path) shutil.copy(path_to_file, backup_folder_path)
# Get all urls for the blender builds # Get all urls for the blender builds
platforms_to_download = { platforms_dict = {
"windows": "zip", "windows": "zip",
"darwin.x86_64": "dmg", "darwin.x86_64": "dmg",
"darwin.arm64": "dmg", "darwin.arm64": "dmg",
"linux": "tar.xz", "linux": "tar.xz",
} }
files_to_download = [] download_info = []
branch_string = "+" + BLENDER_BRANCH branch_string = "+" + BLENDER_BRANCH
reqs = requests.get(HOMEPAGE) reqs = requests.get(HOMEPAGE)
for match in re.findall('<a href=[' "'" '"][^"' "'" ']*[' "'" '"]', reqs.text): for match in re.findall('<a href=[' "'" '"][^"' "'" ']*[' "'" '"]', reqs.text):
if branch_string in match: if branch_string in match:
# Strip href and quotes around the url # Strip href and quotes around the url
download_url = match[9:-1] download_url = match[9:-1]
for platform in platforms_to_download: for platform in platforms_dict:
file_extension = platforms_to_download[platform] file_extension = platforms_dict[platform]
if re.search(platform + ".*" + file_extension + "$", download_url): if re.search(platform + ".*" + file_extension + "$", download_url):
files_to_download.append(download_url) download_info.append((platform, download_url))
updated_current_files = False
new_files_downloaded = False
# Download new builds if the shasums doesn't match # Download new builds if the shasums doesn't match
for url in files_to_download: for info in download_info:
platform = info[0]
file_extension = platforms_dict[platform]
url = info[1]
url_sha = url + ".sha256" url_sha = url + ".sha256"
sha = requests.get(url_sha).text sha = requests.get(url_sha).text.strip().lower()
current_file = download_folder_path / url.split("/")[-1] current_platform_file = glob.glob(f"{download_folder_path}/*{platform}*{file_extension}")
if os.path.exists(current_file): if len(current_platform_file) > 1:
with open(current_file, "rb") as f: print(
digest = hashlib.file_digest(f, "sha256") f"Platform {platform} has multiple downloaded files in the artifacts directory, exiting!"
if digest.hexdigest() == sha: )
exit(1)
# Check if we need to download the file by looking at the shasum of the currently downloaded file (if any)
if len(current_platform_file) == 1:
current_file = current_platform_file[0]
if shasum_matches(current_file, sha):
# We already have the current file # We already have the current file
continue continue
else:
updated_current_files = True
os.remove(current_file)
os.remove(current_file + ".sha256")
downloaded_file = wget.download(url, out=download_folder_path) download_file(url_sha, download_folder_path)
downloaded_file = download_file(url, download_folder_path)
# Check that the file we downloaded is not corrupt
if not shasum_matches(downloaded_file, sha):
print(f"Downloaded file {downloaded_file} does not match its shasum, exiting!")
exit(1)
new_files_downloaded = True
if new_files_downloaded:
# Save download date for use in the rollback script
with open(download_folder_path / "download_date", "w") as date_file:
date_file.write(email.utils.formatdate(localtime=True))
print("Updated to the latest files")
if updated_current_files:
backup_path = download_folder_path / "previous"
# Put the current backup first in the directory listing
os.rename(backup_folder_path, backup_path / "00")
backup_dirs = os.listdir(backup_path)
backup_dirs.sort(reverse=True)
# Remove older backup folders if there are more than 10
folders_to_remove = len(backup_dirs) - 10
if folders_to_remove > 0:
for dir in backup_dirs[:folders_to_remove]:
shutil.rmtree(dir)
backup_dirs = backup_dirs[folders_to_remove:]
# Bump all folder names
# Assign a number to each file, reverse the processing order to not overwrite any files.
folder_number = len(backup_dirs)
for dir in backup_dirs:
os.rename(dir, backup_path / str(folder_number).zfill(2))
folder_number -= 1
else:
shutil.rmtree(backup_folder_path)
print("Nothing downloaded, everything was up to date")