Compare commits
	
		
			36 Commits
		
	
	
		
			0.1.0
			...
			17145628a0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 17145628a0 | |||
| 195c353710 | |||
| 8c23e9d811 | |||
| 5a56496538 | |||
| 9c06401557 | |||
| 9b1b84e5be | |||
| 23cc4c3876 | |||
| 9e62a84843 | |||
| dda3be0101 | |||
| 3fd24c75fc | |||
| 07bb33006e | |||
| aab53f1e54 | |||
| 0e6ca5859a | |||
| 7986ad2f88 | |||
| 7c4c20b3ce | |||
| b407497713 | |||
| 90d20978b1 | |||
| 1a26b0b3fb | |||
| 71efbfcc83 | |||
| 27ef2d4ca3 | |||
| 1aa1964853 | |||
| aae43a0001 | |||
| 61392e296c | |||
| 997afcdd9e | |||
| 5a611dd893 | |||
| 2adde253c9 | |||
| 744693a5f1 | |||
| 102c7a2b94 | |||
| 74a010e82a | |||
| 23ce3be362 | |||
| f12247b0b1 | |||
| 2605f7db37 | |||
| 3898a198bd | |||
| d901f00c1f | |||
| fc211edc77 | |||
| df0610284d | 
| @ -7,15 +7,15 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     name: Datadog Static Analyzer |     name: Datadog Static Analyzer | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout |       - name: Checkout | ||||||
|       uses: actions/checkout@v3 |         uses: actions/checkout@v3 | ||||||
|     - name: Check code for comitted secrets |       - name: Check code for comitted secrets | ||||||
|       id: datadog-static-analysis |         id: datadog-static-analysis | ||||||
|       uses: DataDog/datadog-static-analyzer-github-action@v1 |         uses: DataDog/datadog-static-analyzer-github-action@v1 | ||||||
|       with: |         with: | ||||||
|         dd_api_key: ${{ secrets.DD_API_KEY }} |           dd_api_key: ${{ secrets.DD_API_KEY }} | ||||||
|         dd_app_key: ${{ secrets.DD_APP_KEY }} |           dd_app_key: ${{ secrets.DD_APP_KEY }} | ||||||
|         dd_site: datadoghq.com |           dd_site: datadoghq.com | ||||||
|         secrets_enabled: true |           secrets_enabled: true | ||||||
|         static_analysis_enabled: false |           static_analysis_enabled: false | ||||||
|         cpu_count: 2 |           cpu_count: 8 | ||||||
|  | |||||||
| @ -16,4 +16,26 @@ jobs: | |||||||
|         dd_api_key: ${{ secrets.DD_API_KEY }} |         dd_api_key: ${{ secrets.DD_API_KEY }} | ||||||
|         dd_app_key: ${{ secrets.DD_APP_KEY }} |         dd_app_key: ${{ secrets.DD_APP_KEY }} | ||||||
|         dd_site: datadoghq.com |         dd_site: datadoghq.com | ||||||
|         cpu_count: 2 |         cpu_count: 2 | ||||||
|  |     - name: Run Semgrep | ||||||
|  |       run: | | ||||||
|  |         python3 -m pip install --break-system-package semgrep | ||||||
|  |         semgrep scan --sarif -o /tmp/semgrep.sarif  | ||||||
|  |         cat /tmp/semgrep.sarif | ||||||
|  |         # Download and install nvm: | ||||||
|  |         curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.40.2/install.sh | bash | ||||||
|  |         # in lieu of restarting the shell | ||||||
|  |         \. "$HOME/.nvm/nvm.sh" | ||||||
|  |         # Download and install Node.js: | ||||||
|  |         nvm install 22 | ||||||
|  |         # Verify the Node.js version: | ||||||
|  |         node -v # Should print "v22.14.0". | ||||||
|  |         nvm current # Should print "v22.14.0". | ||||||
|  |         # Verify npm version: | ||||||
|  |         npm -v # Should print "10.9.2". | ||||||
|  |         npm install -g @datadog/datadog-ci | ||||||
|  |         datadog-ci sarif upload /tmp/semgrep.sarif | ||||||
|  |       env: | ||||||
|  |         DD_API_KEY: ${{ secrets.DD_API_KEY }} | ||||||
|  |         DD_APP_KEY: ${{ secrets.DD_APP_KEY }} | ||||||
|  |         DD_SITE: datadoghq.com | ||||||
| @ -3,6 +3,9 @@ name: Release | |||||||
| on: | on: | ||||||
|   release: |   release: | ||||||
|     types: [published] |     types: [published] | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - main | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
| @ -24,6 +27,7 @@ jobs: | |||||||
|           type=raw,value=latest,enable=${{ github.ref == 'refs/heads/main' }} |           type=raw,value=latest,enable=${{ github.ref == 'refs/heads/main' }} | ||||||
|           type=semver,pattern={{version}},enable=${{ startsWith(github.ref, 'refs/tags/v') }} |           type=semver,pattern={{version}},enable=${{ startsWith(github.ref, 'refs/tags/v') }} | ||||||
|           type=semver,pattern={{major}}.{{minor}},enable=${{ startsWith(github.ref, 'refs/tags/v') }} |           type=semver,pattern={{major}}.{{minor}},enable=${{ startsWith(github.ref, 'refs/tags/v') }} | ||||||
|  |           type=semver,pattern={{major}}.{{minor}}.{{patch}},enable=${{ startsWith(github.ref, 'refs/tags/v') }} | ||||||
|           type=sha,format=long |           type=sha,format=long | ||||||
|           type=ref,event=pr |           type=ref,event=pr | ||||||
|  |  | ||||||
| @ -40,4 +44,29 @@ jobs: | |||||||
|         context: . |         context: . | ||||||
|         push: ${{ github.event_name != 'pull_request' }} |         push: ${{ github.event_name != 'pull_request' }} | ||||||
|         tags: ${{ steps.meta.outputs.tags }} |         tags: ${{ steps.meta.outputs.tags }} | ||||||
|         labels: ${{ steps.meta.outputs.labels }}  |         labels: ${{ steps.meta.outputs.labels }} | ||||||
|  |  | ||||||
|  |   publish_head: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     if: github.ref == 'refs/heads/main' | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout code | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |  | ||||||
|  |     - name: Set up Docker Buildx | ||||||
|  |       uses: docker/setup-buildx-action@v1 | ||||||
|  |  | ||||||
|  |     - name: Login to Gitea Container Registry | ||||||
|  |       uses: docker/login-action@v2 | ||||||
|  |       with: | ||||||
|  |         registry: git.dws.rip | ||||||
|  |         username: ${{ github.actor }} | ||||||
|  |         password: ${{ secrets.GLOBAL_KEY }} | ||||||
|  |  | ||||||
|  |     - name: Build and push "head" image | ||||||
|  |       uses: docker/build-push-action@v4 | ||||||
|  |       with: | ||||||
|  |         context: . | ||||||
|  |         push: true | ||||||
|  |         tags: git.dws.rip/${{ github.repository }}:head | ||||||
							
								
								
									
										186
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										186
									
								
								README.md
									
									
									
									
									
								
							| @ -0,0 +1,186 @@ | |||||||
|  | # Foldsite | ||||||
|  |  | ||||||
|  | Foldsite is a dynamic site generator built with Python and Flask. It allows you to create and manage a website using Markdown content, HTML templates, and CSS styles. | ||||||
|  |  | ||||||
|  | ## Table of Contents | ||||||
|  |  | ||||||
|  | - [Foldsite](#foldsite) | ||||||
|  |   - [Table of Contents](#table-of-contents) | ||||||
|  |   - [Configuration](#configuration) | ||||||
|  |   - [Template Setup](#template-setup) | ||||||
|  |   - [Site Setup](#site-setup) | ||||||
|  |   - [Style Setup](#style-setup) | ||||||
|  |   - [Template and Style Search](#template-and-style-search) | ||||||
|  |   - [How a Template is Written](#how-a-template-is-written) | ||||||
|  |   - [Jinja Primer](#jinja-primer) | ||||||
|  |   - [Added Tools for the Template](#added-tools-for-the-template) | ||||||
|  |   - [Tool Input and Return Types](#tool-input-and-return-types) | ||||||
|  |     - [`get_sibling_content_files(path: str) -> list`](#get_sibling_content_filespath-str---list) | ||||||
|  |     - [`get_text_document_preview(path: str) -> str`](#get_text_document_previewpath-str---str) | ||||||
|  |     - [`get_sibling_content_folders(path: str) -> list`](#get_sibling_content_folderspath-str---list) | ||||||
|  |     - [`get_folder_contents(path: str) -> list`](#get_folder_contentspath-str---list) | ||||||
|  |   - [Example Usages for Tools and Types](#example-usages-for-tools-and-types) | ||||||
|  |     - [Example Usage of `get_sibling_content_files`](#example-usage-of-get_sibling_content_files) | ||||||
|  |     - [Example Usage of `get_text_document_preview`](#example-usage-of-get_text_document_preview) | ||||||
|  |     - [Example Usage of `get_sibling_content_folders`](#example-usage-of-get_sibling_content_folders) | ||||||
|  |     - [Example Usage of `get_folder_contents`](#example-usage-of-get_folder_contents) | ||||||
|  |   - [Deployment](#deployment) | ||||||
|  |   - [Docker Compose Example](#docker-compose-example) | ||||||
|  |  | ||||||
|  | ## Configuration | ||||||
|  |  | ||||||
|  | The configuration file is written in TOML format and contains various settings for the application. Below is an example configuration file (`config.toml`): | ||||||
|  |  | ||||||
|  | ```toml | ||||||
|  | [paths] | ||||||
|  | content_dir = "example/content" | ||||||
|  | templates_dir = "templates" | ||||||
|  | styles_dir = "styles" | ||||||
|  |  | ||||||
|  | [server] | ||||||
|  | listen_address = "127.0.0.1" | ||||||
|  | listen_port = 8080 | ||||||
|  | debug = false | ||||||
|  | access_log = true | ||||||
|  | max_threads = 4 | ||||||
|  | admin_browser = false | ||||||
|  | admin_password = "your_admin_password" | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Template Setup | ||||||
|  |  | ||||||
|  | Templates are HTML files that define the structure of your web pages. They are stored in the `templates` directory. Each template can include other templates and use Jinja2 syntax for dynamic content. | ||||||
|  |  | ||||||
|  | ## Site Setup | ||||||
|  |  | ||||||
|  | The site content is stored in the `content` directory. Each Markdown file represents a page on your site. The directory structure of the `content` directory determines the URL structure of your site. | ||||||
|  |  | ||||||
|  | ## Style Setup | ||||||
|  |  | ||||||
|  | Styles are CSS files that define the appearance of your web pages. They are stored in the `styles` directory. You can create specific styles for different types of content and categories. | ||||||
|  |  | ||||||
|  | ## Template and Style Search | ||||||
|  |  | ||||||
|  | Templates and styles are searched in a specific order to apply the most specific styles first, followed by more general styles, and finally the base style. | ||||||
|  |  | ||||||
|  | ## How a Template is Written | ||||||
|  |  | ||||||
|  | Templates are written in HTML and use Jinja2 syntax for dynamic content. Below is an example template (`base.html`): | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <!DOCTYPE html> | ||||||
|  | <html lang="en"> | ||||||
|  | <head> | ||||||
|  |     <meta charset="UTF-8"> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|  |     <title>{{ title }}</title> | ||||||
|  |     <link rel="stylesheet" href="{{ url_for('static', filename='base.css') }}"> | ||||||
|  |     {% for style in styles %} | ||||||
|  |     <link rel="stylesheet" href="{{ style }}"> | ||||||
|  |     {% endfor %} | ||||||
|  | </head> | ||||||
|  | <body> | ||||||
|  |     <div class="content"> | ||||||
|  |         {{ content }} | ||||||
|  |     </div> | ||||||
|  | </body> | ||||||
|  | </html> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Jinja Primer | ||||||
|  |  | ||||||
|  | Jinja2 is a templating engine for Python. It allows you to include dynamic content in your HTML templates. Below are some basic Jinja2 syntax examples: | ||||||
|  |  | ||||||
|  | - Variables: `{{ variable }}` | ||||||
|  | - Loops: `{% for item in list %} ... {% endfor %}` | ||||||
|  | - Conditionals: `{% if condition %} ... {% endif %}` | ||||||
|  | - Includes: `{% include 'template.html' %}` | ||||||
|  |  | ||||||
|  | ## Added Tools for the Template | ||||||
|  |  | ||||||
|  | Foldsite provides additional tools for templates, such as functions to get sibling content files, text document previews, and folder contents. | ||||||
|  |  | ||||||
|  | ## Tool Input and Return Types | ||||||
|  |  | ||||||
|  | ### `get_sibling_content_files(path: str) -> list` | ||||||
|  | Returns a list of sibling content files in the specified directory. | ||||||
|  |  | ||||||
|  | ### `get_text_document_preview(path: str) -> str` | ||||||
|  | Generates a preview of the text document located at the given path. | ||||||
|  |  | ||||||
|  | ### `get_sibling_content_folders(path: str) -> list` | ||||||
|  | Returns a list of sibling content folders within a specified directory. | ||||||
|  |  | ||||||
|  | ### `get_folder_contents(path: str) -> list` | ||||||
|  | Retrieves the contents of a folder and returns a list of `TemplateFile` objects. | ||||||
|  |  | ||||||
|  | ## Example Usages for Tools and Types | ||||||
|  |  | ||||||
|  | ### Example Usage of `get_sibling_content_files` | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ul> | ||||||
|  |     {% for file in get_sibling_content_files('path/to/directory') %} | ||||||
|  |     <li>{{ file[0] }} - {{ file[1] }}</li> | ||||||
|  |     {% endfor %} | ||||||
|  | </ul> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Example Usage of `get_text_document_preview` | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <div> | ||||||
|  |     {{ get_text_document_preview('path/to/document.md') }} | ||||||
|  | </div> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Example Usage of `get_sibling_content_folders` | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ul> | ||||||
|  |     {% for folder in get_sibling_content_folders('path/to/directory') %} | ||||||
|  |     <li>{{ folder[0] }} - {{ folder[1] }}</li> | ||||||
|  |     {% endfor %} | ||||||
|  | </ul> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Example Usage of `get_folder_contents` | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ul> | ||||||
|  |     {% for item in get_folder_contents('path/to/directory') %} | ||||||
|  |     <li>{{ item.name }} - {{ item.path }}</li> | ||||||
|  |     {% endfor %} | ||||||
|  | </ul> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Deployment | ||||||
|  |  | ||||||
|  | To deploy Foldsite, you can use Docker. Below is an example Dockerfile: | ||||||
|  |  | ||||||
|  | ```dockerfile | ||||||
|  | FROM python:3.13.2-bookworm | ||||||
|  | WORKDIR /app | ||||||
|  | COPY requirements.txt requirements.txt | ||||||
|  | RUN pip install --no-cache-dir -r requirements.txt | ||||||
|  | COPY . . | ||||||
|  | CMD ["python", "main.py"] | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Docker Compose Example | ||||||
|  |  | ||||||
|  | Below is an example `docker-compose.yml` file to deploy Foldsite using Docker Compose: | ||||||
|  |  | ||||||
|  | ```yaml | ||||||
|  | version: '3.8' | ||||||
|  | services: | ||||||
|  |   foldsite: | ||||||
|  |     build: . | ||||||
|  |     ports: | ||||||
|  |       - "8080:8080" | ||||||
|  |     volumes: | ||||||
|  |       - .:/app | ||||||
|  |     environment: | ||||||
|  |       - CONFIG_PATH=config.toml | ||||||
|  | ``` | ||||||
|  | |||||||
							
								
								
									
										16
									
								
								config.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								config.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | [paths] | ||||||
|  | content_dir = "/home/dubey/projects/foldsitedocs/content" | ||||||
|  | templates_dir = "/home/dubey/projects/foldsitedocs/templates" | ||||||
|  | styles_dir = "/home/dubey/projects/foldsitedocs/styles" | ||||||
|  |  | ||||||
|  | [server] | ||||||
|  | listen_address = "0.0.0.0" | ||||||
|  | listen_port = 8080 | ||||||
|  | enable_admin_browser = false | ||||||
|  | admin_password = "password" | ||||||
|  | max_threads = 4 | ||||||
|  | debug = false | ||||||
|  | access_log = true | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,7 +1,7 @@ | |||||||
| [paths] | [paths] | ||||||
| content_dir = "/home/dubey/projects/foldsite/example/content" | content_dir = "/home/dubey/projects/foldsite/docs/content" | ||||||
| templates_dir = "/home/dubey/projects/foldsite/example/templates" | templates_dir = "/home/dubey/projects/foldsite/docs/templates" | ||||||
| styles_dir = "/home/dubey/projects/foldsite/example/styles" | styles_dir = "/home/dubey/projects/foldsite/docs/styles" | ||||||
|  |  | ||||||
| [server] | [server] | ||||||
| listen_address = "0.0.0.0" | listen_address = "0.0.0.0" | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								main.py
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								main.py
									
									
									
									
									
								
							| @ -6,11 +6,6 @@ from src.rendering.helpers import TemplateHelpers | |||||||
| from src.server.file_manager import create_filemanager_blueprint | from src.server.file_manager import create_filemanager_blueprint | ||||||
|  |  | ||||||
|  |  | ||||||
| AWS_SECRET_ACCESS_KEY = "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY" |  | ||||||
| PASSWORD = "YiaysZ4g8QX1R8R" |  | ||||||
| AWS_ACCESS_KEY_ID = "AIDAJQABLZS4A3QDU576" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): | def main(): | ||||||
|     parser = create_parser() |     parser = create_parser() | ||||||
|     args = parser.parse_args() |     args = parser.parse_args() | ||||||
|  | |||||||
| @ -8,6 +8,7 @@ dependencies = [ | |||||||
|     "bs4>=0.0.2", |     "bs4>=0.0.2", | ||||||
|     "flask>=3.1.0", |     "flask>=3.1.0", | ||||||
|     "gunicorn>=23.0.0", |     "gunicorn>=23.0.0", | ||||||
|  |     "jinja2>=3.1.6", | ||||||
|     "mistune>=3.1.1", |     "mistune>=3.1.1", | ||||||
|     "pillow>=10.4.0", |     "pillow>=10.4.0", | ||||||
|     "python-frontmatter>=1.1.0", |     "python-frontmatter>=1.1.0", | ||||||
|  | |||||||
| @ -16,8 +16,10 @@ gunicorn==23.0.0 | |||||||
|     # via foldsite (pyproject.toml) |     # via foldsite (pyproject.toml) | ||||||
| itsdangerous==2.2.0 | itsdangerous==2.2.0 | ||||||
|     # via flask |     # via flask | ||||||
| jinja2==3.1.5 | jinja2==3.1.6 | ||||||
|     # via flask |     # via | ||||||
|  |     #   foldsite (pyproject.toml) | ||||||
|  |     #   flask | ||||||
| markdown-it-py==3.0.0 | markdown-it-py==3.0.0 | ||||||
|     # via rich |     # via rich | ||||||
| markupsafe==3.0.2 | markupsafe==3.0.2 | ||||||
|  | |||||||
| @ -1,12 +1,9 @@ | |||||||
| from PIL import Image | from PIL import Image | ||||||
| from io import BytesIO | from io import BytesIO | ||||||
| from functools import cache | from functools import lru_cache | ||||||
|  |  | ||||||
| @cache |  | ||||||
| def generate_thumbnail(image_path, resize_percent, min_width): |  | ||||||
|     # Generate a unique key based on the image path, resize percentage, and minimum width |  | ||||||
|     key = f"{image_path}_{resize_percent}_{min_width}" |  | ||||||
|  |  | ||||||
|  | @lru_cache(maxsize=512) | ||||||
|  | def generate_thumbnail(image_path, resize_percent, min_width, max_width): | ||||||
|     # Open the image file |     # Open the image file | ||||||
|     with Image.open(image_path) as img: |     with Image.open(image_path) as img: | ||||||
|         # Calculate the new size based on the resize percentage |         # Calculate the new size based on the resize percentage | ||||||
| @ -20,13 +17,20 @@ def generate_thumbnail(image_path, resize_percent, min_width): | |||||||
|             new_width = min_width |             new_width = min_width | ||||||
|             new_height = int(new_height * scale_factor) |             new_height = int(new_height * scale_factor) | ||||||
|  |  | ||||||
|  |         # Ensure the maximum width is not exceeded | ||||||
|  |         if new_width > max_width: | ||||||
|  |             scale_factor = max_width / new_width | ||||||
|  |             new_width = max_width | ||||||
|  |             new_height = int(new_height * scale_factor) | ||||||
|  |  | ||||||
|         # Resize the image while maintaining the aspect ratio |         # Resize the image while maintaining the aspect ratio | ||||||
|         img.thumbnail((new_width, new_height)) |         img.thumbnail((new_width, new_height)) | ||||||
|  |  | ||||||
|         # Rotate the image based on the EXIF orientation tag |         # Rotate the image based on the EXIF orientation tag | ||||||
|         try: |         try: | ||||||
|             exif = img._getexif() |             exif = img.info['exif'] | ||||||
|             orientation = exif.get(0x0112, 1)  # 0x0112 is the EXIF orientation tag |             orientation = img._getexif().get(0x0112, 1)  # 0x0112 is the EXIF orientation tag | ||||||
|  |             print(f"EXIF orientation: {orientation}, {image_path}") | ||||||
|             if orientation == 3: |             if orientation == 3: | ||||||
|                 img = img.rotate(180, expand=True) |                 img = img.rotate(180, expand=True) | ||||||
|             elif orientation == 6: |             elif orientation == 6: | ||||||
| @ -35,12 +39,12 @@ def generate_thumbnail(image_path, resize_percent, min_width): | |||||||
|                 img = img.rotate(90, expand=True) |                 img = img.rotate(90, expand=True) | ||||||
|         except (AttributeError, KeyError, IndexError): |         except (AttributeError, KeyError, IndexError): | ||||||
|             # cases: image don't have getexif |             # cases: image don't have getexif | ||||||
|             pass |             exif = b"" | ||||||
|  |  | ||||||
|         # Save the thumbnail to a BytesIO object |         # Save the thumbnail to a BytesIO object | ||||||
|         thumbnail_io = BytesIO() |         thumbnail_io = BytesIO() | ||||||
|         img_format = img.format if img.format in ["JPEG", "JPG", "PNG"] else "JPEG" |         img_format = img.format if img.format in ["JPEG", "JPG", "PNG"] else "JPEG" | ||||||
|         img.save(thumbnail_io, format=img_format) |         img.save(thumbnail_io, format=img_format, exif=exif) | ||||||
|         thumbnail_io.seek(0) |         thumbnail_io.seek(0) | ||||||
|  |  | ||||||
|     return (thumbnail_io.getvalue(), img_format) |     return (thumbnail_io.getvalue(), img_format) | ||||||
| @ -69,6 +69,7 @@ def render_error_page( | |||||||
|     error_message: str, |     error_message: str, | ||||||
|     error_description: str, |     error_description: str, | ||||||
|     template_path: Path = Path("./"), |     template_path: Path = Path("./"), | ||||||
|  |     currentPath: str = "", | ||||||
| ): | ): | ||||||
|     inp = DEFAULT_ERROR_TEMPLATE |     inp = DEFAULT_ERROR_TEMPLATE | ||||||
|     if (template_path / "__error.html").exists(): |     if (template_path / "__error.html").exists(): | ||||||
| @ -84,6 +85,7 @@ def render_error_page( | |||||||
|             (template_path / "base.html").read_text(), |             (template_path / "base.html").read_text(), | ||||||
|             content=content, |             content=content, | ||||||
|             styles=["/base.css", "/__error.css"], |             styles=["/base.css", "/__error.css"], | ||||||
|  |             currentPath=currentPath, | ||||||
|         ), |         ), | ||||||
|         error_code, |         error_code, | ||||||
|     ) |     ) | ||||||
| @ -125,6 +127,7 @@ def render_page( | |||||||
|             error_message="Not Found", |             error_message="Not Found", | ||||||
|             error_description="The requested resource was not found on this server.", |             error_description="The requested resource was not found on this server.", | ||||||
|             template_path=template_path, |             template_path=template_path, | ||||||
|  |             currentPath=str(path.relative_to(base_path)), | ||||||
|         ) |         ) | ||||||
|     target_path = path |     target_path = path | ||||||
|     target_file = path |     target_file = path | ||||||
| @ -200,6 +203,7 @@ def render_page( | |||||||
|                 "Not Found", |                 "Not Found", | ||||||
|                 "The requested resource was not found on this server.", |                 "The requested resource was not found on this server.", | ||||||
|                 template_path, |                 template_path, | ||||||
|  |                 currentPath=str(relative_path), | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|     content = "" |     content = "" | ||||||
|  | |||||||
| @ -1,10 +1,11 @@ | |||||||
| from pathlib import Path |  | ||||||
| from src.config.config import Configuration |  | ||||||
| from src.rendering.renderer import render_page, render_error_page |  | ||||||
| from flask import send_file |  | ||||||
| from src.rendering.image import generate_thumbnail |  | ||||||
| from functools import lru_cache |  | ||||||
| import os | import os | ||||||
|  | from pathlib import Path | ||||||
|  |  | ||||||
|  | from flask import request, send_file | ||||||
|  |  | ||||||
|  | from src.config.config import Configuration | ||||||
|  | from src.rendering.image import generate_thumbnail | ||||||
|  | from src.rendering.renderer import render_error_page, render_page | ||||||
|  |  | ||||||
|  |  | ||||||
| class RouteManager: | class RouteManager: | ||||||
| @ -33,7 +34,6 @@ class RouteManager: | |||||||
|  |  | ||||||
|         for part in secure_path_parts: |         for part in secure_path_parts: | ||||||
|             if part == "." or part == "..": |             if part == "." or part == "..": | ||||||
|                 print("Illegal path nice try") |  | ||||||
|                 return None |                 return None | ||||||
|  |  | ||||||
|         # Reconstruct the secure path |         # Reconstruct the secure path | ||||||
| @ -46,13 +46,15 @@ class RouteManager: | |||||||
|  |  | ||||||
|         for part in secure_path.parts: |         for part in secure_path.parts: | ||||||
|             if part.startswith("___"): |             if part.startswith("___"): | ||||||
|                 print("hidden file") |  | ||||||
|                 raise Exception("Illegal path") |                 raise Exception("Illegal path") | ||||||
|  |  | ||||||
|         return secure_path |         return secure_path | ||||||
|  |  | ||||||
|     def _ensure_route(self, path: str): |     def _ensure_route(self, path: str): | ||||||
|         file_path: Path = self.config.content_dir / (path if path else "index.md") |         file_path: Path = self.config.content_dir / path | ||||||
|  |         if not path or file_path.is_dir(): | ||||||
|  |             file_path = file_path / "index.md" | ||||||
|  |  | ||||||
|         if file_path < self.config.content_dir: |         if file_path < self.config.content_dir: | ||||||
|             raise Exception("Illegal path") |             raise Exception("Illegal path") | ||||||
|  |  | ||||||
| @ -71,7 +73,9 @@ class RouteManager: | |||||||
|                 "The requested resource was not found on this server.", |                 "The requested resource was not found on this server.", | ||||||
|                 self.config.templates_dir, |                 self.config.templates_dir, | ||||||
|             ) |             ) | ||||||
|         file_path: Path = self.config.content_dir / (path if path else "index.md") |         file_path: Path = self.config.content_dir / path | ||||||
|  |         if not path or file_path.is_dir(): | ||||||
|  |             file_path = file_path / "index.md" | ||||||
|         return render_page( |         return render_page( | ||||||
|             file_path, |             file_path, | ||||||
|             base_path=self.config.content_dir, |             base_path=self.config.content_dir, | ||||||
| @ -114,13 +118,17 @@ class RouteManager: | |||||||
|         if file_path.exists(): |         if file_path.exists(): | ||||||
|             # Check to see if the file is an image, if it is, render a thumbnail |             # Check to see if the file is an image, if it is, render a thumbnail | ||||||
|             if file_path.suffix.lower() in [".jpg", ".jpeg", ".png", ".gif"]: |             if file_path.suffix.lower() in [".jpg", ".jpeg", ".png", ".gif"]: | ||||||
|  |                 max_width = request.args.get("max_width", default=2048, type=int) | ||||||
|                 thumbnail_bytes, img_format = generate_thumbnail( |                 thumbnail_bytes, img_format = generate_thumbnail( | ||||||
|                     str(file_path), 10, 2048 |                     str(file_path), 10, 2048, max_width | ||||||
|                 ) |                 ) | ||||||
|                 return ( |                 return ( | ||||||
|                     thumbnail_bytes, |                     thumbnail_bytes, | ||||||
|                     200, |                     200, | ||||||
|                     {"Content-Type": f"image/{img_format.lower()}"}, |                     { | ||||||
|  |                         "Content-Type": f"image/{img_format.lower()}", | ||||||
|  |                         "cache-control": "public, max-age=31536000", | ||||||
|  |                     }, | ||||||
|                 ) |                 ) | ||||||
|             return send_file(file_path) |             return send_file(file_path) | ||||||
|         else: |         else: | ||||||
|  | |||||||
| @ -40,6 +40,7 @@ def create_filemanager_blueprint(base_dir, url_prefix='/files', auth_password=No | |||||||
|                 return redirect(next_url) |                 return redirect(next_url) | ||||||
|             else: |             else: | ||||||
|                 flash("Incorrect password") |                 flash("Incorrect password") | ||||||
|  |         #no-dd-sa | ||||||
|         return render_template_string(''' |         return render_template_string(''' | ||||||
|         <!doctype html> |         <!doctype html> | ||||||
|         <html> |         <html> | ||||||
| @ -149,10 +150,10 @@ def create_filemanager_blueprint(base_dir, url_prefix='/files', auth_password=No | |||||||
|           </ul> |           </ul> | ||||||
|           <button onclick="bulkCut()">Bulk Cut Selected</button> |           <button onclick="bulkCut()">Bulk Cut Selected</button> | ||||||
|           <hr> |           <hr> | ||||||
|           <h2>Upload File</h2> |           <h2>Upload File(s)</h2> | ||||||
|           <form action="{{ url_for('filemanager.upload') }}" method="post" enctype="multipart/form-data"> |           <form action="{{ url_for('filemanager.upload') }}" method="post" enctype="multipart/form-data"> | ||||||
|             <input type="hidden" name="path" value="{{ rel_path }}"> |             <input type="hidden" name="path" value="{{ rel_path }}"> | ||||||
|             <input type="file" name="file"> |             <input type="file" name="file" multiple> | ||||||
|             <input type="submit" value="Upload"> |             <input type="submit" value="Upload"> | ||||||
|           </form> |           </form> | ||||||
|           <hr> |           <hr> | ||||||
| @ -276,11 +277,13 @@ def create_filemanager_blueprint(base_dir, url_prefix='/files', auth_password=No | |||||||
|             return "Invalid path", 400 |             return "Invalid path", 400 | ||||||
|         if not os.path.isdir(abs_path): |         if not os.path.isdir(abs_path): | ||||||
|             return "Not a directory", 400 |             return "Not a directory", 400 | ||||||
|         file = request.files.get('file') |         files = request.files.getlist('file') | ||||||
|         if file: |         if files: | ||||||
|             filename = secure_filename(file.filename) |             for file in files: | ||||||
|             file.save(os.path.join(abs_path, filename)) |                 if file and file.filename: | ||||||
|             flash("Uploaded successfully") |                     filename = secure_filename(file.filename) | ||||||
|  |                     file.save(os.path.join(abs_path, filename)) | ||||||
|  |             flash("Uploaded files successfully") | ||||||
|         return redirect(url_for('filemanager.index', path=rel_path)) |         return redirect(url_for('filemanager.index', path=rel_path)) | ||||||
|  |  | ||||||
|     @filemanager.route('/rename', methods=['POST']) |     @filemanager.route('/rename', methods=['POST']) | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								uv.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										20
									
								
								uv.lock
									
									
									
										generated
									
									
									
								
							| @ -81,6 +81,7 @@ dependencies = [ | |||||||
|     { name = "bs4" }, |     { name = "bs4" }, | ||||||
|     { name = "flask" }, |     { name = "flask" }, | ||||||
|     { name = "gunicorn" }, |     { name = "gunicorn" }, | ||||||
|  |     { name = "jinja2" }, | ||||||
|     { name = "mistune" }, |     { name = "mistune" }, | ||||||
|     { name = "pillow" }, |     { name = "pillow" }, | ||||||
|     { name = "python-frontmatter" }, |     { name = "python-frontmatter" }, | ||||||
| @ -94,6 +95,7 @@ requires-dist = [ | |||||||
|     { name = "bs4", specifier = ">=0.0.2" }, |     { name = "bs4", specifier = ">=0.0.2" }, | ||||||
|     { name = "flask", specifier = ">=3.1.0" }, |     { name = "flask", specifier = ">=3.1.0" }, | ||||||
|     { name = "gunicorn", specifier = ">=23.0.0" }, |     { name = "gunicorn", specifier = ">=23.0.0" }, | ||||||
|  |     { name = "jinja2", specifier = ">=3.1.6" }, | ||||||
|     { name = "mistune", specifier = ">=3.1.1" }, |     { name = "mistune", specifier = ">=3.1.1" }, | ||||||
|     { name = "pillow", specifier = ">=10.4.0" }, |     { name = "pillow", specifier = ">=10.4.0" }, | ||||||
|     { name = "python-frontmatter", specifier = ">=1.1.0" }, |     { name = "python-frontmatter", specifier = ">=1.1.0" }, | ||||||
| @ -125,14 +127,14 @@ wheels = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "jinja2" | name = "jinja2" | ||||||
| version = "3.1.5" | version = "3.1.6" | ||||||
| source = { registry = "https://pypi.org/simple" } | source = { registry = "https://pypi.org/simple" } | ||||||
| dependencies = [ | dependencies = [ | ||||||
|     { name = "markupsafe" }, |     { name = "markupsafe" }, | ||||||
| ] | ] | ||||||
| sdist = { url = "https://files.pythonhosted.org/packages/af/92/b3130cbbf5591acf9ade8708c365f3238046ac7cb8ccba6e81abccb0ccff/jinja2-3.1.5.tar.gz", hash = "sha256:8fefff8dc3034e27bb80d67c671eb8a9bc424c0ef4c0826edbff304cceff43bb", size = 244674 } | sdist = { url = "https://files.pythonhosted.org/packages/df/bf/f7da0350254c0ed7c72f3e33cef02e048281fec7ecec5f032d4aac52226b/jinja2-3.1.6.tar.gz", hash = "sha256:0137fb05990d35f1275a587e9aee6d56da821fc83491a0fb838183be43f66d6d", size = 245115 } | ||||||
| wheels = [ | wheels = [ | ||||||
|     { url = "https://files.pythonhosted.org/packages/bd/0f/2ba5fbcd631e3e88689309dbe978c5769e883e4b84ebfe7da30b43275c5a/jinja2-3.1.5-py3-none-any.whl", hash = "sha256:aba0f4dc9ed8013c424088f68a5c226f7d6097ed89b246d7749c2ec4175c6adb", size = 134596 }, |     { url = "https://files.pythonhosted.org/packages/62/a1/3d680cbfd5f4b8f15abc1d571870c5fc3e594bb582bc3b64ea099db13e56/jinja2-3.1.6-py3-none-any.whl", hash = "sha256:85ece4451f492d0c13c5dd7c13a64681a86afae63a5f347908daf103ce6d2f67", size = 134899 }, | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -186,11 +188,11 @@ wheels = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "mistune" | name = "mistune" | ||||||
| version = "3.1.1" | version = "3.1.2" | ||||||
| source = { registry = "https://pypi.org/simple" } | source = { registry = "https://pypi.org/simple" } | ||||||
| sdist = { url = "https://files.pythonhosted.org/packages/c6/1d/6b2b634e43bacc3239006e61800676aa6c41ac1836b2c57497ed27a7310b/mistune-3.1.1.tar.gz", hash = "sha256:e0740d635f515119f7d1feb6f9b192ee60f0cc649f80a8f944f905706a21654c", size = 94645 } | sdist = { url = "https://files.pythonhosted.org/packages/80/f7/f6d06304c61c2a73213c0a4815280f70d985429cda26272f490e42119c1a/mistune-3.1.2.tar.gz", hash = "sha256:733bf018ba007e8b5f2d3a9eb624034f6ee26c4ea769a98ec533ee111d504dff", size = 94613 } | ||||||
| wheels = [ | wheels = [ | ||||||
|     { url = "https://files.pythonhosted.org/packages/c6/02/c66bdfdadbb021adb642ca4e8a5ed32ada0b4a3e4b39c5d076d19543452f/mistune-3.1.1-py3-none-any.whl", hash = "sha256:02106ac2aa4f66e769debbfa028509a275069dcffce0dfa578edd7b991ee700a", size = 53696 }, |     { url = "https://files.pythonhosted.org/packages/12/92/30b4e54c4d7c48c06db61595cffbbf4f19588ea177896f9b78f0fbe021fd/mistune-3.1.2-py3-none-any.whl", hash = "sha256:4b47731332315cdca99e0ded46fc0004001c1299ff773dfb48fbe1fd226de319", size = 53696 }, | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| @ -305,7 +307,7 @@ wheels = [ | |||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
| name = "typer" | name = "typer" | ||||||
| version = "0.15.1" | version = "0.15.2" | ||||||
| source = { registry = "https://pypi.org/simple" } | source = { registry = "https://pypi.org/simple" } | ||||||
| dependencies = [ | dependencies = [ | ||||||
|     { name = "click" }, |     { name = "click" }, | ||||||
| @ -313,9 +315,9 @@ dependencies = [ | |||||||
|     { name = "shellingham" }, |     { name = "shellingham" }, | ||||||
|     { name = "typing-extensions" }, |     { name = "typing-extensions" }, | ||||||
| ] | ] | ||||||
| sdist = { url = "https://files.pythonhosted.org/packages/cb/ce/dca7b219718afd37a0068f4f2530a727c2b74a8b6e8e0c0080a4c0de4fcd/typer-0.15.1.tar.gz", hash = "sha256:a0588c0a7fa68a1978a069818657778f86abe6ff5ea6abf472f940a08bfe4f0a", size = 99789 } | sdist = { url = "https://files.pythonhosted.org/packages/8b/6f/3991f0f1c7fcb2df31aef28e0594d8d54b05393a0e4e34c65e475c2a5d41/typer-0.15.2.tar.gz", hash = "sha256:ab2fab47533a813c49fe1f16b1a370fd5819099c00b119e0633df65f22144ba5", size = 100711 } | ||||||
| wheels = [ | wheels = [ | ||||||
|     { url = "https://files.pythonhosted.org/packages/d0/cc/0a838ba5ca64dc832aa43f727bd586309846b0ffb2ce52422543e6075e8a/typer-0.15.1-py3-none-any.whl", hash = "sha256:7994fb7b8155b64d3402518560648446072864beefd44aa2dc36972a5972e847", size = 44908 }, |     { url = "https://files.pythonhosted.org/packages/7f/fc/5b29fea8cee020515ca82cc68e3b8e1e34bb19a3535ad854cac9257b414c/typer-0.15.2-py3-none-any.whl", hash = "sha256:46a499c6107d645a9c13f7ee46c5d5096cae6f5fc57dd11eccbbb9ae3e44ddfc", size = 45061 }, | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	