Compare commits
No commits in common. "main" and "1.0" have entirely different histories.
@ -3,11 +3,9 @@ name: Docker Build and Publish
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches: [ main ]
|
branches: [ main ]
|
||||||
tags: [ 'v*.*.*' ]
|
tags: [ 'v*' ]
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: [ main ]
|
branches: [ main ]
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@ -22,11 +20,11 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
images: git.dws.rip/${{ github.repository }}
|
images: git.dws.rip/${{ github.repository }}
|
||||||
tags: |
|
tags: |
|
||||||
type=raw,value=latest,enable=${{ github.ref == 'refs/heads/main' }}
|
type=ref,event=branch
|
||||||
type=semver,pattern={{version}},enable=${{ startsWith(github.ref, 'refs/tags/v') }}
|
|
||||||
type=semver,pattern={{major}}.{{minor}},enable=${{ startsWith(github.ref, 'refs/tags/v') }}
|
|
||||||
type=sha,format=long
|
|
||||||
type=ref,event=pr
|
type=ref,event=pr
|
||||||
|
type=semver,pattern={{version}}
|
||||||
|
type=semver,pattern={{major}}.{{minor}}
|
||||||
|
type=sha
|
||||||
|
|
||||||
- name: Login to Gitea Container Registry
|
- name: Login to Gitea Container Registry
|
||||||
uses: docker/login-action@v2
|
uses: docker/login-action@v2
|
||||||
|
10
README.md
10
README.md
@ -114,12 +114,4 @@ spectra/
|
|||||||
|
|
||||||
- `FLASK_ENV`: Set to 'production' in production
|
- `FLASK_ENV`: Set to 'production' in production
|
||||||
- `WORKERS`: Number of Gunicorn workers (default: 4)
|
- `WORKERS`: Number of Gunicorn workers (default: 4)
|
||||||
- `PORT`: Override default port (default: 5000)
|
- `PORT`: Override default port (default: 5000)
|
||||||
|
|
||||||
## Release Process
|
|
||||||
|
|
||||||
To create a release:
|
|
||||||
- Create and push a tag: `git tag v1.0.0 && git push origin v1.0.0`
|
|
||||||
- Create a release in Gitea UI using that tag
|
|
||||||
- The workflow will build and push the Docker image with appropriate version tags
|
|
||||||
- The Docker image will be available at: `git.dws.rip/your-repo/image:v1.0.0`
|
|
51
app.py
51
app.py
@ -289,7 +289,7 @@ def get_images():
|
|||||||
images = []
|
images = []
|
||||||
for photo in photos:
|
for photo in photos:
|
||||||
factor = random.randint(2, 3)
|
factor = random.randint(2, 3)
|
||||||
if photo.height < 4000 and photo.width < 4000:
|
if photo.height < 4000 or photo.width < 4000:
|
||||||
factor = 1
|
factor = 1
|
||||||
if photo.orientation == 6 or photo.orientation == 8:
|
if photo.orientation == 6 or photo.orientation == 8:
|
||||||
width, height = photo.height, photo.width
|
width, height = photo.height, photo.width
|
||||||
@ -461,22 +461,17 @@ def admin_upload():
|
|||||||
file_path = os.path.join(app.config["UPLOAD_FOLDER"], filename)
|
file_path = os.path.join(app.config["UPLOAD_FOLDER"], filename)
|
||||||
file.save(file_path)
|
file.save(file_path)
|
||||||
|
|
||||||
# Extract EXIF data with error handling
|
# Extract EXIF data
|
||||||
exif = {}
|
exif = None
|
||||||
exifraw = None
|
exifraw = None
|
||||||
width = height = 0
|
with Image.open(file_path) as img:
|
||||||
try:
|
exifraw = img.info["exif"]
|
||||||
with Image.open(file_path) as img:
|
width, height = img.size
|
||||||
width, height = img.size
|
exif = {
|
||||||
if hasattr(img, '_getexif') and img._getexif() is not None:
|
ExifTags.TAGS[k]: v
|
||||||
exifraw = img.info.get("exif")
|
for k, v in img._getexif().items()
|
||||||
exif = {
|
if k in ExifTags.TAGS
|
||||||
ExifTags.TAGS[k]: v
|
}
|
||||||
for k, v in img._getexif().items()
|
|
||||||
if k in ExifTags.TAGS
|
|
||||||
}
|
|
||||||
except Exception as e:
|
|
||||||
logger.warning(f"Error reading EXIF data for {filename}: {str(e)}")
|
|
||||||
|
|
||||||
# Generate a unique key for the image
|
# Generate a unique key for the image
|
||||||
unique_key = hashlib.sha256(
|
unique_key = hashlib.sha256(
|
||||||
@ -494,25 +489,25 @@ def admin_upload():
|
|||||||
# Generate thumbnails
|
# Generate thumbnails
|
||||||
generate_thumbnails(filename)
|
generate_thumbnails(filename)
|
||||||
|
|
||||||
# Handle exposure time with error handling
|
# Get image dimensions
|
||||||
try:
|
with Image.open(file_path) as img:
|
||||||
exposure_time = exif.get("ExposureTime", 0)
|
width, height = img.size
|
||||||
if isinstance(exposure_time, tuple):
|
|
||||||
exposure_fraction = f"{exposure_time[0]}/{exposure_time[1]}"
|
|
||||||
else:
|
|
||||||
exposure_fraction = f"1/{int(1/float(exposure_time))}" if exposure_time else "0"
|
|
||||||
except (TypeError, ZeroDivisionError):
|
|
||||||
exposure_fraction = "0"
|
|
||||||
|
|
||||||
# Create database entry with safe defaults
|
exposure_time = exif["ExposureTime"]
|
||||||
|
if isinstance(exposure_time, tuple):
|
||||||
|
exposure_fraction = f"{exposure_time[0]}/{exposure_time[1]}"
|
||||||
|
else:
|
||||||
|
exposure_fraction = f"1/{int(1/float(exposure_time))}"
|
||||||
|
|
||||||
|
# Create database entry
|
||||||
db_session = DBSession()
|
db_session = DBSession()
|
||||||
new_photo = Photo(
|
new_photo = Photo(
|
||||||
input_filename=filename,
|
input_filename=filename,
|
||||||
thumbnail_filename=f"{os.path.splitext(filename)[0]}/256_{filename}",
|
thumbnail_filename=f"{os.path.splitext(filename)[0]}/256_{filename}",
|
||||||
focal_length=str(
|
focal_length=str(
|
||||||
exif.get("FocalLengthIn35mmFilm", exif.get("FocalLength", "0"))
|
exif.get("FocalLengthIn35mmFilm", exif.get("FocalLength", ""))
|
||||||
),
|
),
|
||||||
aperture=str(exif.get("FNumber", "0")),
|
aperture=str(exif.get("FNumber", "")),
|
||||||
shutter_speed=exposure_fraction,
|
shutter_speed=exposure_fraction,
|
||||||
date_taken=datetime.strptime(
|
date_taken=datetime.strptime(
|
||||||
str(exif.get("DateTime", "1970:01:01 00:00:00")), "%Y:%m:%d %H:%M:%S"
|
str(exif.get("DateTime", "1970:01:01 00:00:00")), "%Y:%m:%d %H:%M:%S"
|
||||||
|
@ -203,8 +203,8 @@
|
|||||||
<td class="editable" data-field="iso">{{ photo.iso }}</td>
|
<td class="editable" data-field="iso">{{ photo.iso }}</td>
|
||||||
<td>{{ photo.width }}x{{ photo.height }}</td>
|
<td>{{ photo.width }}x{{ photo.height }}</td>
|
||||||
<td>
|
<td>
|
||||||
<button id="save-btn">Save</button>
|
<button onclick="saveChanges(this)">Save</button>
|
||||||
<button class="delete-btn" id="delete-btn">Delete</button>
|
<button onclick="deletePhoto(this)" class="delete-btn">Delete</button>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
@ -241,11 +241,8 @@
|
|||||||
<input type="text" id="about.location" name="about.location" value="{{ config.about.location }}">
|
<input type="text" id="about.location" name="about.location" value="{{ config.about.location }}">
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="about.profile_image">Profile Image:</label>
|
<label for="about.profile_image">Profile Image Path:</label>
|
||||||
<div style="display: flex; align-items: center; gap: 1rem;">
|
<input type="text" id="about.profile_image" name="about.profile_image" value="{{ config.about.profile_image }}">
|
||||||
<img id="profile-preview" src="/static/profile.jpeg" alt="Profile" style="width: 100px; height: 100px; object-fit: cover; border-radius: 50%;">
|
|
||||||
<input type="file" id="profile_image_upload" accept="image/jpeg,image/png" style="flex: 1;">
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="about.bio">Bio (Markdown):</label>
|
<label for="about.bio">Bio (Markdown):</label>
|
||||||
@ -338,38 +335,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
document.getElementById('delete-btn').addEventListener('click', deletePhoto);
|
|
||||||
document.getElementById('save-btn').addEventListener('click', saveChanges);
|
|
||||||
|
|
||||||
document.getElementById('profile_image_upload').addEventListener('change', async (e) => {
|
|
||||||
const file = e.target.files[0];
|
|
||||||
if (!file) return;
|
|
||||||
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('profile_image', file);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const response = await fetch('/admin/upload_profile', {
|
|
||||||
method: 'POST',
|
|
||||||
body: formData
|
|
||||||
});
|
|
||||||
|
|
||||||
const result = await response.json();
|
|
||||||
if (result.success) {
|
|
||||||
document.getElementById('profile-preview').src = '/static/profile.jpeg?' + new Date().getTime();
|
|
||||||
} else {
|
|
||||||
alert('Error uploading profile image: ' + result.error);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
alert('Error uploading profile image: ' + error);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
document.getElementById('configForm').addEventListener('submit', async (e) => {
|
document.getElementById('configForm').addEventListener('submit', async (e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
const formData = {};
|
const formData = {};
|
||||||
const inputs = e.target.querySelectorAll('input:not([type="file"]), textarea');
|
const inputs = e.target.querySelectorAll('input, textarea');
|
||||||
|
|
||||||
inputs.forEach(input => {
|
inputs.forEach(input => {
|
||||||
formData[input.name] = input.value;
|
formData[input.name] = input.value;
|
||||||
|
Loading…
Reference in New Issue
Block a user