Skip to content

Latest commit

 

History

History
189 lines (166 loc) · 10.2 KB

ReadMe.md

File metadata and controls

189 lines (166 loc) · 10.2 KB

Anime Atsume (back-end)

Simple search engine for anime that includes links for watching episodes

Notes:

General

  • To run on specific port, server.port option is necessary, either in application.properties or via command line
    • CMD: java -jar anime-atsume.war --server.port=80
    • Port 80 for HTTP, 443 for HTTPS
    • Usually, running on ports 80 or 443 will require sudo
  • If running in a hosted VM, use the nohup command to allow the process to run even after your SSH session has ended. Make sure to run in background
    • e.g. nohup java -jar anime-atsume.war --server.port=80 &
  • DNS types.

Docker

  • Build image: docker build -t app .
    • Builds new image named "app" using the Dockerfile at root dir "."
    • Needs to be run in same directory where the Dockerfile is (in this case, server/).
  • Run app in (new) image: docker run -p 8080:8080 app
    • docker run is a combo of docker create and docker start
      • Only run supports exposing ports, unless EXPOSE is in the Dockerfile
    • Runs the newly built image, "app"
    • -p host_port:container_port
    • Add env vars read in CMD line via -e flag
      • docker run -e PORT=3000 -p 80:3000 app
  • Run bash in image: docker run -it app /bin/bash
  • Notes on Dockerfile:
    • It renames the .war file to anime-atsume.war so version number isn't needed (and CMD can just be this renamed filename).
    • Since ports are mapped in docker run, there's no need to force --server.port=80.
    • SQLite3 was added as an installation dependency (since it's not guaranteed that the docker image will support it out of the box).
    • Allow DB to be writeable via chmod.
  • Other docker commands:
    • List images or running containers: docker [container|image] ls
    • List running containers: docker ps
    • New bash terminal on running container: docker exec -it <id_from_ps> /bin/bash
    • Delete image: docker rmi <image_id>
    • Stop container: docker stop <container_id>
    • Stop all containers: docker stop $(docker container ls -q)

Deploying to AWS

  • Create IAM user:
    • Name: root (arbitrary).
    • Check "Provide user access to the AWS Management Console" box.
    • Uncheck "Users must create a new password at next sign-in" box.
    • "I want to create an IAM user" (not "Specify a user in Identity Center").
    • Finish.
  • Modify IAM user:
    • Navigate to: IAM --> Users --> root --> Security Credentials.
    • Create access key.
    • CLI.
    • Copy access key ID and secret access key.
  • Grant IAM user permissions:
    • Create a group with these permissions:
      • AdministratorAccess
      • AmazonAPIGatewayAdministrator
      • AmazonAPIGatewayPushToCloudWatchLogs
      • AmazonAppFlowFullAccess
      • Optional:
      • AdministratorAccess-Amplify
      • AdministratorAccess-AWSElasticBeanstalk
      • AmazonAppStreamFullAccess
      • AmazonChimeSDK
      • AmazonCloudDirectoryFullAccess
      • AmazonAppFlowReadOnlyAccess
      • AmazonAthenaFullAccess
      • AmazonAugmentedAIFullAccess
      • AmazonBraketFullAccess
      • AmazonBraketJobsExecutionPolicy
  • Launch an EC2 instance.
    • Name: web-server (arbitrary).
    • Quick Start: Search for "Ubuntu".
      • Alternative: Amazon Linux 2023 AMI (Fedora).
    • Instance type: t2.micro (may change in the future).
    • Key pair:
      • Create new pair.
      • Name: aws-ssh-key.
      • Type: RSA.
      • Key file format: .pem.
    • Network Settings:
      • Create security group.
      • Allow SSH, HTTP, and HTTPS traffic from anywhere.
    • Advanced details: Leave everything as default.
  • Log in using AWS CLI:
    • aws configure.
  • SSH into the server and install Docker:
    • (Optional) Remove previous Docker packages: sudo apt-get remove docker docker-engine docker.io containerd runc
    • Add GPG key (Debian): curl -fsSL https://download.docker.com/linux/debian/gpg | sudo apt-key add -
    • Add the repository source: sudo add-apt-repository "deb [arch=amd64] https://download.docker.com/linux/ubuntu $(lsb_release -cs) stable"
    • Add the user to the Docker group so it can run commands without root:
      • sudo groupadd docker
      • sudo usermod -aG docker $(whoami)
    • (Optional) Enable Docker service:
      • systemctl start docker
      • systemctl enable docker
    • Log out to reflect changes.

Deploying to Heroku

  • General
    • View logs:
      • heroku logs -n <num_lines> view latest n lines.
      • heroku logs -t tails logs live.
    • If you want heroku to log your process' memory usage stats, run heroku labs:enable log-runtime-metrics and re-deploy. Disable with labs:disable.
    • Use Kaffeine or cron-job to keep the app awake forever (pings the specified URL every so often).
      • Adding a credit card bumps the number of awake hours for your account(all apps) up from 550 hours/month to 1000, which is plenty to keep a single app running forever.
    • List apps: heroku apps
    • Stop an app/dyno: heroku ps:stop "appname"
  • Heroku doesn't know how to handle nested folders.
    • Since the front-end is built to the back-end's resources/ directory, we need only deploy the server/ folder.
    • git subtree push --prefix server/ heroku master
  • Heroku doesn't have any JavaFX support.
    • UI4J uses JavaFX so we need to add support manually.
    • Add buildpacks so Herokou finds all necessary classes in its pre-commit hook:
      • heroku buildpacks:add -i 1 https://github.com/jkutner/heroku-buildpack-javafx - add JavaFX to build path.
      • heroku buildpacks:add heroku/gradle - default gradle/Spring build process.
    • Alternatively (but possibly still mandatory), add JavaFX as a dependency in build.gradle.
  • For deploying war file:
    • Make sure to include the .db file with --includes anime_atsume.db
    • heroku plugins:install java
    • heroku war:deploy <path_to_war_file> --app <app_name> --includes anime_atsume.db
  • For deploying with Dockerfile
    • Luckily, Heroku doesn't need you to build the image, nor to push the subtree, nor do anything else to deploy the container. It will build the image using any required files from your filesystem (e.g. .war, .db) and push it to its own container registry.
      • Note: You might need server.port=${PORT} in application.properties so Heroku's PORT var will overwrite Spring's port.
      • Likely not needed since Spring and Heroku both default to port 8080.
    • Make sure you're in the server/ directory. Then run:
      • heroku container:login
      • heroku container:push web (depending on if you set this app as the default Heroku app, you'll need to add -a anime-atsume to specify which app to update in the push)
      • heroku container:release web (same as above with -a flag)
      • Final command: ./gradlew clean build && heroku container:push web -a anime-atsume && heroku container:release web -a anime-atsume
    • Heroku's "free" and "hobby" tiers only allow a maximum of 512 MB.
      • Force JVM to not exceed this cap by adding -Xmx512m to app run.
      • -Xmx is to set maximum memory usage, m specifies megabytes.
      • For Heroku, set JAVA_OPTS=-Xmx512m in App > Settings > Config Vars.
        • Dockerfile's CMD line applies JAVA_OPTS from the -e flag passed to docker run.
        • Heroku will pass any Config Vars to docker run, e.g. docker run -e JAVA_OPTS=$JAVA_OPTS.
        • JAVA_OPTS doesn't have to be set to run the container since CMD is in 'shell' form instead of 'exec' form.
      • Since Heroku's memory cap is a soft limit (will only kill the app if you go too much above the limit), -Xmx640m will likely work, giving the app more memory without triggering Heroku's kill switch.

Deploying to gcloud

  • If all else fails, rewrite code with Selenium and allow it via this method
  • Only deploying .war file has worked so far.
    • Requires app.yaml file to be present in the root dir of where you're deploying
      • Will mean app.yaml needs to be in same dir as .war file:
      runtime: java
      env: flex
      
        handlers:
        - url: /.*
          script: this field is required, but ignored
      
    • Deploy (in server/build/libs/): gcloud app deploy app.yaml
    • Logs: gcloud app logs tail -s default
    • Open in browser: gcloud app browse
  • Cloud Compute (VM instance)
    • Needed:
      • Java 8: sudo apt install openjdk-8-jre
      • JavaFX: sudo apt install openjfx=8u161-b12-1ubuntu2 libopenjfx-jni=8u161-b12-1ubuntu2 libopenjfx-java=8u161-b12-1ubuntu2
        • Didn't find the solution to run with monocle.platform=Headless until after installing openjfx
        • Gotten from stackoverflow
        • Make sure to prevent them from updating: sudo apt-mark hold openjfx libopenjfx-jni libopenjfx-java
      • Alternatively, try full JDK+JFX packages:
    • Not sure which of the following are needed, if any at all:
      • sudo apt install xvfb
      • sudo apt install xorg libgtk2.0-0
    • Run with headless mode:
      • java -Dglass.platform=Monocle -Dmonocle.platform=Headless -jar anime-atsume-1.0.war
    • Final command to run with headless mode, port 80, and as background process:
      • sudo nohup java -Dglass.platform=Monocle -Dmonocle.platform=Headless -jar anime-atsume-1.0.war --server.port=80 &