diff --git a/app.py b/app.py index aa07f16..9e051cf 100644 --- a/app.py +++ b/app.py @@ -1,4 +1,4 @@ -from flask import Flask, request, jsonify, send_from_directory, render_template, url_for, redirect, send_file, session +from flask import Flask, request, jsonify, send_from_directory, render_template, url_for, redirect, send_file, session, make_response, flash from werkzeug.utils import secure_filename import shortuuid import os @@ -15,6 +15,7 @@ from pygments.util import ClassNotFound import json from flask_login import LoginManager, UserMixin, login_user, login_required, logout_user, current_user, login_remembered import hashlib +import secrets app = Flask(__name__) app.secret_key = 'your_secret_key_here' # Add this line @@ -40,14 +41,38 @@ def init_db(): db.cursor().executescript(f.read()) db.commit() +def migrate_db(): + db = get_db() + cursor = db.cursor() + + # Check if api_key column exists + cursor.execute("PRAGMA table_info(users)") + columns = [column[1] for column in cursor.fetchall()] + + if 'api_key' not in columns: + print("Adding api_key column to users table") + cursor.execute("ALTER TABLE users ADD COLUMN api_key TEXT") + db.commit() + +# Call this function after init_db() +init_db() +migrate_db() + @app.teardown_appcontext def close_connection(exception): db = getattr(threading.current_thread(), '_database', None) if db is not None: db.close() -# Initialize database -init_db() +# Add this function near the top of your file, after the imports +def get_username(user_id): + if user_id is None: + return 'Anonymous' + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT username FROM users WHERE id = ?", (user_id,)) + user = cursor.fetchone() + return user[0] if user else 'Unknown' # Add this function to delete old files def delete_old_files(): @@ -85,10 +110,11 @@ login_manager.init_app(app) login_manager.login_view = 'login' class User(UserMixin): - def __init__(self, id, username, password_hash): + def __init__(self, id, username, password_hash, api_key=None): self.id = id self.username = username self.password_hash = password_hash + self.api_key = api_key @staticmethod def hash_password(password): @@ -103,6 +129,10 @@ class User(UserMixin): new_key = hashlib.pbkdf2_hmac('sha256', provided_password.encode('utf-8'), salt, 100000) return stored_key == new_key + @staticmethod + def generate_api_key(): + return secrets.token_urlsafe(32) + @login_manager.user_loader def load_user(user_id): db = get_db() @@ -110,13 +140,24 @@ def load_user(user_id): cursor.execute("SELECT * FROM users WHERE id = ?", (user_id,)) user = cursor.fetchone() if user: - return User(user[0], user[1], user[2]) + # Print debug information + print(f"User data: {user}") + # Check if we have all required fields + if len(user) >= 4: + return User(user[0], user[1], user[2], user[3]) + else: + print(f"Incomplete user data for user_id: {user_id}") + return None + print(f"No user found for user_id: {user_id}") return None @app.route('/') def index(): - if current_user.is_authenticated: - return render_template('index.html', user=current_user) + try: + if current_user.is_authenticated: + return render_template('index.html', user=current_user) + except Exception as e: + print(f"Error in index route: {str(e)}") return render_template('index.html', user=None) @app.route('/u/') @@ -185,20 +226,25 @@ def serve_user_page(username, filename=None): parent_url=parent_url, current_folder=current_folder) -@app.route('/') -def redirect_vanity(path): - parts = path.rstrip('/').split('/') - vanity = parts[0] - subpath = '/'.join(parts[1:]) if len(parts) > 1 else '' - +@app.route('/') +def redirect_vanity(vanity): db = get_db() cursor = db.cursor() cursor.execute("SELECT * FROM content WHERE vanity = ?", (vanity,)) - target = cursor.fetchone() + content = cursor.fetchone() - if target: - content_type, content_data = target[1], target[2] - if content_type == 'pastebin': + if content: + content_type, content_data, created_at, user_id = content[1], content[2], content[3], content[4] + + if content_type == 'file': + file_path = os.path.join(app.config['UPLOAD_FOLDER'], content_data) + if os.path.exists(file_path): + return send_file(file_path) + else: + return "File not found", 404 + elif content_type == 'url': + return redirect(content_data) + elif content_type == 'pastebin': try: lexer = guess_lexer(content_data) language = lexer.aliases[0] @@ -210,28 +256,20 @@ def redirect_vanity(path): highlighted_code = highlight(content_data, lexer, formatter) css = formatter.get_style_defs('.source') return render_template('content.html', + content={ + 'user_id': user_id, + 'username': get_username(user_id), + 'created_at': created_at, + 'vanity': vanity + }, highlighted_content=highlighted_code, css=css, raw_content=content_data, - created_at=target[3], - vanity=vanity, - language=language) - elif content_type == 'file': - file_path = os.path.join(app.config['UPLOAD_FOLDER'], f'{vanity}_{content_data}') - file_info = { - 'name': content_data, - 'size': os.path.getsize(file_path), - 'modified_at': datetime.fromtimestamp(os.path.getmtime(file_path)).strftime('%Y-%m-%d %H:%M:%S'), - 'url': url_for('download_file', vanity=vanity) - } - return render_template('file.html', **file_info) - elif content_type == 'folder': - return redirect(url_for('folder_content', vanity=vanity)) - elif content_type == 'url': - return render_template('content.html', content=content_data, url=content_data) + language=language, + url=None) # Set url to None for pastebins return render_template('404.html'), 404 -@app.route('//raw', methods=['GET']) +@app.route('//raw') def raw_vanity(vanity): db = get_db() cursor = db.cursor() @@ -242,96 +280,17 @@ def raw_vanity(vanity): return target[2], 200, {'Content-Type': 'text/plain; charset=utf-8'} return 'Not Found', 404 -@app.route('/folder/', methods=['GET']) -def folder_content(vanity): - db = get_db() - cursor = db.cursor() - cursor.execute("SELECT * FROM content WHERE vanity = ? AND type = 'folder'", (vanity,)) - target = cursor.fetchone() - if target: - folder_path = os.path.join(app.config['UPLOAD_FOLDER'], vanity) - files = [] - for root, _, filenames in os.walk(folder_path): - for filename in filenames: - file_path = os.path.join(root, filename) - relative_path = os.path.relpath(file_path, folder_path) - file_url = url_for('download_folder_file', vanity=vanity, file_name=relative_path) - files.append({'name': relative_path, 'url': file_url}) - - # Pagination - per_page = 10 - page = int(request.args.get('page', 1)) - start = (page - 1) * per_page - end = start + per_page - total_files = len(files) - files = files[start:end] +# Replace the LoginForm and RegistrationForm classes with simple classes +class LoginForm: + def __init__(self, username, password, remember): + self.username = username + self.password = password + self.remember = remember - prev_url = url_for('folder_content', vanity=vanity, page=page-1) if page > 1 else None - next_url = url_for('folder_content', vanity=vanity, page=page+1) if end < total_files else None - - download_all_url = url_for('download_folder_as_zip', vanity=vanity) - - # Get the current folder name - current_folder = os.path.basename(folder_path) - - return render_template('folder.html', - files=files, - prev_url=prev_url, - next_url=next_url, - download_all_url=download_all_url, - current_folder=current_folder) - - return 'Not Found', 404 - -@app.route('/folder//download', methods=['GET']) -def download_folder_as_zip(vanity): - db = get_db() - cursor = db.cursor() - cursor.execute("SELECT * FROM content WHERE vanity = ? AND type = 'folder'", (vanity,)) - target = cursor.fetchone() - if target: - folder_path = os.path.join(app.config['UPLOAD_FOLDER'], vanity) - zip_path = os.path.join(app.config['UPLOAD_FOLDER'], f'{vanity}.zip') - - with zipfile.ZipFile(zip_path, 'w', zipfile.ZIP_DEFLATED) as zipf: - for root, _, files in os.walk(folder_path): - for file in files: - file_path = os.path.join(root, file) - zipf.write(file_path, os.path.relpath(file_path, folder_path)) - - return send_from_directory(app.config['UPLOAD_FOLDER'], f'{vanity}.zip', as_attachment=True) - return 'Not Found', 404 - -@app.route('/folder//', methods=['GET']) -def download_folder_file(vanity, file_name): - folder_path = os.path.join(app.config['UPLOAD_FOLDER'], vanity) - file_path = os.path.join(folder_path, file_name) - if os.path.isfile(file_path): - return send_from_directory(folder_path, file_name, as_attachment=True) - return 'Not Found', 404 - -@app.route('/register', methods=['GET', 'POST']) -def register(): - if request.method == 'POST': - username = request.form['username'] - password = request.form['password'] - db = get_db() - cursor = db.cursor() - cursor.execute("SELECT * FROM users WHERE username = ?", (username,)) - if cursor.fetchone(): - return "Username already exists" - hashed_password = User.hash_password(password) - cursor.execute("INSERT INTO users (username, password_hash) VALUES (?, ?)", - (username, hashed_password)) - db.commit() - - # Create user directory - user_folder = os.path.join(app.config['UPLOAD_FOLDER'], username) - if not os.path.exists(user_folder): - os.makedirs(user_folder) - - return redirect(url_for('login')) - return render_template('register.html') +class RegistrationForm: + def __init__(self, username, password): + self.username = username + self.password = password @app.route('/login', methods=['GET', 'POST']) def login(): @@ -339,17 +298,48 @@ def login(): username = request.form['username'] password = request.form['password'] remember = 'remember' in request.form + form = LoginForm(username, password, remember) db = get_db() cursor = db.cursor() - cursor.execute("SELECT * FROM users WHERE username = ?", (username,)) + cursor.execute("SELECT * FROM users WHERE username = ?", (form.username,)) user = cursor.fetchone() - if user and User.verify_password(user[2], password): - user_obj = User(user[0], user[1], user[2]) - login_user(user_obj, remember=remember) - return redirect(url_for('user_files', username=username)) + if user and User.verify_password(user[2], form.password): + user_obj = User(user[0], user[1], user[2], user[3]) + login_user(user_obj, remember=form.remember) + return redirect(url_for('user_files', username=form.username)) return "Invalid username or password" return render_template('login.html') +@app.route('/register', methods=['GET', 'POST']) +def register(): + if request.method == 'POST': + username = request.form['username'] + password = request.form['password'] + form = RegistrationForm(username, password) + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT * FROM users WHERE username = ?", (form.username,)) + if cursor.fetchone(): + return "Username already exists" + hashed_password = User.hash_password(form.password) + api_key = User.generate_api_key() + try: + cursor.execute("INSERT INTO users (username, password_hash, api_key) VALUES (?, ?, ?)", + (form.username, hashed_password, api_key)) + except sqlite3.OperationalError: + # If api_key column doesn't exist, insert without it + cursor.execute("INSERT INTO users (username, password_hash) VALUES (?, ?)", + (form.username, hashed_password)) + db.commit() + + # Create user directory + user_folder = os.path.join(app.config['UPLOAD_FOLDER'], form.username) + if not os.path.exists(user_folder): + os.makedirs(user_folder) + + return redirect(url_for('login')) + return render_template('register.html') + @app.route('/logout') @login_required def logout(): @@ -388,35 +378,47 @@ def user_files(username, subpath=''): items.append({'name': item, 'type': 'folder', 'path': relative_path}) folders.append(relative_path) + # Fetch user's uploads (including files, pastebins, and shortened URLs) + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT * FROM content WHERE user_id = ?", (current_user.id,)) + user_uploads = cursor.fetchall() + + uploads = [] + for upload in user_uploads: + uploads.append({ + 'type': upload[1], + 'vanity': upload[0], + 'data': upload[2], + 'created_at': upload[3] + }) + parent_folder = os.path.dirname(subpath.rstrip('/')) if subpath else None current_folder = os.path.basename(current_path) - # Check if index.html exists in the current folder - index_exists = 'index.html' in [item['name'] for item in items if item['type'] == 'file'] - - # Get the current setting for ignoring index.html ignore_index = session.get(f'ignore_index_{username}', False) - + return render_template('user_files.html', username=username, items=items, folders=folders, + uploads=uploads, current_path=subpath.rstrip('/'), parent_folder=parent_folder, current_folder=current_folder, - index_exists=index_exists, ignore_index=ignore_index) -@app.route('/dash//toggle_index') +@app.route('/dash//toggle_index', methods=['POST']) @login_required def toggle_index(username): if current_user.username != username: - return "Unauthorized", 401 + return jsonify({"success": False, "error": "Unauthorized"}), 401 current_setting = session.get(f'ignore_index_{username}', False) - session[f'ignore_index_{username}'] = not current_setting + new_setting = not current_setting + session[f'ignore_index_{username}'] = new_setting - return redirect(url_for('user_files', username=username)) + return jsonify({"success": True, "ignore_index": new_setting}) @app.route('/dash//upload', methods=['POST']) @login_required @@ -436,15 +438,23 @@ def upload_user_file(username): file.save(file_path) return redirect(url_for('user_files', username=username, subpath=subpath)) -@app.route('/dash//delete/', methods=['POST']) +@app.route('/dash//delete/', methods=['POST']) @login_required def delete_user_file(username, filename): if current_user.username != username: return "Unauthorized", 401 file_path = os.path.join(app.config['UPLOAD_FOLDER'], username, filename) - if os.path.exists(file_path): - os.remove(file_path) - return redirect(url_for('user_files', username=username)) + try: + if os.path.exists(file_path): + if os.path.isfile(file_path): + os.remove(file_path) + elif os.path.isdir(file_path): + shutil.rmtree(file_path) + return redirect(url_for('user_files', username=username)) + except PermissionError: + return "Permission denied: Unable to delete the file or folder", 403 + except Exception as e: + return f"An error occurred: {str(e)}", 500 @app.route('/dash//rename', methods=['POST']) @login_required @@ -554,5 +564,314 @@ def debug_users(): users = cursor.fetchall() return jsonify(users) +@app.route('/upload/pastebin', methods=['POST']) +def upload_pastebin(): + try: + data = request.get_json() + if not data or 'content' not in data: + return jsonify({'success': False, 'error': 'Content is required'}), 400 + + content = data['content'] + vanity = shortuuid.uuid()[:8] + + user_id = current_user.id if current_user.is_authenticated else None + + db = get_db() + cursor = db.cursor() + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity, 'pastebin', content, datetime.now(), user_id)) + db.commit() + + short_url = url_for('redirect_vanity', vanity=vanity, _external=True) + deletion_url = url_for('delete_content', vanity=vanity, _external=True) + return jsonify({'success': True, 'vanity': vanity, 'url': short_url, 'deletion_url': deletion_url}), 200 + except Exception as e: + print("Exception occurred:", str(e)) + return jsonify({'success': False, 'error': str(e)}), 400 + +@app.route('/upload/file', methods=['POST']) +def upload_file(): + if 'file' not in request.files: + return jsonify({'success': False, 'error': 'No file part'}), 400 + file = request.files['file'] + if file.filename == '': + return jsonify({'success': False, 'error': 'No selected file'}), 400 + if file: + filename = secure_filename(file.filename) + extension = os.path.splitext(filename)[1].lower() + vanity = shortuuid.uuid()[:8] + vanity_with_extension = f"{vanity}{extension}" + new_filename = vanity_with_extension + file_path = os.path.join(app.config['UPLOAD_FOLDER'], new_filename) + file.save(file_path) + + user_id = current_user.id if current_user.is_authenticated else None + + db = get_db() + cursor = db.cursor() + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity_with_extension, 'file', new_filename, datetime.now(), user_id)) + db.commit() + + short_url = url_for('redirect_vanity', vanity=vanity_with_extension, _external=True) + deletion_url = url_for('delete_content', vanity=vanity_with_extension, _external=True) + return jsonify({'success': True, 'vanity': vanity_with_extension, 'url': short_url, 'deletion_url': deletion_url, 'filename': new_filename}), 200 + +@app.route('/shorten', methods=['POST']) +def shorten_url(): + try: + data = request.get_json() + if not data or 'url' not in data: + return jsonify({'success': False, 'error': 'URL is required'}), 400 + + long_url = data['url'] + vanity = shortuuid.uuid()[:8] + + user_id = current_user.id if current_user.is_authenticated else None + + db = get_db() + cursor = db.cursor() + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity, 'url', long_url, datetime.now(), user_id)) + db.commit() + + short_url = url_for('redirect_vanity', vanity=vanity, _external=True) + return jsonify({'success': True, 'vanity': vanity, 'short_url': short_url}), 200 + except Exception as e: + print("Exception occurred:", str(e)) + return jsonify({'success': False, 'error': str(e)}), 400 + +@app.route('/edit/content/', methods=['GET', 'POST']) +@login_required +def edit_content(vanity): + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT * FROM content WHERE vanity = ? OR data LIKE ?", (vanity, f"{vanity}%")) + content = cursor.fetchone() + + if not content or content[4] != current_user.id: + return jsonify({'success': False, 'error': 'Unauthorized'}), 401 + + content_type, content_data = content[1], content[2] + + if request.method == 'POST': + new_content = request.form.get('content') + if new_content is not None: + cursor.execute("UPDATE content SET data = ? WHERE vanity = ?", (new_content, content[0])) + db.commit() + return redirect(url_for('redirect_vanity', vanity=content[0])) + + if content_type == 'file': + file_path = os.path.join(app.config['UPLOAD_FOLDER'], content_data) + if os.path.exists(file_path): + with open(file_path, 'r') as file: + file_content = file.read() + return render_template('edit_content.html', content=file_content, vanity=content[0], content_type=content_type) + elif content_type == 'pastebin': + return render_template('edit_content.html', content=content_data, vanity=content[0], content_type=content_type) + + return jsonify({'success': False, 'error': 'Unsupported content type for editing'}), 400 + +@app.route('/delete/content/', methods=['POST']) +@login_required +def delete_content(vanity): + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT * FROM content WHERE vanity = ? OR data LIKE ?", (vanity, f"{vanity}%")) + content = cursor.fetchone() + + if not content or content[4] != current_user.id: + return jsonify({'success': False, 'error': 'Unauthorized'}), 401 + + cursor.execute("DELETE FROM content WHERE vanity = ? OR data LIKE ?", (vanity, f"{vanity}%")) + db.commit() + + # If it's a file, delete the actual file from the filesystem + if content[1] == 'file': + file_path = os.path.join(app.config['UPLOAD_FOLDER'], content[2]) + if os.path.exists(file_path): + os.remove(file_path) + + return jsonify({'success': True}), 200 + +@app.route('//info') +def content_info(vanity): + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT * FROM content WHERE vanity = ?", (vanity,)) + content = cursor.fetchone() + + if content: + content_type, content_data, created_at, user_id = content[1], content[2], content[3], content[4] + + username = get_username(user_id) + + file_size = None + is_media = False + if content_type == 'file': + file_path = os.path.join(app.config['UPLOAD_FOLDER'], content_data) + if os.path.exists(file_path): + file_size = os.path.getsize(file_path) + file_extension = os.path.splitext(content_data)[1].lower() + is_media = file_extension in ['.jpg', '.jpeg', '.png', '.gif', '.mp3', '.wav', '.mp4', '.webm'] + + info = { + 'type': content_type, + 'vanity': content_data if content_type == 'file' else vanity, + 'data': content_data, + 'created_at': created_at, + 'username': username, + 'file_size': file_size, + 'is_media': is_media + } + + return render_template('content_info.html', info=info) + + return render_template('404.html'), 404 + +@app.route('/sharex-config') +@login_required +def generate_sharex_config(): + base_url = request.url_root.rstrip('/') + config = { + "Version": "13.7.0", + "Name": "aCloud", + "DestinationType": "ImageUploader, TextUploader, FileUploader, URLShortener", + "RequestMethod": "POST", + "RequestURL": f"{base_url}/api/upload", + "Headers": { + "X-API-Key": current_user.api_key + }, + "Body": "MultipartFormData", + "FileFormName": "file", + "TextFormName": "text", + "URLShortenerFormName": "url", + "URL": "$json:url$", + "DeletionURL": "$json:deletion_url$" + } + + response = make_response(json.dumps(config, indent=2)) + response.headers.set('Content-Type', 'application/json') + response.headers.set('Content-Disposition', 'attachment', filename='aCloud_ShareX.sxcu') + return response + +@app.route('/api/upload', methods=['POST']) +def api_upload(): + api_key = request.headers.get('X-API-Key') + if not api_key: + return jsonify({'error': 'API key is missing'}), 401 + + db = get_db() + cursor = db.cursor() + cursor.execute("SELECT * FROM users WHERE api_key = ?", (api_key,)) + user = cursor.fetchone() + + if not user: + return jsonify({'error': 'Invalid API key'}), 401 + + if 'file' in request.files: + file = request.files['file'] + if file.filename == '': + return jsonify({'error': 'No selected file'}), 400 + if file: + filename = secure_filename(file.filename) + extension = os.path.splitext(filename)[1].lower() + + if extension == '.txt': + # Handle text files as pastebins + content = file.read().decode('utf-8') + vanity = shortuuid.uuid()[:8] + + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity, 'pastebin', content, datetime.now(), user[0])) + db.commit() + + url = url_for('redirect_vanity', vanity=vanity, _external=True) + delete_url = url_for('delete_content', vanity=vanity, _external=True) + return jsonify({ + 'status': 'success', + 'url': url, + 'deletion_url': delete_url, + }) + else: + # Handle other file types + vanity = shortuuid.uuid()[:8] + new_filename = f"{vanity}{extension}" + file_path = os.path.join(app.config['UPLOAD_FOLDER'], new_filename) + file.save(file_path) + + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (new_filename, 'file', new_filename, datetime.now(), user[0])) + db.commit() + + url = url_for('redirect_vanity', vanity=new_filename, _external=True) + delete_url = url_for('delete_content', vanity=new_filename, _external=True) + return jsonify({ + 'status': 'success', + 'url': url, + 'deletion_url': delete_url, + }) + elif 'text' in request.form: + content = request.form['text'] + vanity = shortuuid.uuid()[:8] + + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity, 'pastebin', content, datetime.now(), user[0])) + db.commit() + + url = url_for('redirect_vanity', vanity=vanity, _external=True) + delete_url = url_for('delete_content', vanity=vanity, _external=True) + return jsonify({ + 'status': 'success', + 'url': url, + 'deletion_url': delete_url, + }) + elif 'url' in request.form: + long_url = request.form['url'] + vanity = shortuuid.uuid()[:8] + + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity, 'url', long_url, datetime.now(), user[0])) + db.commit() + + short_url = url_for('redirect_vanity', vanity=vanity, _external=True) + delete_url = url_for('delete_content', vanity=vanity, _external=True) + return jsonify({ + 'status': 'success', + 'url': short_url, + 'deletion_url': delete_url, + }) + + return jsonify({'error': 'No file, text, or URL content provided'}), 400 + +@app.route('/dash//create_file', methods=['POST']) +@login_required +def create_new_file(username): + if current_user.username != username: + return "Unauthorized", 401 + subpath = request.form.get('subpath', '').rstrip('/') + file_name = secure_filename(request.form['file_name']) + file_path = os.path.join(app.config['UPLOAD_FOLDER'], username, subpath, file_name) + + if not os.path.exists(os.path.dirname(file_path)): + os.makedirs(os.path.dirname(file_path)) + + if not os.path.exists(file_path): + with open(file_path, 'w') as f: + f.write('') # Create an empty file + + # Add the file to the content database + db = get_db() + cursor = db.cursor() + vanity = shortuuid.uuid()[:8] + cursor.execute("INSERT INTO content (vanity, type, data, created_at, user_id) VALUES (?, ?, ?, ?, ?)", + (vanity, 'file', os.path.join(subpath, file_name), datetime.now(), current_user.id)) + db.commit() + flash('File created successfully.', 'success') + else: + flash('File already exists.', 'error') + + return redirect(url_for('user_files', username=username, subpath=subpath)) + if __name__ == '__main__': - app.run(debug=True) + app.run(debug=True,host='0.0.0.0',port=7123) \ No newline at end of file diff --git a/schema.sql b/schema.sql index 8af69c3..bb9a182 100644 --- a/schema.sql +++ b/schema.sql @@ -2,12 +2,14 @@ CREATE TABLE IF NOT EXISTS content ( vanity TEXT PRIMARY KEY, type TEXT NOT NULL, data TEXT NOT NULL, - created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP, - user_id INTEGER + created_at TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, + user_id INTEGER, + FOREIGN KEY (user_id) REFERENCES users (id) ); CREATE TABLE IF NOT EXISTS users ( id INTEGER PRIMARY KEY AUTOINCREMENT, username TEXT UNIQUE NOT NULL, - password_hash TEXT NOT NULL + password_hash TEXT NOT NULL, + api_key TEXT ); \ No newline at end of file diff --git a/templates/content.html b/templates/content.html index fdb09ef..2b2f7ac 100644 --- a/templates/content.html +++ b/templates/content.html @@ -86,6 +86,8 @@ /* Ensure proper contrast for syntax highlighting */ .highlight pre { color: var(--text-color); + white-space: pre-wrap; + word-wrap: break-word; } /* Override Pygments styles for dark mode */ @@ -155,46 +157,64 @@

Content

- {% if content.user_id %} + {% if content %}

Uploaded by: {{ content.username }}

+

Created at: {{ content.created_at }}

+ + {% if highlighted_content %} + +
+ {{ highlighted_content|safe }} +
+ + View Raw + {% if current_user.is_authenticated and current_user.id == content.user_id %} + Edit + {% endif %} + {% elif url %} +

Shortened URL: {{ url }}

+ + {% elif raw_content %} +
{{ raw_content }}
+ + View Raw + {% if current_user.is_authenticated and current_user.id == content.user_id %} + Edit + {% endif %} + {% else %} +

No content available

+ {% endif %} + + {% if current_user.is_authenticated and current_user.id == content.user_id %} +
+ +
+ {% endif %} {% else %} -

Uploaded by: Anonymous

- {% endif %} -

Created at: {{ content.created_at }}

- - {% if highlighted_content %} - - {{ highlighted_content|safe }} - {% elif url %} -

Shortened URL: {{ url }}

- {% else %} -
{{ raw_content }}
- {% endif %} - - {% if current_user.is_authenticated and current_user.id == content.user_id %} - Edit -
- -
+

No content found.

{% endif %}
-

Edit Content

+

Edit Content

- -
- +
{{ content }}
+ +
- Back to Dashboard
+ \ No newline at end of file diff --git a/templates/index.html b/templates/index.html index e81a1ed..3f29c66 100644 --- a/templates/index.html +++ b/templates/index.html @@ -128,7 +128,6 @@
-
@@ -148,14 +147,6 @@
-
-

Upload Folder

-
- - -
-
-

Shorten URL

@@ -252,48 +243,45 @@ fetch('/upload/pastebin', { method: 'POST', headers: { - 'Content-Type': 'application/x-www-form-urlencoded', + 'Content-Type': 'application/json', }, - body: new URLSearchParams({ content }), + body: JSON.stringify({ content: content }), }) .then(response => response.json()) .then(data => { - document.getElementById('textResult').innerHTML = `Text uploaded. Access it here.`; + if (data.success) { + const simpleUrl = `${window.location.origin}/${data.vanity}`; + document.getElementById('textResult').innerHTML = `Pastebin created. Access it ${simpleUrl}`; + } else { + document.getElementById('textResult').innerHTML = `Error: ${data.error}`; + } + }) + .catch(error => { + console.error('Error:', error); + document.getElementById('textResult').innerHTML = `An error occurred: ${error}`; }); } function uploadFile() { const formData = new FormData(); - formData.append('file', document.querySelector('#fileForm input[type="file"]').files[0]); + const fileInput = document.querySelector('#fileForm input[type="file"]'); + formData.append('file', fileInput.files[0]); fetch('/upload/file', { method: 'POST', body: formData, }) .then(response => response.json()) .then(data => { - document.getElementById('fileResult').innerHTML = `File uploaded. Download it here.`; - }); - } - - function uploadFolder() { - const files = document.querySelector('#folderForm input[type="file"]').files; - if (files.length === 0) { - alert('Please select a folder.'); - return; - } - - const formData = new FormData(); - for (const file of files) { - formData.append('file', file); - } - - fetch('/upload/folder', { - method: 'POST', - body: formData, + if (data.success) { + const simpleUrl = `${window.location.origin}/${data.vanity}`; + document.getElementById('fileResult').innerHTML = `File uploaded. Access it ${simpleUrl}`; + } else { + document.getElementById('fileResult').innerHTML = `Error: ${data.error}`; + } }) - .then(response => response.json()) - .then(data => { - document.getElementById('folderResult').innerHTML = `Folder uploaded. View its contents here.`; + .catch(error => { + console.error('Error:', error); + document.getElementById('fileResult').innerHTML = `An error occurred: ${error}`; }); } @@ -302,13 +290,21 @@ fetch('/shorten', { method: 'POST', headers: { - 'Content-Type': 'application/x-www-form-urlencoded', + 'Content-Type': 'application/json', }, - body: new URLSearchParams({ url }), + body: JSON.stringify({ url: url }), }) .then(response => response.json()) .then(data => { - document.getElementById('urlResult').innerHTML = `URL shortened. Access it here.`; + if (data.success) { + document.getElementById('urlResult').innerHTML = `URL shortened. Access it ${data.short_url}`; + } else { + document.getElementById('urlResult').innerHTML = `Error: ${data.error}`; + } + }) + .catch(error => { + console.error('Error:', error); + document.getElementById('urlResult').innerHTML = `An error occurred: ${error}`; }); } diff --git a/templates/login.html b/templates/login.html index 9595b92..2fdaed1 100644 --- a/templates/login.html +++ b/templates/login.html @@ -3,66 +3,99 @@ - Login + Login - aCloud -