Compare commits
	
		
			26 Commits
		
	
	
		
			v1.0.1
			...
			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 | 
| @ -7,15 +7,15 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     name: Datadog Static Analyzer | ||||
|     steps: | ||||
|     - name: Checkout | ||||
|       uses: actions/checkout@v3 | ||||
|     - name: Check code for comitted secrets | ||||
|       id: datadog-static-analysis | ||||
|       uses: DataDog/datadog-static-analyzer-github-action@v1 | ||||
|       with: | ||||
|         dd_api_key: ${{ secrets.DD_API_KEY }} | ||||
|         dd_app_key: ${{ secrets.DD_APP_KEY }} | ||||
|         dd_site: datadoghq.com | ||||
|         secrets_enabled: true | ||||
|         static_analysis_enabled: false | ||||
|         cpu_count: 2 | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Check code for comitted secrets | ||||
|         id: datadog-static-analysis | ||||
|         uses: DataDog/datadog-static-analyzer-github-action@v1 | ||||
|         with: | ||||
|           dd_api_key: ${{ secrets.DD_API_KEY }} | ||||
|           dd_app_key: ${{ secrets.DD_APP_KEY }} | ||||
|           dd_site: datadoghq.com | ||||
|           secrets_enabled: true | ||||
|           static_analysis_enabled: false | ||||
|           cpu_count: 8 | ||||
|  | ||||
| @ -17,3 +17,25 @@ jobs: | ||||
|         dd_app_key: ${{ secrets.DD_APP_KEY }} | ||||
|         dd_site: datadoghq.com | ||||
|         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 | ||||
| @ -167,6 +167,7 @@ COPY . . | ||||
| CMD ["python", "main.py"] | ||||
| ``` | ||||
|  | ||||
|  | ||||
| ## Docker Compose Example | ||||
|  | ||||
| Below is an example `docker-compose.yml` file to deploy Foldsite using Docker Compose: | ||||
|  | ||||
							
								
								
									
										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,3 +0,0 @@ | ||||
| # 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. | ||||
| @ -1,19 +0,0 @@ | ||||
| # 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" | ||||
| ``` | ||||
| @ -1,12 +0,0 @@ | ||||
| # 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"] | ||||
| ``` | ||||
| @ -1,16 +0,0 @@ | ||||
| # 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 | ||||
| ``` | ||||
| @ -1,39 +0,0 @@ | ||||
| # 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> | ||||
| ``` | ||||
| @ -1,23 +0,0 @@ | ||||
| # 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> | ||||
| ``` | ||||
| @ -1,20 +0,0 @@ | ||||
| # Foldsite Documentation | ||||
|  | ||||
| Welcome to the Foldsite documentation. This site will guide you through the setup, configuration, and usage of Foldsite. | ||||
|  | ||||
| ## Table of Contents | ||||
|  | ||||
| 1. [Introduction](introduction.md) | ||||
| 2. [Configuration](configuration.md) | ||||
| 3. [Template Setup](template-setup.md) | ||||
| 4. [Site Setup](site-setup.md) | ||||
| 5. [Style Setup](style-setup.md) | ||||
| 6. [Template and Style Search](template-style-search.md) | ||||
| 7. [How a Template is Written](how-template-written.md) | ||||
| 8. [Jinja Primer](jinja-primer.md) | ||||
| 9. [Added Tools for the Template](added-tools.md) | ||||
| 10. [Tool Input and Return Types](tool-input-return-types.md) | ||||
| 11. [Example Usages for Tools and Types](example-usages.md) | ||||
| 12. [Deployment](deployment.md) | ||||
| 13. [Docker Compose Example](docker-compose-example.md) | ||||
| 14. [Folder Layout](folder-layout.md) | ||||
| @ -1,5 +0,0 @@ | ||||
| # Introduction | ||||
|  | ||||
| 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. | ||||
|  | ||||
| This documentation will guide you through the setup, configuration, and usage of Foldsite. | ||||
| @ -1,8 +0,0 @@ | ||||
| # 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' %}` | ||||
| @ -1,3 +0,0 @@ | ||||
| # 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. | ||||
| @ -1,3 +0,0 @@ | ||||
| # 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. | ||||
| @ -1,3 +0,0 @@ | ||||
| # 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. | ||||
| @ -1,3 +0,0 @@ | ||||
| # 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. | ||||
| @ -1,13 +0,0 @@ | ||||
| # 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. | ||||
| @ -1,24 +0,0 @@ | ||||
| article { | ||||
|     max-width: 800px; | ||||
|     margin: 0 auto; | ||||
| } | ||||
|  | ||||
| article h1, article h2, article h3, article h4, article h5, article h6 { | ||||
|     margin-top: 1.5rem; | ||||
| } | ||||
|  | ||||
| article p { | ||||
|     margin: 1rem 0; | ||||
| } | ||||
|  | ||||
| article pre { | ||||
|     background: #f4f4f4; | ||||
|     padding: 1rem; | ||||
|     overflow-x: auto; | ||||
| } | ||||
|  | ||||
| article code { | ||||
|     background: #f4f4f4; | ||||
|     padding: 0.2rem 0.4rem; | ||||
|     border-radius: 3px; | ||||
| } | ||||
| @ -1,50 +0,0 @@ | ||||
| body { | ||||
|     font-family: Arial, sans-serif; | ||||
|     line-height: 1.6; | ||||
|     margin: 0; | ||||
|     padding: 0; | ||||
| } | ||||
|  | ||||
| header { | ||||
|     background: #333; | ||||
|     color: #fff; | ||||
|     padding: 1rem 0; | ||||
|     text-align: center; | ||||
| } | ||||
|  | ||||
| header h1 { | ||||
|     margin: 0; | ||||
| } | ||||
|  | ||||
| nav ul { | ||||
|     list-style: none; | ||||
|     padding: 0; | ||||
| } | ||||
|  | ||||
| nav ul li { | ||||
|     display: inline; | ||||
|     margin-right: 1rem; | ||||
| } | ||||
|  | ||||
| nav ul li a { | ||||
|     color: #fff; | ||||
|     text-decoration: none; | ||||
| } | ||||
|  | ||||
| nav ul li a:hover { | ||||
|     text-decoration: underline; | ||||
| } | ||||
|  | ||||
| main { | ||||
|     padding: 2rem; | ||||
| } | ||||
|  | ||||
| footer { | ||||
|     background: #333; | ||||
|     color: #fff; | ||||
|     text-align: center; | ||||
|     padding: 1rem 0; | ||||
|     position: fixed; | ||||
|     bottom: 0; | ||||
|     width: 100%; | ||||
| } | ||||
							
								
								
									
										4
									
								
								docs/templates/__file.md.html
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								docs/templates/__file.md.html
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +0,0 @@ | ||||
| <article> | ||||
|     {{ content|safe }} | ||||
| </article> | ||||
|  | ||||
							
								
								
									
										42
									
								
								docs/templates/base.html
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										42
									
								
								docs/templates/base.html
									
									
									
									
										vendored
									
									
								
							| @ -1,42 +0,0 @@ | ||||
| <!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> | ||||
|     <header> | ||||
|         <h1>Foldsite Documentation</h1> | ||||
|         <nav> | ||||
|             <ul> | ||||
|                 <li><a href="{{ url_for('default_route', path='index.md') }}">Home</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='introduction.md') }}">Introduction</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='configuration.md') }}">Configuration</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='template-setup.md') }}">Template Setup</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='site-setup.md') }}">Site Setup</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='style-setup.md') }}">Style Setup</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='template-style-search.md') }}">Template and Style Search</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='how-template-written.md') }}">How a Template is Written</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='jinja-primer.md') }}">Jinja Primer</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='added-tools.md') }}">Added Tools</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='tool-input-return-types.md') }}">Tool Input and Return Types</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='example-usages.md') }}">Example Usages</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='deployment.md') }}">Deployment</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='docker-compose-example.md') }}">Docker Compose Example</a></li> | ||||
|                 <li><a href="{{ url_for('default_route', path='folder-layout.md') }}">Folder Layout</a></li> | ||||
|             </ul> | ||||
|         </nav> | ||||
|     </header> | ||||
|     <main> | ||||
|         {{ content|safe }} | ||||
|     </main> | ||||
|     <footer> | ||||
|         <p>© 2023 Foldsite</p> | ||||
|     </footer> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										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 | ||||
|  | ||||
|  | ||||
| AWS_SECRET_ACCESS_KEY = "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY" | ||||
| PASSWORD = "YiaysZ4g8QX1R8R" | ||||
| AWS_ACCESS_KEY_ID = "AIDAJQABLZS4A3QDU576" | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     parser = create_parser() | ||||
|     args = parser.parse_args() | ||||
|  | ||||
| @ -34,9 +34,9 @@ def generate_thumbnail(image_path, resize_percent, min_width, max_width): | ||||
|             if orientation == 3: | ||||
|                 img = img.rotate(180, expand=True) | ||||
|             elif orientation == 6: | ||||
|                 img = img.rotate(0, expand=True) | ||||
|                 img = img.rotate(270, expand=True) | ||||
|             elif orientation == 8: | ||||
|                 img = img.rotate(180, expand=True) | ||||
|                 img = img.rotate(90, expand=True) | ||||
|         except (AttributeError, KeyError, IndexError): | ||||
|             # cases: image don't have getexif | ||||
|             exif = b"" | ||||
|  | ||||
| @ -69,6 +69,7 @@ def render_error_page( | ||||
|     error_message: str, | ||||
|     error_description: str, | ||||
|     template_path: Path = Path("./"), | ||||
|     currentPath: str = "", | ||||
| ): | ||||
|     inp = DEFAULT_ERROR_TEMPLATE | ||||
|     if (template_path / "__error.html").exists(): | ||||
| @ -84,6 +85,7 @@ def render_error_page( | ||||
|             (template_path / "base.html").read_text(), | ||||
|             content=content, | ||||
|             styles=["/base.css", "/__error.css"], | ||||
|             currentPath=currentPath, | ||||
|         ), | ||||
|         error_code, | ||||
|     ) | ||||
| @ -125,6 +127,7 @@ def render_page( | ||||
|             error_message="Not Found", | ||||
|             error_description="The requested resource was not found on this server.", | ||||
|             template_path=template_path, | ||||
|             currentPath=str(path.relative_to(base_path)), | ||||
|         ) | ||||
|     target_path = path | ||||
|     target_file = path | ||||
| @ -200,6 +203,7 @@ def render_page( | ||||
|                 "Not Found", | ||||
|                 "The requested resource was not found on this server.", | ||||
|                 template_path, | ||||
|                 currentPath=str(relative_path), | ||||
|             ) | ||||
|  | ||||
|     content = "" | ||||
|  | ||||
| @ -1,9 +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, request | ||||
| from src.rendering.image import generate_thumbnail | ||||
| 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: | ||||
| @ -32,7 +34,6 @@ class RouteManager: | ||||
|  | ||||
|         for part in secure_path_parts: | ||||
|             if part == "." or part == "..": | ||||
|                 print("Illegal path nice try") | ||||
|                 return None | ||||
|  | ||||
|         # Reconstruct the secure path | ||||
| @ -45,13 +46,15 @@ class RouteManager: | ||||
|  | ||||
|         for part in secure_path.parts: | ||||
|             if part.startswith("___"): | ||||
|                 print("hidden file") | ||||
|                 raise Exception("Illegal path") | ||||
|  | ||||
|         return secure_path | ||||
|  | ||||
|     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: | ||||
|             raise Exception("Illegal path") | ||||
|  | ||||
| @ -70,7 +73,9 @@ class RouteManager: | ||||
|                 "The requested resource was not found on this server.", | ||||
|                 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( | ||||
|             file_path, | ||||
|             base_path=self.config.content_dir, | ||||
| @ -120,8 +125,10 @@ class RouteManager: | ||||
|                 return ( | ||||
|                     thumbnail_bytes, | ||||
|                     200, | ||||
|                     {"Content-Type": f"image/{img_format.lower()}", | ||||
|                      "cache-control": "public, max-age=31536000"}, | ||||
|                     { | ||||
|                         "Content-Type": f"image/{img_format.lower()}", | ||||
|                         "cache-control": "public, max-age=31536000", | ||||
|                     }, | ||||
|                 ) | ||||
|             return send_file(file_path) | ||||
|         else: | ||||
|  | ||||
| @ -40,6 +40,7 @@ def create_filemanager_blueprint(base_dir, url_prefix='/files', auth_password=No | ||||
|                 return redirect(next_url) | ||||
|             else: | ||||
|                 flash("Incorrect password") | ||||
|         #no-dd-sa | ||||
|         return render_template_string(''' | ||||
|         <!doctype html> | ||||
|         <html> | ||||
| @ -149,10 +150,10 @@ def create_filemanager_blueprint(base_dir, url_prefix='/files', auth_password=No | ||||
|           </ul> | ||||
|           <button onclick="bulkCut()">Bulk Cut Selected</button> | ||||
|           <hr> | ||||
|           <h2>Upload File</h2> | ||||
|           <h2>Upload File(s)</h2> | ||||
|           <form action="{{ url_for('filemanager.upload') }}" method="post" enctype="multipart/form-data"> | ||||
|             <input type="hidden" name="path" value="{{ rel_path }}"> | ||||
|             <input type="file" name="file"> | ||||
|             <input type="file" name="file" multiple> | ||||
|             <input type="submit" value="Upload"> | ||||
|           </form> | ||||
|           <hr> | ||||
| @ -276,11 +277,13 @@ def create_filemanager_blueprint(base_dir, url_prefix='/files', auth_password=No | ||||
|             return "Invalid path", 400 | ||||
|         if not os.path.isdir(abs_path): | ||||
|             return "Not a directory", 400 | ||||
|         file = request.files.get('file') | ||||
|         if file: | ||||
|             filename = secure_filename(file.filename) | ||||
|             file.save(os.path.join(abs_path, filename)) | ||||
|             flash("Uploaded successfully") | ||||
|         files = request.files.getlist('file') | ||||
|         if files: | ||||
|             for file in files: | ||||
|                 if file and file.filename: | ||||
|                     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)) | ||||
|  | ||||
|     @filemanager.route('/rename', methods=['POST']) | ||||
|  | ||||
		Reference in New Issue
	
	Block a user
	