123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- """Building the app and initializing all prerequisites."""
- from __future__ import annotations
- import json
- import os
- import random
- import subprocess
- from pathlib import Path
- from typing import Optional, Union
- from rich.progress import Progress
- from reflex import constants
- from reflex.config import get_config
- from reflex.utils import path_ops, prerequisites
- from reflex.utils.processes import new_process
- def update_json_file(file_path: str, update_dict: dict[str, Union[int, str]]):
- """Update the contents of a json file.
- Args:
- file_path: the path to the JSON file.
- update_dict: object to update json.
- """
- fp = Path(file_path)
- # create file if it doesn't exist
- fp.touch(exist_ok=True)
- # create an empty json object if file is empty
- fp.write_text("{}") if fp.stat().st_size == 0 else None
- with open(fp) as f: # type: ignore
- json_object: dict = json.load(f)
- json_object.update(update_dict)
- with open(fp, "w") as f:
- json.dump(json_object, f, ensure_ascii=False)
- def set_reflex_project_hash():
- """Write the hash of the Reflex project to a REFLEX_JSON."""
- update_json_file(constants.REFLEX_JSON, {"project_hash": random.getrandbits(128)})
- def set_environment_variables():
- """Write the upload url to a REFLEX_JSON."""
- update_json_file(
- constants.ENV_JSON,
- {
- "uploadUrl": constants.Endpoint.UPLOAD.get_url(),
- "eventUrl": constants.Endpoint.EVENT.get_url(),
- "pingUrl": constants.Endpoint.PING.get_url(),
- },
- )
- def set_os_env(**kwargs):
- """Set os environment variables.
- Args:
- kwargs: env key word args.
- """
- for key, value in kwargs.items():
- if not value:
- continue
- os.environ[key.upper()] = value
- def generate_sitemap(deploy_url: str):
- """Generate the sitemap config file.
- Args:
- deploy_url: The URL of the deployed app.
- """
- # Import here to avoid circular imports.
- from reflex.compiler import templates
- config = json.dumps(
- {
- "siteUrl": deploy_url,
- "generateRobotsTxt": True,
- }
- )
- with open(constants.SITEMAP_CONFIG_FILE, "w") as f:
- f.write(templates.SITEMAP_CONFIG(config=config))
- def export_app(
- backend: bool = True,
- frontend: bool = True,
- zip: bool = False,
- deploy_url: Optional[str] = None,
- loglevel: constants.LogLevel = constants.LogLevel.ERROR,
- ):
- """Zip up the app for deployment.
- Args:
- backend: Whether to zip up the backend app.
- frontend: Whether to zip up the frontend app.
- zip: Whether to zip the app.
- deploy_url: The URL of the deployed app.
- loglevel: The log level to use.
- """
- # Remove the static folder.
- path_ops.rm(constants.WEB_STATIC_DIR)
- # Generate the sitemap file.
- if deploy_url is not None:
- generate_sitemap(deploy_url)
- # Create a progress object
- progress = Progress()
- # Add a single task to the progress object
- task = progress.add_task("Building app... ", total=500)
- # Start the subprocess with the progress bar.
- with progress, new_process(
- [prerequisites.get_package_manager(), "run", "export"],
- cwd=constants.WEB_DIR,
- ) as export_process:
- assert export_process.stdout is not None, "No stdout for export process."
- for line in export_process.stdout:
- # Print the line in debug mode.
- if loglevel == constants.LogLevel.DEBUG:
- print(line, end="")
- # Check for special strings and update the progress bar.
- if "Linting and checking " in line:
- progress.update(task, advance=100)
- elif "Compiled successfully" in line:
- progress.update(task, advance=100)
- elif "Route (pages)" in line:
- progress.update(task, advance=100)
- elif "automatically rendered as static HTML" in line:
- progress.update(task, advance=100)
- elif "Export successful" in line:
- progress.update(task, completed=500)
- break # Exit the loop if the completion message is found
- print("Export process completed.")
- # Zip up the app.
- if zip:
- if os.name == "posix":
- posix_export(backend, frontend)
- if os.name == "nt":
- nt_export(backend, frontend)
- def nt_export(backend: bool = True, frontend: bool = True):
- """Export for nt (Windows) systems.
- Args:
- backend: Whether to zip up the backend app.
- frontend: Whether to zip up the frontend app.
- """
- cmd = r""
- if frontend:
- cmd = r'''powershell -Command "Set-Location .web/_static; Compress-Archive -Path .\* -DestinationPath ..\..\frontend.zip -Force"'''
- os.system(cmd)
- if backend:
- cmd = r'''powershell -Command "Get-ChildItem -File | Where-Object { $_.Name -notin @('.web', 'assets', 'frontend.zip', 'backend.zip') } | Compress-Archive -DestinationPath backend.zip -Update"'''
- os.system(cmd)
- def posix_export(backend: bool = True, frontend: bool = True):
- """Export for posix (Linux, OSX) systems.
- Args:
- backend: Whether to zip up the backend app.
- frontend: Whether to zip up the frontend app.
- """
- cmd = r""
- if frontend:
- cmd = r"cd .web/_static && zip -r ../../frontend.zip ./*"
- os.system(cmd)
- if backend:
- cmd = r"zip -r backend.zip ./* -x .web/\* ./assets\* ./frontend.zip\* ./backend.zip\*"
- os.system(cmd)
- def setup_frontend(
- root: Path,
- loglevel: constants.LogLevel = constants.LogLevel.ERROR,
- disable_telemetry: bool = True,
- ):
- """Set up the frontend.
- Args:
- root: The root path of the project.
- loglevel: The log level to use.
- disable_telemetry: Whether to disable the Next telemetry.
- """
- # Validate bun version.
- prerequisites.validate_and_install_bun(initialize=False)
- # Initialize the web directory if it doesn't exist.
- web_dir = prerequisites.create_web_directory(root)
- # Install frontend packages.
- prerequisites.install_frontend_packages(web_dir)
- # Copy asset files to public folder.
- path_ops.cp(
- src=str(root / constants.APP_ASSETS_DIR),
- dest=str(root / constants.WEB_ASSETS_DIR),
- )
- # set the environment variables in client(env.json)
- set_environment_variables()
- # Disable the Next telemetry.
- if disable_telemetry:
- new_process(
- [
- prerequisites.get_package_manager(),
- "run",
- "next",
- "telemetry",
- "disable",
- ],
- cwd=constants.WEB_DIR,
- stdout=subprocess.DEVNULL,
- )
- def setup_frontend_prod(
- root: Path,
- loglevel: constants.LogLevel = constants.LogLevel.ERROR,
- disable_telemetry: bool = True,
- ):
- """Set up the frontend for prod mode.
- Args:
- root: The root path of the project.
- loglevel: The log level to use.
- disable_telemetry: Whether to disable the Next telemetry.
- """
- setup_frontend(root, loglevel, disable_telemetry)
- export_app(loglevel=loglevel)
- def setup_backend():
- """Set up backend.
- Specifically ensures backend database is updated when running --no-frontend.
- """
- # Import here to avoid circular imports.
- from reflex.model import Model
- config = get_config()
- if config.db_url is not None:
- Model.create_all()
|