first
This commit is contained in:
commit
e5878ea429
230 changed files with 5673 additions and 0 deletions
51
.dockerignore
Normal file
51
.dockerignore
Normal file
|
|
@ -0,0 +1,51 @@
|
||||||
|
# See https://docs.docker.com/engine/reference/builder/#dockerignore-file for more about ignoring files.
|
||||||
|
|
||||||
|
# Ignore git directory.
|
||||||
|
/.git/
|
||||||
|
/.gitignore
|
||||||
|
|
||||||
|
# Ignore bundler config.
|
||||||
|
/.bundle
|
||||||
|
|
||||||
|
# Ignore all environment files.
|
||||||
|
/.env*
|
||||||
|
|
||||||
|
# Ignore all default key files.
|
||||||
|
/config/master.key
|
||||||
|
/config/credentials/*.key
|
||||||
|
|
||||||
|
# Ignore all logfiles and tempfiles.
|
||||||
|
/log/*
|
||||||
|
/tmp/*
|
||||||
|
!/log/.keep
|
||||||
|
!/tmp/.keep
|
||||||
|
|
||||||
|
# Ignore pidfiles, but keep the directory.
|
||||||
|
/tmp/pids/*
|
||||||
|
!/tmp/pids/.keep
|
||||||
|
|
||||||
|
# Ignore storage (uploaded files in development and any SQLite databases).
|
||||||
|
/storage/*
|
||||||
|
!/storage/.keep
|
||||||
|
/tmp/storage/*
|
||||||
|
!/tmp/storage/.keep
|
||||||
|
|
||||||
|
# Ignore assets.
|
||||||
|
/node_modules/
|
||||||
|
/app/assets/builds/*
|
||||||
|
!/app/assets/builds/.keep
|
||||||
|
/public/assets
|
||||||
|
|
||||||
|
# Ignore CI service files.
|
||||||
|
/.github
|
||||||
|
|
||||||
|
# Ignore Kamal files.
|
||||||
|
/config/deploy*.yml
|
||||||
|
/.kamal
|
||||||
|
|
||||||
|
# Ignore development files
|
||||||
|
/.devcontainer
|
||||||
|
|
||||||
|
# Ignore Docker-related files
|
||||||
|
/.dockerignore
|
||||||
|
/Dockerfile*
|
||||||
9
.gitattributes
vendored
Normal file
9
.gitattributes
vendored
Normal file
|
|
@ -0,0 +1,9 @@
|
||||||
|
# See https://git-scm.com/docs/gitattributes for more about git attribute files.
|
||||||
|
|
||||||
|
# Mark the database schema as having been generated.
|
||||||
|
db/schema.rb linguist-generated
|
||||||
|
|
||||||
|
# Mark any vendored files as having been vendored.
|
||||||
|
vendor/* linguist-vendored
|
||||||
|
config/credentials/*.yml.enc diff=rails_credentials
|
||||||
|
config/credentials.yml.enc diff=rails_credentials
|
||||||
12
.github/dependabot.yml
vendored
Normal file
12
.github/dependabot.yml
vendored
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: bundler
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: weekly
|
||||||
|
open-pull-requests-limit: 10
|
||||||
|
- package-ecosystem: github-actions
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: weekly
|
||||||
|
open-pull-requests-limit: 10
|
||||||
124
.github/workflows/ci.yml
vendored
Normal file
124
.github/workflows/ci.yml
vendored
Normal file
|
|
@ -0,0 +1,124 @@
|
||||||
|
name: CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
push:
|
||||||
|
branches: [ main ]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
scan_ruby:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Set up Ruby
|
||||||
|
uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
bundler-cache: true
|
||||||
|
|
||||||
|
- name: Scan for common Rails security vulnerabilities using static analysis
|
||||||
|
run: bin/brakeman --no-pager
|
||||||
|
|
||||||
|
- name: Scan for known security vulnerabilities in gems used
|
||||||
|
run: bin/bundler-audit
|
||||||
|
|
||||||
|
scan_js:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Set up Ruby
|
||||||
|
uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
bundler-cache: true
|
||||||
|
|
||||||
|
- name: Scan for security vulnerabilities in JavaScript dependencies
|
||||||
|
run: bin/importmap audit
|
||||||
|
|
||||||
|
lint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
RUBOCOP_CACHE_ROOT: tmp/rubocop
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Set up Ruby
|
||||||
|
uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
bundler-cache: true
|
||||||
|
|
||||||
|
- name: Prepare RuboCop cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
env:
|
||||||
|
DEPENDENCIES_HASH: ${{ hashFiles('.ruby-version', '**/.rubocop.yml', '**/.rubocop_todo.yml', 'Gemfile.lock') }}
|
||||||
|
with:
|
||||||
|
path: ${{ env.RUBOCOP_CACHE_ROOT }}
|
||||||
|
key: rubocop-${{ runner.os }}-${{ env.DEPENDENCIES_HASH }}-${{ github.ref_name == github.event.repository.default_branch && github.run_id || 'default' }}
|
||||||
|
restore-keys: |
|
||||||
|
rubocop-${{ runner.os }}-${{ env.DEPENDENCIES_HASH }}-
|
||||||
|
|
||||||
|
- name: Lint code for consistent style
|
||||||
|
run: bin/rubocop -f github
|
||||||
|
|
||||||
|
test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
# services:
|
||||||
|
# redis:
|
||||||
|
# image: valkey/valkey:8
|
||||||
|
# ports:
|
||||||
|
# - 6379:6379
|
||||||
|
# options: --health-cmd "redis-cli ping" --health-interval 10s --health-timeout 5s --health-retries 5
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Set up Ruby
|
||||||
|
uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
bundler-cache: true
|
||||||
|
|
||||||
|
- name: Run tests
|
||||||
|
env:
|
||||||
|
RAILS_ENV: test
|
||||||
|
# RAILS_MASTER_KEY: ${{ secrets.RAILS_MASTER_KEY }}
|
||||||
|
# REDIS_URL: redis://localhost:6379/0
|
||||||
|
run: bin/rails db:test:prepare test
|
||||||
|
|
||||||
|
system-test:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
# services:
|
||||||
|
# redis:
|
||||||
|
# image: valkey/valkey:8
|
||||||
|
# ports:
|
||||||
|
# - 6379:6379
|
||||||
|
# options: --health-cmd "redis-cli ping" --health-interval 10s --health-timeout 5s --health-retries 5
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Set up Ruby
|
||||||
|
uses: ruby/setup-ruby@v1
|
||||||
|
with:
|
||||||
|
bundler-cache: true
|
||||||
|
|
||||||
|
- name: Run System Tests
|
||||||
|
env:
|
||||||
|
RAILS_ENV: test
|
||||||
|
# RAILS_MASTER_KEY: ${{ secrets.RAILS_MASTER_KEY }}
|
||||||
|
# REDIS_URL: redis://localhost:6379/0
|
||||||
|
run: bin/rails db:test:prepare test:system
|
||||||
|
|
||||||
|
- name: Keep screenshots from failed system tests
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
if: failure()
|
||||||
|
with:
|
||||||
|
name: screenshots
|
||||||
|
path: ${{ github.workspace }}/tmp/screenshots
|
||||||
|
if-no-files-found: ignore
|
||||||
39
.gitignore
vendored
Normal file
39
.gitignore
vendored
Normal file
|
|
@ -0,0 +1,39 @@
|
||||||
|
# See https://help.github.com/articles/ignoring-files for more about ignoring files.
|
||||||
|
#
|
||||||
|
# Temporary files generated by your text editor or operating system
|
||||||
|
# belong in git's global ignore instead:
|
||||||
|
# `$XDG_CONFIG_HOME/git/ignore` or `~/.config/git/ignore`
|
||||||
|
|
||||||
|
# Ignore bundler config.
|
||||||
|
/.bundle
|
||||||
|
|
||||||
|
# Ignore all environment files.
|
||||||
|
/.env*
|
||||||
|
|
||||||
|
# Ignore all logfiles and tempfiles.
|
||||||
|
/log/*
|
||||||
|
/tmp/*
|
||||||
|
!/log/.keep
|
||||||
|
!/tmp/.keep
|
||||||
|
|
||||||
|
# Ignore pidfiles, but keep the directory.
|
||||||
|
/tmp/pids/*
|
||||||
|
!/tmp/pids/
|
||||||
|
!/tmp/pids/.keep
|
||||||
|
|
||||||
|
# Ignore storage (uploaded files in development and any SQLite databases).
|
||||||
|
# /storage/*
|
||||||
|
!/storage/.keep
|
||||||
|
/tmp/storage/*
|
||||||
|
!/tmp/storage/
|
||||||
|
!/tmp/storage/.keep
|
||||||
|
/app/views/passwords
|
||||||
|
/app/views/active_storage
|
||||||
|
/public/assets
|
||||||
|
|
||||||
|
|
||||||
|
# Ignore key files for decrypting credentials and more.
|
||||||
|
/config/*.key
|
||||||
|
/public/assets
|
||||||
|
db/
|
||||||
|
storage/
|
||||||
3
.kamal/hooks/docker-setup.sample
Executable file
3
.kamal/hooks/docker-setup.sample
Executable file
|
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
echo "Docker set up on $KAMAL_HOSTS..."
|
||||||
3
.kamal/hooks/post-app-boot.sample
Executable file
3
.kamal/hooks/post-app-boot.sample
Executable file
|
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
echo "Booted app version $KAMAL_VERSION on $KAMAL_HOSTS..."
|
||||||
14
.kamal/hooks/post-deploy.sample
Executable file
14
.kamal/hooks/post-deploy.sample
Executable file
|
|
@ -0,0 +1,14 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# A sample post-deploy hook
|
||||||
|
#
|
||||||
|
# These environment variables are available:
|
||||||
|
# KAMAL_RECORDED_AT
|
||||||
|
# KAMAL_PERFORMER
|
||||||
|
# KAMAL_VERSION
|
||||||
|
# KAMAL_HOSTS
|
||||||
|
# KAMAL_ROLES (if set)
|
||||||
|
# KAMAL_DESTINATION (if set)
|
||||||
|
# KAMAL_RUNTIME
|
||||||
|
|
||||||
|
echo "$KAMAL_PERFORMER deployed $KAMAL_VERSION to $KAMAL_DESTINATION in $KAMAL_RUNTIME seconds"
|
||||||
3
.kamal/hooks/post-proxy-reboot.sample
Executable file
3
.kamal/hooks/post-proxy-reboot.sample
Executable file
|
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
echo "Rebooted kamal-proxy on $KAMAL_HOSTS"
|
||||||
3
.kamal/hooks/pre-app-boot.sample
Executable file
3
.kamal/hooks/pre-app-boot.sample
Executable file
|
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
echo "Booting app version $KAMAL_VERSION on $KAMAL_HOSTS..."
|
||||||
51
.kamal/hooks/pre-build.sample
Executable file
51
.kamal/hooks/pre-build.sample
Executable file
|
|
@ -0,0 +1,51 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# A sample pre-build hook
|
||||||
|
#
|
||||||
|
# Checks:
|
||||||
|
# 1. We have a clean checkout
|
||||||
|
# 2. A remote is configured
|
||||||
|
# 3. The branch has been pushed to the remote
|
||||||
|
# 4. The version we are deploying matches the remote
|
||||||
|
#
|
||||||
|
# These environment variables are available:
|
||||||
|
# KAMAL_RECORDED_AT
|
||||||
|
# KAMAL_PERFORMER
|
||||||
|
# KAMAL_VERSION
|
||||||
|
# KAMAL_HOSTS
|
||||||
|
# KAMAL_ROLES (if set)
|
||||||
|
# KAMAL_DESTINATION (if set)
|
||||||
|
|
||||||
|
if [ -n "$(git status --porcelain)" ]; then
|
||||||
|
echo "Git checkout is not clean, aborting..." >&2
|
||||||
|
git status --porcelain >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
first_remote=$(git remote)
|
||||||
|
|
||||||
|
if [ -z "$first_remote" ]; then
|
||||||
|
echo "No git remote set, aborting..." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
current_branch=$(git branch --show-current)
|
||||||
|
|
||||||
|
if [ -z "$current_branch" ]; then
|
||||||
|
echo "Not on a git branch, aborting..." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
remote_head=$(git ls-remote $first_remote --tags $current_branch | cut -f1)
|
||||||
|
|
||||||
|
if [ -z "$remote_head" ]; then
|
||||||
|
echo "Branch not pushed to remote, aborting..." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "$KAMAL_VERSION" != "$remote_head" ]; then
|
||||||
|
echo "Version ($KAMAL_VERSION) does not match remote HEAD ($remote_head), aborting..." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
||||||
47
.kamal/hooks/pre-connect.sample
Executable file
47
.kamal/hooks/pre-connect.sample
Executable file
|
|
@ -0,0 +1,47 @@
|
||||||
|
#!/usr/bin/env ruby
|
||||||
|
|
||||||
|
# A sample pre-connect check
|
||||||
|
#
|
||||||
|
# Warms DNS before connecting to hosts in parallel
|
||||||
|
#
|
||||||
|
# These environment variables are available:
|
||||||
|
# KAMAL_RECORDED_AT
|
||||||
|
# KAMAL_PERFORMER
|
||||||
|
# KAMAL_VERSION
|
||||||
|
# KAMAL_HOSTS
|
||||||
|
# KAMAL_ROLES (if set)
|
||||||
|
# KAMAL_DESTINATION (if set)
|
||||||
|
# KAMAL_RUNTIME
|
||||||
|
|
||||||
|
hosts = ENV["KAMAL_HOSTS"].split(",")
|
||||||
|
results = nil
|
||||||
|
max = 3
|
||||||
|
|
||||||
|
elapsed = Benchmark.realtime do
|
||||||
|
results = hosts.map do |host|
|
||||||
|
Thread.new do
|
||||||
|
tries = 1
|
||||||
|
|
||||||
|
begin
|
||||||
|
Socket.getaddrinfo(host, 0, Socket::AF_UNSPEC, Socket::SOCK_STREAM, nil, Socket::AI_CANONNAME)
|
||||||
|
rescue SocketError
|
||||||
|
if tries < max
|
||||||
|
puts "Retrying DNS warmup: #{host}"
|
||||||
|
tries += 1
|
||||||
|
sleep rand
|
||||||
|
retry
|
||||||
|
else
|
||||||
|
puts "DNS warmup failed: #{host}"
|
||||||
|
host
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
tries
|
||||||
|
end
|
||||||
|
end.map(&:value)
|
||||||
|
end
|
||||||
|
|
||||||
|
retries = results.sum - hosts.size
|
||||||
|
nopes = results.count { |r| r == max }
|
||||||
|
|
||||||
|
puts "Prewarmed %d DNS lookups in %.2f sec: %d retries, %d failures" % [ hosts.size, elapsed, retries, nopes ]
|
||||||
122
.kamal/hooks/pre-deploy.sample
Executable file
122
.kamal/hooks/pre-deploy.sample
Executable file
|
|
@ -0,0 +1,122 @@
|
||||||
|
#!/usr/bin/env ruby
|
||||||
|
|
||||||
|
# A sample pre-deploy hook
|
||||||
|
#
|
||||||
|
# Checks the Github status of the build, waiting for a pending build to complete for up to 720 seconds.
|
||||||
|
#
|
||||||
|
# Fails unless the combined status is "success"
|
||||||
|
#
|
||||||
|
# These environment variables are available:
|
||||||
|
# KAMAL_RECORDED_AT
|
||||||
|
# KAMAL_PERFORMER
|
||||||
|
# KAMAL_VERSION
|
||||||
|
# KAMAL_HOSTS
|
||||||
|
# KAMAL_COMMAND
|
||||||
|
# KAMAL_SUBCOMMAND
|
||||||
|
# KAMAL_ROLES (if set)
|
||||||
|
# KAMAL_DESTINATION (if set)
|
||||||
|
|
||||||
|
# Only check the build status for production deployments
|
||||||
|
if ENV["KAMAL_COMMAND"] == "rollback" || ENV["KAMAL_DESTINATION"] != "production"
|
||||||
|
exit 0
|
||||||
|
end
|
||||||
|
|
||||||
|
require "bundler/inline"
|
||||||
|
|
||||||
|
# true = install gems so this is fast on repeat invocations
|
||||||
|
gemfile(true, quiet: true) do
|
||||||
|
source "https://rubygems.org"
|
||||||
|
|
||||||
|
gem "octokit"
|
||||||
|
gem "faraday-retry"
|
||||||
|
end
|
||||||
|
|
||||||
|
MAX_ATTEMPTS = 72
|
||||||
|
ATTEMPTS_GAP = 10
|
||||||
|
|
||||||
|
def exit_with_error(message)
|
||||||
|
$stderr.puts message
|
||||||
|
exit 1
|
||||||
|
end
|
||||||
|
|
||||||
|
class GithubStatusChecks
|
||||||
|
attr_reader :remote_url, :git_sha, :github_client, :combined_status
|
||||||
|
|
||||||
|
def initialize
|
||||||
|
@remote_url = github_repo_from_remote_url
|
||||||
|
@git_sha = `git rev-parse HEAD`.strip
|
||||||
|
@github_client = Octokit::Client.new(access_token: ENV["GITHUB_TOKEN"])
|
||||||
|
refresh!
|
||||||
|
end
|
||||||
|
|
||||||
|
def refresh!
|
||||||
|
@combined_status = github_client.combined_status(remote_url, git_sha)
|
||||||
|
end
|
||||||
|
|
||||||
|
def state
|
||||||
|
combined_status[:state]
|
||||||
|
end
|
||||||
|
|
||||||
|
def first_status_url
|
||||||
|
first_status = combined_status[:statuses].find { |status| status[:state] == state }
|
||||||
|
first_status && first_status[:target_url]
|
||||||
|
end
|
||||||
|
|
||||||
|
def complete_count
|
||||||
|
combined_status[:statuses].count { |status| status[:state] != "pending"}
|
||||||
|
end
|
||||||
|
|
||||||
|
def total_count
|
||||||
|
combined_status[:statuses].count
|
||||||
|
end
|
||||||
|
|
||||||
|
def current_status
|
||||||
|
if total_count > 0
|
||||||
|
"Completed #{complete_count}/#{total_count} checks, see #{first_status_url} ..."
|
||||||
|
else
|
||||||
|
"Build not started..."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def github_repo_from_remote_url
|
||||||
|
url = `git config --get remote.origin.url`.strip.delete_suffix(".git")
|
||||||
|
if url.start_with?("https://github.com/")
|
||||||
|
url.delete_prefix("https://github.com/")
|
||||||
|
elsif url.start_with?("git@github.com:")
|
||||||
|
url.delete_prefix("git@github.com:")
|
||||||
|
else
|
||||||
|
url
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
$stdout.sync = true
|
||||||
|
|
||||||
|
begin
|
||||||
|
puts "Checking build status..."
|
||||||
|
|
||||||
|
attempts = 0
|
||||||
|
checks = GithubStatusChecks.new
|
||||||
|
|
||||||
|
loop do
|
||||||
|
case checks.state
|
||||||
|
when "success"
|
||||||
|
puts "Checks passed, see #{checks.first_status_url}"
|
||||||
|
exit 0
|
||||||
|
when "failure"
|
||||||
|
exit_with_error "Checks failed, see #{checks.first_status_url}"
|
||||||
|
when "pending"
|
||||||
|
attempts += 1
|
||||||
|
end
|
||||||
|
|
||||||
|
exit_with_error "Checks are still pending, gave up after #{MAX_ATTEMPTS * ATTEMPTS_GAP} seconds" if attempts == MAX_ATTEMPTS
|
||||||
|
|
||||||
|
puts checks.current_status
|
||||||
|
sleep(ATTEMPTS_GAP)
|
||||||
|
checks.refresh!
|
||||||
|
end
|
||||||
|
rescue Octokit::NotFound
|
||||||
|
exit_with_error "Build status could not be found"
|
||||||
|
end
|
||||||
3
.kamal/hooks/pre-proxy-reboot.sample
Executable file
3
.kamal/hooks/pre-proxy-reboot.sample
Executable file
|
|
@ -0,0 +1,3 @@
|
||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
echo "Rebooting kamal-proxy on $KAMAL_HOSTS..."
|
||||||
19
.kamal/secrets
Normal file
19
.kamal/secrets
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
# Secrets defined here are available for reference under registry/password, env/secret, builder/secrets,
|
||||||
|
# and accessories/*/env/secret in config/deploy.yml. All secrets should be pulled from either
|
||||||
|
# password manager, ENV, or a file. DO NOT ENTER RAW CREDENTIALS HERE! This file needs to be safe for git.
|
||||||
|
|
||||||
|
# Example of extracting secrets from 1password (or another compatible pw manager)
|
||||||
|
# SECRETS=$(kamal secrets fetch --adapter 1password --account your-account --from Vault/Item KAMAL_REGISTRY_PASSWORD RAILS_MASTER_KEY)
|
||||||
|
# KAMAL_REGISTRY_PASSWORD=$(kamal secrets extract KAMAL_REGISTRY_PASSWORD ${SECRETS})
|
||||||
|
# RAILS_MASTER_KEY=$(kamal secrets extract RAILS_MASTER_KEY ${SECRETS})
|
||||||
|
|
||||||
|
# Example of extracting secrets from Rails credentials
|
||||||
|
# KAMAL_REGISTRY_PASSWORD=$(rails credentials:fetch kamal.registry_password)
|
||||||
|
|
||||||
|
# Use a GITHUB_TOKEN if private repositories are needed for the image
|
||||||
|
# GITHUB_TOKEN=$(gh config get -h github.com oauth_token)
|
||||||
|
|
||||||
|
# Grab the registry password from ENV
|
||||||
|
KAMAL_REGISTRY_PASSWORD=$KAMAL_REGISTRY_PASSWORD
|
||||||
|
# Improve security by using a password manager. Never check config/master.key into git!
|
||||||
|
RAILS_MASTER_KEY=$(cat config/master.key)
|
||||||
8
.rubocop.yml
Normal file
8
.rubocop.yml
Normal file
|
|
@ -0,0 +1,8 @@
|
||||||
|
# Omakase Ruby styling for Rails
|
||||||
|
inherit_gem: { rubocop-rails-omakase: rubocop.yml }
|
||||||
|
|
||||||
|
# Overwrite or add rules to create your own house style
|
||||||
|
#
|
||||||
|
# # Use `[a, [b, c]]` not `[ a, [ b, c ] ]`
|
||||||
|
# Layout/SpaceInsideArrayLiteralBrackets:
|
||||||
|
# Enabled: false
|
||||||
1
.ruby-version
Normal file
1
.ruby-version
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
ruby-3.4.7
|
||||||
76
Dockerfile
Normal file
76
Dockerfile
Normal file
|
|
@ -0,0 +1,76 @@
|
||||||
|
# syntax=docker/dockerfile:1
|
||||||
|
# check=error=true
|
||||||
|
|
||||||
|
# This Dockerfile is designed for production, not development. Use with Kamal or build'n'run by hand:
|
||||||
|
# docker build -t galaxy .
|
||||||
|
# docker run -d -p 80:80 -e RAILS_MASTER_KEY=<value from config/master.key> --name galaxy galaxy
|
||||||
|
|
||||||
|
# For a containerized dev environment, see Dev Containers: https://guides.rubyonrails.org/getting_started_with_devcontainer.html
|
||||||
|
|
||||||
|
# Make sure RUBY_VERSION matches the Ruby version in .ruby-version
|
||||||
|
ARG RUBY_VERSION=3.4.7
|
||||||
|
FROM docker.io/library/ruby:$RUBY_VERSION-slim AS base
|
||||||
|
|
||||||
|
# Rails app lives here
|
||||||
|
WORKDIR /rails
|
||||||
|
|
||||||
|
# Install base packages
|
||||||
|
RUN apt-get update -qq && \
|
||||||
|
apt-get install --no-install-recommends -y curl libjemalloc2 libvips sqlite3 && \
|
||||||
|
ln -s /usr/lib/$(uname -m)-linux-gnu/libjemalloc.so.2 /usr/local/lib/libjemalloc.so && \
|
||||||
|
rm -rf /var/lib/apt/lists /var/cache/apt/archives
|
||||||
|
|
||||||
|
# Set production environment variables and enable jemalloc for reduced memory usage and latency.
|
||||||
|
ENV RAILS_ENV="production" \
|
||||||
|
BUNDLE_DEPLOYMENT="1" \
|
||||||
|
BUNDLE_PATH="/usr/local/bundle" \
|
||||||
|
BUNDLE_WITHOUT="development" \
|
||||||
|
LD_PRELOAD="/usr/local/lib/libjemalloc.so"
|
||||||
|
|
||||||
|
# Throw-away build stage to reduce size of final image
|
||||||
|
FROM base AS build
|
||||||
|
|
||||||
|
# Install packages needed to build gems
|
||||||
|
RUN apt-get update -qq && \
|
||||||
|
apt-get install --no-install-recommends -y build-essential git libyaml-dev pkg-config && \
|
||||||
|
rm -rf /var/lib/apt/lists /var/cache/apt/archives
|
||||||
|
|
||||||
|
# Install application gems
|
||||||
|
COPY Gemfile Gemfile.lock vendor ./
|
||||||
|
|
||||||
|
RUN bundle install && \
|
||||||
|
rm -rf ~/.bundle/ "${BUNDLE_PATH}"/ruby/*/cache "${BUNDLE_PATH}"/ruby/*/bundler/gems/*/.git && \
|
||||||
|
# -j 1 disable parallel compilation to avoid a QEMU bug: https://github.com/rails/bootsnap/issues/495
|
||||||
|
bundle exec bootsnap precompile -j 1 --gemfile
|
||||||
|
|
||||||
|
# Copy application code
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
# Precompile bootsnap code for faster boot times.
|
||||||
|
# -j 1 disable parallel compilation to avoid a QEMU bug: https://github.com/rails/bootsnap/issues/495
|
||||||
|
RUN bundle exec bootsnap precompile -j 1 app/ lib/
|
||||||
|
|
||||||
|
# Precompiling assets for production without requiring secret RAILS_MASTER_KEY
|
||||||
|
RUN SECRET_KEY_BASE_DUMMY=1 ./bin/rails assets:precompile
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Final stage for app image
|
||||||
|
FROM base
|
||||||
|
|
||||||
|
# Run and own only the runtime files as a non-root user for security
|
||||||
|
RUN groupadd --system --gid 1000 rails && \
|
||||||
|
useradd rails --uid 1000 --gid 1000 --create-home --shell /bin/bash
|
||||||
|
USER 1000:1000
|
||||||
|
|
||||||
|
# Copy built artifacts: gems, application
|
||||||
|
COPY --chown=rails:rails --from=build "${BUNDLE_PATH}" "${BUNDLE_PATH}"
|
||||||
|
COPY --chown=rails:rails --from=build /rails /rails
|
||||||
|
|
||||||
|
# Entrypoint prepares the database.
|
||||||
|
ENTRYPOINT ["/rails/bin/docker-entrypoint"]
|
||||||
|
|
||||||
|
# Start server via Thruster by default, this can be overwritten at runtime
|
||||||
|
EXPOSE 3000
|
||||||
|
CMD ["./bin/thrust", "./bin/rails", "server"]
|
||||||
68
Gemfile
Normal file
68
Gemfile
Normal file
|
|
@ -0,0 +1,68 @@
|
||||||
|
source "https://rubygems.org"
|
||||||
|
|
||||||
|
# Bundle edge Rails instead: gem "rails", github: "rails/rails", branch: "main"
|
||||||
|
gem "rails", "~> 8.1.1"
|
||||||
|
# The modern asset pipeline for Rails [https://github.com/rails/propshaft]
|
||||||
|
gem "propshaft"
|
||||||
|
# Use sqlite3 as the database for Active Record
|
||||||
|
gem "sqlite3", ">= 2.1"
|
||||||
|
# Use the Puma web server [https://github.com/puma/puma]
|
||||||
|
gem "puma", ">= 5.0"
|
||||||
|
# Use JavaScript with ESM import maps [https://github.com/rails/importmap-rails]
|
||||||
|
gem "importmap-rails"
|
||||||
|
# Hotwire's SPA-like page accelerator [https://turbo.hotwired.dev]
|
||||||
|
gem "turbo-rails"
|
||||||
|
# Hotwire's modest JavaScript framework [https://stimulus.hotwired.dev]
|
||||||
|
gem "stimulus-rails"
|
||||||
|
# Build JSON APIs with ease [https://github.com/rails/jbuilder]
|
||||||
|
gem "jbuilder"
|
||||||
|
|
||||||
|
# Use Active Model has_secure_password [https://guides.rubyonrails.org/active_model_basics.html#securepassword]
|
||||||
|
gem "bcrypt", "~> 3.1.7"
|
||||||
|
|
||||||
|
# Windows does not include zoneinfo files, so bundle the tzinfo-data gem
|
||||||
|
gem "tzinfo-data", platforms: %i[ windows jruby ]
|
||||||
|
|
||||||
|
# Use the database-backed adapters for Rails.cache, Active Job, and Action Cable
|
||||||
|
gem "solid_cache"
|
||||||
|
gem "solid_queue"
|
||||||
|
gem "solid_cable"
|
||||||
|
|
||||||
|
gem "rss"
|
||||||
|
|
||||||
|
# Reduces boot times through caching; required in config/boot.rb
|
||||||
|
gem "bootsnap", require: false
|
||||||
|
|
||||||
|
# Deploy this application anywhere as a Docker container [https://kamal-deploy.org]
|
||||||
|
gem "kamal", require: false
|
||||||
|
|
||||||
|
# Add HTTP asset caching/compression and X-Sendfile acceleration to Puma [https://github.com/basecamp/thruster/]
|
||||||
|
gem "thruster", require: false
|
||||||
|
|
||||||
|
# Use Active Storage variants [https://guides.rubyonrails.org/active_storage_overview.html#transforming-images]
|
||||||
|
gem "image_processing", "~> 1.2"
|
||||||
|
|
||||||
|
group :development, :test do
|
||||||
|
# See https://guides.rubyonrails.org/debugging_rails_applications.html#debugging-with-the-debug-gem
|
||||||
|
gem "debug", platforms: %i[ mri windows ], require: "debug/prelude"
|
||||||
|
|
||||||
|
# Audits gems for known security defects (use config/bundler-audit.yml to ignore issues)
|
||||||
|
gem "bundler-audit", require: false
|
||||||
|
|
||||||
|
# Static analysis for security vulnerabilities [https://brakemanscanner.org/]
|
||||||
|
gem "brakeman", require: false
|
||||||
|
|
||||||
|
# Omakase Ruby styling [https://github.com/rails/rubocop-rails-omakase/]
|
||||||
|
gem "rubocop-rails-omakase", require: false
|
||||||
|
end
|
||||||
|
|
||||||
|
group :development do
|
||||||
|
# Use console on exceptions pages [https://github.com/rails/web-console]
|
||||||
|
gem "web-console"
|
||||||
|
end
|
||||||
|
|
||||||
|
group :test do
|
||||||
|
# Use system testing [https://guides.rubyonrails.org/testing.html#system-testing]
|
||||||
|
gem "capybara"
|
||||||
|
gem "selenium-webdriver"
|
||||||
|
end
|
||||||
414
Gemfile.lock
Normal file
414
Gemfile.lock
Normal file
|
|
@ -0,0 +1,414 @@
|
||||||
|
GEM
|
||||||
|
remote: https://rubygems.org/
|
||||||
|
specs:
|
||||||
|
action_text-trix (2.1.15)
|
||||||
|
railties
|
||||||
|
actioncable (8.1.1)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
nio4r (~> 2.0)
|
||||||
|
websocket-driver (>= 0.6.1)
|
||||||
|
zeitwerk (~> 2.6)
|
||||||
|
actionmailbox (8.1.1)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
activejob (= 8.1.1)
|
||||||
|
activerecord (= 8.1.1)
|
||||||
|
activestorage (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
mail (>= 2.8.0)
|
||||||
|
actionmailer (8.1.1)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
actionview (= 8.1.1)
|
||||||
|
activejob (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
mail (>= 2.8.0)
|
||||||
|
rails-dom-testing (~> 2.2)
|
||||||
|
actionpack (8.1.1)
|
||||||
|
actionview (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
nokogiri (>= 1.8.5)
|
||||||
|
rack (>= 2.2.4)
|
||||||
|
rack-session (>= 1.0.1)
|
||||||
|
rack-test (>= 0.6.3)
|
||||||
|
rails-dom-testing (~> 2.2)
|
||||||
|
rails-html-sanitizer (~> 1.6)
|
||||||
|
useragent (~> 0.16)
|
||||||
|
actiontext (8.1.1)
|
||||||
|
action_text-trix (~> 2.1.15)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
activerecord (= 8.1.1)
|
||||||
|
activestorage (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
globalid (>= 0.6.0)
|
||||||
|
nokogiri (>= 1.8.5)
|
||||||
|
actionview (8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
builder (~> 3.1)
|
||||||
|
erubi (~> 1.11)
|
||||||
|
rails-dom-testing (~> 2.2)
|
||||||
|
rails-html-sanitizer (~> 1.6)
|
||||||
|
activejob (8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
globalid (>= 0.3.6)
|
||||||
|
activemodel (8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
activerecord (8.1.1)
|
||||||
|
activemodel (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
timeout (>= 0.4.0)
|
||||||
|
activestorage (8.1.1)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
activejob (= 8.1.1)
|
||||||
|
activerecord (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
marcel (~> 1.0)
|
||||||
|
activesupport (8.1.1)
|
||||||
|
base64
|
||||||
|
bigdecimal
|
||||||
|
concurrent-ruby (~> 1.0, >= 1.3.1)
|
||||||
|
connection_pool (>= 2.2.5)
|
||||||
|
drb
|
||||||
|
i18n (>= 1.6, < 2)
|
||||||
|
json
|
||||||
|
logger (>= 1.4.2)
|
||||||
|
minitest (>= 5.1)
|
||||||
|
securerandom (>= 0.3)
|
||||||
|
tzinfo (~> 2.0, >= 2.0.5)
|
||||||
|
uri (>= 0.13.1)
|
||||||
|
addressable (2.8.7)
|
||||||
|
public_suffix (>= 2.0.2, < 7.0)
|
||||||
|
ast (2.4.3)
|
||||||
|
base64 (0.3.0)
|
||||||
|
bcrypt (3.1.20)
|
||||||
|
bcrypt_pbkdf (1.1.1)
|
||||||
|
bigdecimal (3.3.1)
|
||||||
|
bindex (0.8.1)
|
||||||
|
bootsnap (1.18.6)
|
||||||
|
msgpack (~> 1.2)
|
||||||
|
brakeman (7.1.1)
|
||||||
|
racc
|
||||||
|
builder (3.3.0)
|
||||||
|
bundler-audit (0.9.2)
|
||||||
|
bundler (>= 1.2.0, < 3)
|
||||||
|
thor (~> 1.0)
|
||||||
|
capybara (3.40.0)
|
||||||
|
addressable
|
||||||
|
matrix
|
||||||
|
mini_mime (>= 0.1.3)
|
||||||
|
nokogiri (~> 1.11)
|
||||||
|
rack (>= 1.6.0)
|
||||||
|
rack-test (>= 0.6.3)
|
||||||
|
regexp_parser (>= 1.5, < 3.0)
|
||||||
|
xpath (~> 3.2)
|
||||||
|
concurrent-ruby (1.3.5)
|
||||||
|
connection_pool (2.5.4)
|
||||||
|
crass (1.0.6)
|
||||||
|
date (3.5.0)
|
||||||
|
debug (1.11.0)
|
||||||
|
irb (~> 1.10)
|
||||||
|
reline (>= 0.3.8)
|
||||||
|
dotenv (3.1.8)
|
||||||
|
drb (2.2.3)
|
||||||
|
ed25519 (1.4.0)
|
||||||
|
erb (5.1.3)
|
||||||
|
erubi (1.13.1)
|
||||||
|
et-orbi (1.4.0)
|
||||||
|
tzinfo
|
||||||
|
ffi (1.17.2-aarch64-linux-gnu)
|
||||||
|
ffi (1.17.2-aarch64-linux-musl)
|
||||||
|
ffi (1.17.2-arm-linux-gnu)
|
||||||
|
ffi (1.17.2-arm-linux-musl)
|
||||||
|
ffi (1.17.2-x86_64-linux-gnu)
|
||||||
|
ffi (1.17.2-x86_64-linux-musl)
|
||||||
|
fugit (1.12.1)
|
||||||
|
et-orbi (~> 1.4)
|
||||||
|
raabro (~> 1.4)
|
||||||
|
globalid (1.3.0)
|
||||||
|
activesupport (>= 6.1)
|
||||||
|
i18n (1.14.7)
|
||||||
|
concurrent-ruby (~> 1.0)
|
||||||
|
image_processing (1.14.0)
|
||||||
|
mini_magick (>= 4.9.5, < 6)
|
||||||
|
ruby-vips (>= 2.0.17, < 3)
|
||||||
|
importmap-rails (2.2.2)
|
||||||
|
actionpack (>= 6.0.0)
|
||||||
|
activesupport (>= 6.0.0)
|
||||||
|
railties (>= 6.0.0)
|
||||||
|
io-console (0.8.1)
|
||||||
|
irb (1.15.3)
|
||||||
|
pp (>= 0.6.0)
|
||||||
|
rdoc (>= 4.0.0)
|
||||||
|
reline (>= 0.4.2)
|
||||||
|
jbuilder (2.14.1)
|
||||||
|
actionview (>= 7.0.0)
|
||||||
|
activesupport (>= 7.0.0)
|
||||||
|
json (2.16.0)
|
||||||
|
kamal (2.8.2)
|
||||||
|
activesupport (>= 7.0)
|
||||||
|
base64 (~> 0.2)
|
||||||
|
bcrypt_pbkdf (~> 1.0)
|
||||||
|
concurrent-ruby (~> 1.2)
|
||||||
|
dotenv (~> 3.1)
|
||||||
|
ed25519 (~> 1.4)
|
||||||
|
net-ssh (~> 7.3)
|
||||||
|
sshkit (>= 1.23.0, < 2.0)
|
||||||
|
thor (~> 1.3)
|
||||||
|
zeitwerk (>= 2.6.18, < 3.0)
|
||||||
|
language_server-protocol (3.17.0.5)
|
||||||
|
lint_roller (1.1.0)
|
||||||
|
logger (1.7.0)
|
||||||
|
loofah (2.24.1)
|
||||||
|
crass (~> 1.0.2)
|
||||||
|
nokogiri (>= 1.12.0)
|
||||||
|
mail (2.9.0)
|
||||||
|
logger
|
||||||
|
mini_mime (>= 0.1.1)
|
||||||
|
net-imap
|
||||||
|
net-pop
|
||||||
|
net-smtp
|
||||||
|
marcel (1.1.0)
|
||||||
|
matrix (0.4.3)
|
||||||
|
mini_magick (5.3.1)
|
||||||
|
logger
|
||||||
|
mini_mime (1.1.5)
|
||||||
|
minitest (5.26.0)
|
||||||
|
msgpack (1.8.0)
|
||||||
|
net-imap (0.5.12)
|
||||||
|
date
|
||||||
|
net-protocol
|
||||||
|
net-pop (0.1.2)
|
||||||
|
net-protocol
|
||||||
|
net-protocol (0.2.2)
|
||||||
|
timeout
|
||||||
|
net-scp (4.1.0)
|
||||||
|
net-ssh (>= 2.6.5, < 8.0.0)
|
||||||
|
net-sftp (4.0.0)
|
||||||
|
net-ssh (>= 5.0.0, < 8.0.0)
|
||||||
|
net-smtp (0.5.1)
|
||||||
|
net-protocol
|
||||||
|
net-ssh (7.3.0)
|
||||||
|
nio4r (2.7.5)
|
||||||
|
nokogiri (1.18.10-aarch64-linux-gnu)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.18.10-aarch64-linux-musl)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.18.10-arm-linux-gnu)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.18.10-arm-linux-musl)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.18.10-x86_64-linux-gnu)
|
||||||
|
racc (~> 1.4)
|
||||||
|
nokogiri (1.18.10-x86_64-linux-musl)
|
||||||
|
racc (~> 1.4)
|
||||||
|
ostruct (0.6.3)
|
||||||
|
parallel (1.27.0)
|
||||||
|
parser (3.3.10.0)
|
||||||
|
ast (~> 2.4.1)
|
||||||
|
racc
|
||||||
|
pp (0.6.3)
|
||||||
|
prettyprint
|
||||||
|
prettyprint (0.2.0)
|
||||||
|
prism (1.6.0)
|
||||||
|
propshaft (1.3.1)
|
||||||
|
actionpack (>= 7.0.0)
|
||||||
|
activesupport (>= 7.0.0)
|
||||||
|
rack
|
||||||
|
psych (5.2.6)
|
||||||
|
date
|
||||||
|
stringio
|
||||||
|
public_suffix (6.0.2)
|
||||||
|
puma (7.1.0)
|
||||||
|
nio4r (~> 2.0)
|
||||||
|
raabro (1.4.0)
|
||||||
|
racc (1.8.1)
|
||||||
|
rack (3.2.4)
|
||||||
|
rack-session (2.1.1)
|
||||||
|
base64 (>= 0.1.0)
|
||||||
|
rack (>= 3.0.0)
|
||||||
|
rack-test (2.2.0)
|
||||||
|
rack (>= 1.3)
|
||||||
|
rackup (2.2.1)
|
||||||
|
rack (>= 3)
|
||||||
|
rails (8.1.1)
|
||||||
|
actioncable (= 8.1.1)
|
||||||
|
actionmailbox (= 8.1.1)
|
||||||
|
actionmailer (= 8.1.1)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
actiontext (= 8.1.1)
|
||||||
|
actionview (= 8.1.1)
|
||||||
|
activejob (= 8.1.1)
|
||||||
|
activemodel (= 8.1.1)
|
||||||
|
activerecord (= 8.1.1)
|
||||||
|
activestorage (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
bundler (>= 1.15.0)
|
||||||
|
railties (= 8.1.1)
|
||||||
|
rails-dom-testing (2.3.0)
|
||||||
|
activesupport (>= 5.0.0)
|
||||||
|
minitest
|
||||||
|
nokogiri (>= 1.6)
|
||||||
|
rails-html-sanitizer (1.6.2)
|
||||||
|
loofah (~> 2.21)
|
||||||
|
nokogiri (>= 1.15.7, != 1.16.7, != 1.16.6, != 1.16.5, != 1.16.4, != 1.16.3, != 1.16.2, != 1.16.1, != 1.16.0.rc1, != 1.16.0)
|
||||||
|
railties (8.1.1)
|
||||||
|
actionpack (= 8.1.1)
|
||||||
|
activesupport (= 8.1.1)
|
||||||
|
irb (~> 1.13)
|
||||||
|
rackup (>= 1.0.0)
|
||||||
|
rake (>= 12.2)
|
||||||
|
thor (~> 1.0, >= 1.2.2)
|
||||||
|
tsort (>= 0.2)
|
||||||
|
zeitwerk (~> 2.6)
|
||||||
|
rainbow (3.1.1)
|
||||||
|
rake (13.3.1)
|
||||||
|
rdoc (6.15.1)
|
||||||
|
erb
|
||||||
|
psych (>= 4.0.0)
|
||||||
|
tsort
|
||||||
|
regexp_parser (2.11.3)
|
||||||
|
reline (0.6.2)
|
||||||
|
io-console (~> 0.5)
|
||||||
|
rexml (3.4.4)
|
||||||
|
rss (0.3.1)
|
||||||
|
rexml
|
||||||
|
rubocop (1.81.7)
|
||||||
|
json (~> 2.3)
|
||||||
|
language_server-protocol (~> 3.17.0.2)
|
||||||
|
lint_roller (~> 1.1.0)
|
||||||
|
parallel (~> 1.10)
|
||||||
|
parser (>= 3.3.0.2)
|
||||||
|
rainbow (>= 2.2.2, < 4.0)
|
||||||
|
regexp_parser (>= 2.9.3, < 3.0)
|
||||||
|
rubocop-ast (>= 1.47.1, < 2.0)
|
||||||
|
ruby-progressbar (~> 1.7)
|
||||||
|
unicode-display_width (>= 2.4.0, < 4.0)
|
||||||
|
rubocop-ast (1.47.1)
|
||||||
|
parser (>= 3.3.7.2)
|
||||||
|
prism (~> 1.4)
|
||||||
|
rubocop-performance (1.26.1)
|
||||||
|
lint_roller (~> 1.1)
|
||||||
|
rubocop (>= 1.75.0, < 2.0)
|
||||||
|
rubocop-ast (>= 1.47.1, < 2.0)
|
||||||
|
rubocop-rails (2.33.4)
|
||||||
|
activesupport (>= 4.2.0)
|
||||||
|
lint_roller (~> 1.1)
|
||||||
|
rack (>= 1.1)
|
||||||
|
rubocop (>= 1.75.0, < 2.0)
|
||||||
|
rubocop-ast (>= 1.44.0, < 2.0)
|
||||||
|
rubocop-rails-omakase (1.1.0)
|
||||||
|
rubocop (>= 1.72)
|
||||||
|
rubocop-performance (>= 1.24)
|
||||||
|
rubocop-rails (>= 2.30)
|
||||||
|
ruby-progressbar (1.13.0)
|
||||||
|
ruby-vips (2.2.5)
|
||||||
|
ffi (~> 1.12)
|
||||||
|
logger
|
||||||
|
rubyzip (3.2.2)
|
||||||
|
securerandom (0.4.1)
|
||||||
|
selenium-webdriver (4.38.0)
|
||||||
|
base64 (~> 0.2)
|
||||||
|
logger (~> 1.4)
|
||||||
|
rexml (~> 3.2, >= 3.2.5)
|
||||||
|
rubyzip (>= 1.2.2, < 4.0)
|
||||||
|
websocket (~> 1.0)
|
||||||
|
solid_cable (3.0.12)
|
||||||
|
actioncable (>= 7.2)
|
||||||
|
activejob (>= 7.2)
|
||||||
|
activerecord (>= 7.2)
|
||||||
|
railties (>= 7.2)
|
||||||
|
solid_cache (1.0.9)
|
||||||
|
activejob (>= 7.2)
|
||||||
|
activerecord (>= 7.2)
|
||||||
|
railties (>= 7.2)
|
||||||
|
solid_queue (1.2.4)
|
||||||
|
activejob (>= 7.1)
|
||||||
|
activerecord (>= 7.1)
|
||||||
|
concurrent-ruby (>= 1.3.1)
|
||||||
|
fugit (~> 1.11)
|
||||||
|
railties (>= 7.1)
|
||||||
|
thor (>= 1.3.1)
|
||||||
|
sqlite3 (2.8.0-aarch64-linux-gnu)
|
||||||
|
sqlite3 (2.8.0-aarch64-linux-musl)
|
||||||
|
sqlite3 (2.8.0-arm-linux-gnu)
|
||||||
|
sqlite3 (2.8.0-arm-linux-musl)
|
||||||
|
sqlite3 (2.8.0-x86_64-linux-gnu)
|
||||||
|
sqlite3 (2.8.0-x86_64-linux-musl)
|
||||||
|
sshkit (1.24.0)
|
||||||
|
base64
|
||||||
|
logger
|
||||||
|
net-scp (>= 1.1.2)
|
||||||
|
net-sftp (>= 2.1.2)
|
||||||
|
net-ssh (>= 2.8.0)
|
||||||
|
ostruct
|
||||||
|
stimulus-rails (1.3.4)
|
||||||
|
railties (>= 6.0.0)
|
||||||
|
stringio (3.1.7)
|
||||||
|
thor (1.4.0)
|
||||||
|
thruster (0.1.16)
|
||||||
|
thruster (0.1.16-aarch64-linux)
|
||||||
|
thruster (0.1.16-x86_64-linux)
|
||||||
|
timeout (0.4.4)
|
||||||
|
tsort (0.2.0)
|
||||||
|
turbo-rails (2.0.20)
|
||||||
|
actionpack (>= 7.1.0)
|
||||||
|
railties (>= 7.1.0)
|
||||||
|
tzinfo (2.0.6)
|
||||||
|
concurrent-ruby (~> 1.0)
|
||||||
|
unicode-display_width (3.2.0)
|
||||||
|
unicode-emoji (~> 4.1)
|
||||||
|
unicode-emoji (4.1.0)
|
||||||
|
uri (1.1.1)
|
||||||
|
useragent (0.16.11)
|
||||||
|
web-console (4.2.1)
|
||||||
|
actionview (>= 6.0.0)
|
||||||
|
activemodel (>= 6.0.0)
|
||||||
|
bindex (>= 0.4.0)
|
||||||
|
railties (>= 6.0.0)
|
||||||
|
websocket (1.2.11)
|
||||||
|
websocket-driver (0.8.0)
|
||||||
|
base64
|
||||||
|
websocket-extensions (>= 0.1.0)
|
||||||
|
websocket-extensions (0.1.5)
|
||||||
|
xpath (3.2.0)
|
||||||
|
nokogiri (~> 1.8)
|
||||||
|
zeitwerk (2.7.3)
|
||||||
|
|
||||||
|
PLATFORMS
|
||||||
|
aarch64-linux
|
||||||
|
aarch64-linux-gnu
|
||||||
|
aarch64-linux-musl
|
||||||
|
arm-linux-gnu
|
||||||
|
arm-linux-musl
|
||||||
|
x86_64-linux
|
||||||
|
x86_64-linux-gnu
|
||||||
|
x86_64-linux-musl
|
||||||
|
|
||||||
|
DEPENDENCIES
|
||||||
|
bcrypt (~> 3.1.7)
|
||||||
|
bootsnap
|
||||||
|
brakeman
|
||||||
|
bundler-audit
|
||||||
|
capybara
|
||||||
|
debug
|
||||||
|
image_processing (~> 1.2)
|
||||||
|
importmap-rails
|
||||||
|
jbuilder
|
||||||
|
kamal
|
||||||
|
propshaft
|
||||||
|
puma (>= 5.0)
|
||||||
|
rails (~> 8.1.1)
|
||||||
|
rss
|
||||||
|
rubocop-rails-omakase
|
||||||
|
selenium-webdriver
|
||||||
|
solid_cable
|
||||||
|
solid_cache
|
||||||
|
solid_queue
|
||||||
|
sqlite3 (>= 2.1)
|
||||||
|
stimulus-rails
|
||||||
|
thruster
|
||||||
|
turbo-rails
|
||||||
|
tzinfo-data
|
||||||
|
web-console
|
||||||
|
|
||||||
|
BUNDLED WITH
|
||||||
|
2.6.9
|
||||||
0
RAILS_ENV=production
Normal file
0
RAILS_ENV=production
Normal file
1
README.md
Normal file
1
README.md
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
hi this is basically my blog i made for me and my system. it's a mess of old code i wrote and then massively repurposed. you can use this if you want but sorry if it sucks. if it does end up being helpful to you hmu :))))
|
||||||
6
Rakefile
Normal file
6
Rakefile
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
# Add your own tasks in files placed in lib/tasks ending in .rake,
|
||||||
|
# for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
|
||||||
|
|
||||||
|
require_relative "config/application"
|
||||||
|
|
||||||
|
Rails.application.load_tasks
|
||||||
0
app/assets/images/.keep
Normal file
0
app/assets/images/.keep
Normal file
440
app/assets/stylesheets/actiontext.css
Normal file
440
app/assets/stylesheets/actiontext.css
Normal file
|
|
@ -0,0 +1,440 @@
|
||||||
|
/*
|
||||||
|
* Default Trix editor styles. See Action Text overwrites below.
|
||||||
|
*/
|
||||||
|
|
||||||
|
trix-editor {
|
||||||
|
border: 1px solid #bbb;
|
||||||
|
border-radius: 3px;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0.4em 0.6em;
|
||||||
|
min-height: 5em;
|
||||||
|
outline: none; }
|
||||||
|
|
||||||
|
trix-toolbar * {
|
||||||
|
box-sizing: border-box; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button-row {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: nowrap;
|
||||||
|
justify-content: space-between;
|
||||||
|
overflow-x: auto; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button-group {
|
||||||
|
display: flex;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
border: 1px solid #bbb;
|
||||||
|
border-top-color: #ccc;
|
||||||
|
border-bottom-color: #888;
|
||||||
|
border-radius: 3px; }
|
||||||
|
trix-toolbar .trix-button-group:not(:first-child) {
|
||||||
|
margin-left: 1.5vw; }
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
trix-toolbar .trix-button-group:not(:first-child) {
|
||||||
|
margin-left: 0; } }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button-group-spacer {
|
||||||
|
flex-grow: 1; }
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
trix-toolbar .trix-button-group-spacer {
|
||||||
|
display: none; } }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button {
|
||||||
|
position: relative;
|
||||||
|
float: left;
|
||||||
|
color: rgba(0, 0, 0, 0.6);
|
||||||
|
font-size: 0.75em;
|
||||||
|
font-weight: 600;
|
||||||
|
white-space: nowrap;
|
||||||
|
padding: 0 0.5em;
|
||||||
|
margin: 0;
|
||||||
|
outline: none;
|
||||||
|
border: none;
|
||||||
|
border-bottom: 1px solid #ddd;
|
||||||
|
border-radius: 0;
|
||||||
|
background: transparent; }
|
||||||
|
trix-toolbar .trix-button:not(:first-child) {
|
||||||
|
border-left: 1px solid #ccc; }
|
||||||
|
trix-toolbar .trix-button.trix-active {
|
||||||
|
background: #cbeefa;
|
||||||
|
color: black; }
|
||||||
|
trix-toolbar .trix-button:not(:disabled) {
|
||||||
|
cursor: pointer; }
|
||||||
|
trix-toolbar .trix-button:disabled {
|
||||||
|
color: rgba(0, 0, 0, 0.125); }
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
trix-toolbar .trix-button {
|
||||||
|
letter-spacing: -0.01em;
|
||||||
|
padding: 0 0.3em; } }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon {
|
||||||
|
font-size: inherit;
|
||||||
|
width: 2.6em;
|
||||||
|
height: 1.6em;
|
||||||
|
max-width: calc(0.8em + 4vw);
|
||||||
|
text-indent: -9999px; }
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
trix-toolbar .trix-button--icon {
|
||||||
|
height: 2em;
|
||||||
|
max-width: calc(0.8em + 3.5vw); } }
|
||||||
|
trix-toolbar .trix-button--icon::before {
|
||||||
|
display: inline-block;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
opacity: 0.6;
|
||||||
|
content: "";
|
||||||
|
background-position: center;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: contain; }
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
trix-toolbar .trix-button--icon::before {
|
||||||
|
right: 6%;
|
||||||
|
left: 6%; } }
|
||||||
|
trix-toolbar .trix-button--icon.trix-active::before {
|
||||||
|
opacity: 1; }
|
||||||
|
trix-toolbar .trix-button--icon:disabled::before {
|
||||||
|
opacity: 0.125; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-attach::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M10.5%2018V7.5c0-2.25%203-2.25%203%200V18c0%204.125-6%204.125-6%200V7.5c0-6.375%209-6.375%209%200V18%22%20stroke%3D%22%23000%22%20stroke-width%3D%222%22%20stroke-miterlimit%3D%2210%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%2F%3E%3C%2Fsvg%3E");
|
||||||
|
top: 8%;
|
||||||
|
bottom: 4%; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-bold::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M6.522%2019.242a.5.5%200%200%201-.5-.5V5.35a.5.5%200%200%201%20.5-.5h5.783c1.347%200%202.46.345%203.24.982.783.64%201.216%201.562%201.216%202.683%200%201.13-.587%202.129-1.476%202.71a.35.35%200%200%200%20.049.613c1.259.56%202.101%201.742%202.101%203.22%200%201.282-.483%202.334-1.363%203.063-.876.726-2.132%201.12-3.66%201.12h-5.89ZM9.27%207.347v3.362h1.97c.766%200%201.347-.17%201.733-.464.38-.291.587-.716.587-1.27%200-.53-.183-.928-.513-1.198-.334-.273-.838-.43-1.505-.43H9.27Zm0%205.606v3.791h2.389c.832%200%201.448-.177%201.853-.497.399-.315.614-.786.614-1.423%200-.62-.22-1.077-.63-1.385-.418-.313-1.053-.486-1.905-.486H9.27Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-italic::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M9%205h6.5v2h-2.23l-2.31%2010H13v2H6v-2h2.461l2.306-10H9V5Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-link::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M18.948%205.258a4.337%204.337%200%200%200-6.108%200L11.217%206.87a.993.993%200%200%200%200%201.41c.392.39%201.027.39%201.418%200l1.623-1.613a2.323%202.323%200%200%201%203.271%200%202.29%202.29%200%200%201%200%203.251l-2.393%202.38a3.021%203.021%200%200%201-4.255%200l-.05-.049a1.007%201.007%200%200%200-1.418%200%20.993.993%200%200%200%200%201.41l.05.049a5.036%205.036%200%200%200%207.091%200l2.394-2.38a4.275%204.275%200%200%200%200-6.072Zm-13.683%2013.6a4.337%204.337%200%200%200%206.108%200l1.262-1.255a.993.993%200%200%200%200-1.41%201.007%201.007%200%200%200-1.418%200L9.954%2017.45a2.323%202.323%200%200%201-3.27%200%202.29%202.29%200%200%201%200-3.251l2.344-2.331a2.579%202.579%200%200%201%203.631%200c.392.39%201.027.39%201.419%200a.993.993%200%200%200%200-1.41%204.593%204.593%200%200%200-6.468%200l-2.345%202.33a4.275%204.275%200%200%200%200%206.072Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-strike::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M6%2014.986c.088%202.647%202.246%204.258%205.635%204.258%203.496%200%205.713-1.728%205.713-4.463%200-.275-.02-.536-.062-.781h-3.461c.398.293.573.654.573%201.123%200%201.035-1.074%201.787-2.646%201.787-1.563%200-2.773-.762-2.91-1.924H6ZM6.432%2010h3.763c-.632-.314-.914-.715-.914-1.273%200-1.045.977-1.739%202.432-1.739%201.475%200%202.52.723%202.617%201.914h2.764c-.05-2.548-2.11-4.238-5.39-4.238-3.145%200-5.392%201.719-5.392%204.316%200%20.363.04.703.12%201.02ZM4%2011a1%201%200%201%200%200%202h15a1%201%200%201%200%200-2H4Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-quote::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M4.581%208.471c.44-.5%201.056-.834%201.758-.995C8.074%207.17%209.201%207.822%2010%208.752c1.354%201.578%201.33%203.555.394%205.277-.941%201.731-2.788%203.163-4.988%203.56a.622.622%200%200%201-.653-.317c-.113-.205-.121-.49.16-.764.294-.286.567-.566.791-.835.222-.266.413-.54.524-.815.113-.28.156-.597.026-.908-.128-.303-.39-.524-.72-.69a3.02%203.02%200%200%201-1.674-2.7c0-.905.283-1.59.72-2.088Zm9.419%200c.44-.5%201.055-.834%201.758-.995%201.734-.306%202.862.346%203.66%201.276%201.355%201.578%201.33%203.555.395%205.277-.941%201.731-2.789%203.163-4.988%203.56a.622.622%200%200%201-.653-.317c-.113-.205-.122-.49.16-.764.294-.286.567-.566.791-.835.222-.266.412-.54.523-.815.114-.28.157-.597.026-.908-.127-.303-.39-.524-.72-.69a3.02%203.02%200%200%201-1.672-2.701c0-.905.283-1.59.72-2.088Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-heading-1::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M21.5%207.5v-3h-12v3H14v13h3v-13h4.5ZM9%2013.5h3.5v-3h-10v3H6v7h3v-7Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-code::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M3.293%2011.293a1%201%200%200%200%200%201.414l4%204a1%201%200%201%200%201.414-1.414L5.414%2012l3.293-3.293a1%201%200%200%200-1.414-1.414l-4%204Zm13.414%205.414%204-4a1%201%200%200%200%200-1.414l-4-4a1%201%200%201%200-1.414%201.414L18.586%2012l-3.293%203.293a1%201%200%200%200%201.414%201.414Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-bullet-list::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M5%207.5a1.5%201.5%200%201%200%200-3%201.5%201.5%200%200%200%200%203ZM8%206a1%201%200%200%201%201-1h11a1%201%200%201%201%200%202H9a1%201%200%200%201-1-1Zm1%205a1%201%200%201%200%200%202h11a1%201%200%201%200%200-2H9Zm0%206a1%201%200%201%200%200%202h11a1%201%200%201%200%200-2H9Zm-2.5-5a1.5%201.5%200%201%201-3%200%201.5%201.5%200%200%201%203%200ZM5%2019.5a1.5%201.5%200%201%200%200-3%201.5%201.5%200%200%200%200%203Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-number-list::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M3%204h2v4H4V5H3V4Zm5%202a1%201%200%200%201%201-1h11a1%201%200%201%201%200%202H9a1%201%200%200%201-1-1Zm1%205a1%201%200%201%200%200%202h11a1%201%200%201%200%200-2H9Zm0%206a1%201%200%201%200%200%202h11a1%201%200%201%200%200-2H9Zm-3.5-7H6v1l-1.5%202H6v1H3v-1l1.667-2H3v-1h2.5ZM3%2017v-1h3v4H3v-1h2v-.5H4v-1h1V17H3Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-undo::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M3%2014a1%201%200%200%200%201%201h6a1%201%200%201%200%200-2H6.257c2.247-2.764%205.151-3.668%207.579-3.264%202.589.432%204.739%202.356%205.174%205.405a1%201%200%200%200%201.98-.283c-.564-3.95-3.415-6.526-6.825-7.095C11.084%207.25%207.63%208.377%205%2011.39V8a1%201%200%200%200-2%200v6Zm2-1Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-redo::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M21%2014a1%201%200%200%201-1%201h-6a1%201%200%201%201%200-2h3.743c-2.247-2.764-5.151-3.668-7.579-3.264-2.589.432-4.739%202.356-5.174%205.405a1%201%200%200%201-1.98-.283c.564-3.95%203.415-6.526%206.826-7.095%203.08-.513%206.534.614%209.164%203.626V8a1%201%200%201%201%202%200v6Zm-2-1Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-decrease-nesting-level::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M5%206a1%201%200%200%201%201-1h12a1%201%200%201%201%200%202H6a1%201%200%200%201-1-1Zm4%205a1%201%200%201%200%200%202h9a1%201%200%201%200%200-2H9Zm-3%206a1%201%200%201%200%200%202h12a1%201%200%201%200%200-2H6Zm-3.707-5.707a1%201%200%200%200%200%201.414l2%202a1%201%200%201%200%201.414-1.414L4.414%2012l1.293-1.293a1%201%200%200%200-1.414-1.414l-2%202Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--icon-increase-nesting-level::before {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20width%3D%2224%22%20height%3D%2224%22%20fill%3D%22none%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M5%206a1%201%200%200%201%201-1h12a1%201%200%201%201%200%202H6a1%201%200%200%201-1-1Zm4%205a1%201%200%201%200%200%202h9a1%201%200%201%200%200-2H9Zm-3%206a1%201%200%201%200%200%202h12a1%201%200%201%200%200-2H6Zm-2.293-2.293%202-2a1%201%200%200%200%200-1.414l-2-2a1%201%200%201%200-1.414%201.414L3.586%2012l-1.293%201.293a1%201%200%201%200%201.414%201.414Z%22%20fill%3D%22%23000%22%2F%3E%3C%2Fsvg%3E"); }
|
||||||
|
|
||||||
|
trix-toolbar .trix-dialogs {
|
||||||
|
position: relative; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-dialog {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
font-size: 0.75em;
|
||||||
|
padding: 15px 10px;
|
||||||
|
background: #fff;
|
||||||
|
box-shadow: 0 0.3em 1em #ccc;
|
||||||
|
border-top: 2px solid #888;
|
||||||
|
border-radius: 5px;
|
||||||
|
z-index: 5; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-input--dialog {
|
||||||
|
font-size: inherit;
|
||||||
|
font-weight: normal;
|
||||||
|
padding: 0.5em 0.8em;
|
||||||
|
margin: 0 10px 0 0;
|
||||||
|
border-radius: 3px;
|
||||||
|
border: 1px solid #bbb;
|
||||||
|
background-color: #fff;
|
||||||
|
box-shadow: none;
|
||||||
|
outline: none;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
-moz-appearance: none; }
|
||||||
|
trix-toolbar .trix-input--dialog.validate:invalid {
|
||||||
|
box-shadow: #F00 0px 0px 1.5px 1px; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-button--dialog {
|
||||||
|
font-size: inherit;
|
||||||
|
padding: 0.5em;
|
||||||
|
border-bottom: none; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-dialog--link {
|
||||||
|
max-width: 600px; }
|
||||||
|
|
||||||
|
trix-toolbar .trix-dialog__link-fields {
|
||||||
|
display: flex;
|
||||||
|
align-items: baseline; }
|
||||||
|
trix-toolbar .trix-dialog__link-fields .trix-input {
|
||||||
|
flex: 1; }
|
||||||
|
trix-toolbar .trix-dialog__link-fields .trix-button-group {
|
||||||
|
flex: 0 0 content;
|
||||||
|
margin: 0; }
|
||||||
|
|
||||||
|
trix-editor [data-trix-mutable]:not(.attachment__caption-editor) {
|
||||||
|
-webkit-user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
-ms-user-select: none;
|
||||||
|
user-select: none; }
|
||||||
|
|
||||||
|
trix-editor [data-trix-mutable]::-moz-selection,
|
||||||
|
trix-editor [data-trix-cursor-target]::-moz-selection, trix-editor [data-trix-mutable] ::-moz-selection {
|
||||||
|
background: none; }
|
||||||
|
|
||||||
|
trix-editor [data-trix-mutable]::selection,
|
||||||
|
trix-editor [data-trix-cursor-target]::selection, trix-editor [data-trix-mutable] ::selection {
|
||||||
|
background: none; }
|
||||||
|
|
||||||
|
trix-editor .attachment__caption-editor:focus[data-trix-mutable]::-moz-selection {
|
||||||
|
background: highlight; }
|
||||||
|
|
||||||
|
trix-editor .attachment__caption-editor:focus[data-trix-mutable]::selection {
|
||||||
|
background: highlight; }
|
||||||
|
|
||||||
|
trix-editor [data-trix-mutable].attachment.attachment--file {
|
||||||
|
box-shadow: 0 0 0 2px highlight;
|
||||||
|
border-color: transparent; }
|
||||||
|
|
||||||
|
trix-editor [data-trix-mutable].attachment img {
|
||||||
|
box-shadow: 0 0 0 2px highlight; }
|
||||||
|
|
||||||
|
trix-editor .attachment {
|
||||||
|
position: relative; }
|
||||||
|
trix-editor .attachment:hover {
|
||||||
|
cursor: default; }
|
||||||
|
|
||||||
|
trix-editor .attachment--preview .attachment__caption:hover {
|
||||||
|
cursor: text; }
|
||||||
|
|
||||||
|
trix-editor .attachment__progress {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1;
|
||||||
|
height: 20px;
|
||||||
|
top: calc(50% - 10px);
|
||||||
|
left: 5%;
|
||||||
|
width: 90%;
|
||||||
|
opacity: 0.9;
|
||||||
|
transition: opacity 200ms ease-in; }
|
||||||
|
trix-editor .attachment__progress[value="100"] {
|
||||||
|
opacity: 0; }
|
||||||
|
|
||||||
|
trix-editor .attachment__caption-editor {
|
||||||
|
display: inline-block;
|
||||||
|
width: 100%;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
font-size: inherit;
|
||||||
|
font-family: inherit;
|
||||||
|
line-height: inherit;
|
||||||
|
color: inherit;
|
||||||
|
text-align: center;
|
||||||
|
vertical-align: top;
|
||||||
|
border: none;
|
||||||
|
outline: none;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
-moz-appearance: none; }
|
||||||
|
|
||||||
|
trix-editor .attachment__toolbar {
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1;
|
||||||
|
top: -0.9em;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
text-align: center; }
|
||||||
|
|
||||||
|
trix-editor .trix-button-group {
|
||||||
|
display: inline-flex; }
|
||||||
|
|
||||||
|
trix-editor .trix-button {
|
||||||
|
position: relative;
|
||||||
|
float: left;
|
||||||
|
color: #666;
|
||||||
|
white-space: nowrap;
|
||||||
|
font-size: 80%;
|
||||||
|
padding: 0 0.8em;
|
||||||
|
margin: 0;
|
||||||
|
outline: none;
|
||||||
|
border: none;
|
||||||
|
border-radius: 0;
|
||||||
|
background: transparent; }
|
||||||
|
trix-editor .trix-button:not(:first-child) {
|
||||||
|
border-left: 1px solid #ccc; }
|
||||||
|
trix-editor .trix-button.trix-active {
|
||||||
|
background: #cbeefa; }
|
||||||
|
trix-editor .trix-button:not(:disabled) {
|
||||||
|
cursor: pointer; }
|
||||||
|
|
||||||
|
trix-editor .trix-button--remove {
|
||||||
|
text-indent: -9999px;
|
||||||
|
display: inline-block;
|
||||||
|
padding: 0;
|
||||||
|
outline: none;
|
||||||
|
width: 1.8em;
|
||||||
|
height: 1.8em;
|
||||||
|
line-height: 1.8em;
|
||||||
|
border-radius: 50%;
|
||||||
|
background-color: #fff;
|
||||||
|
border: 2px solid highlight;
|
||||||
|
box-shadow: 1px 1px 6px rgba(0, 0, 0, 0.25); }
|
||||||
|
trix-editor .trix-button--remove::before {
|
||||||
|
display: inline-block;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
opacity: 0.7;
|
||||||
|
content: "";
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg%20height%3D%2224%22%20width%3D%2224%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M19%206.41%2017.59%205%2012%2010.59%206.41%205%205%206.41%2010.59%2012%205%2017.59%206.41%2019%2012%2013.41%2017.59%2019%2019%2017.59%2013.41%2012z%22%2F%3E%3Cpath%20d%3D%22M0%200h24v24H0z%22%20fill%3D%22none%22%2F%3E%3C%2Fsvg%3E");
|
||||||
|
background-position: center;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: 90%; }
|
||||||
|
trix-editor .trix-button--remove:hover {
|
||||||
|
border-color: #333; }
|
||||||
|
trix-editor .trix-button--remove:hover::before {
|
||||||
|
opacity: 1; }
|
||||||
|
|
||||||
|
trix-editor .attachment__metadata-container {
|
||||||
|
position: relative; }
|
||||||
|
|
||||||
|
trix-editor .attachment__metadata {
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
top: 2em;
|
||||||
|
transform: translate(-50%, 0);
|
||||||
|
max-width: 90%;
|
||||||
|
padding: 0.1em 0.6em;
|
||||||
|
font-size: 0.8em;
|
||||||
|
color: #fff;
|
||||||
|
background-color: rgba(0, 0, 0, 0.7);
|
||||||
|
border-radius: 3px; }
|
||||||
|
trix-editor .attachment__metadata .attachment__name {
|
||||||
|
display: inline-block;
|
||||||
|
max-width: 100%;
|
||||||
|
vertical-align: bottom;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap; }
|
||||||
|
trix-editor .attachment__metadata .attachment__size {
|
||||||
|
margin-left: 0.2em;
|
||||||
|
white-space: nowrap; }
|
||||||
|
|
||||||
|
.trix-content {
|
||||||
|
line-height: 1.5;
|
||||||
|
overflow-wrap: break-word;
|
||||||
|
word-break: break-word; }
|
||||||
|
.trix-content * {
|
||||||
|
box-sizing: border-box;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0; }
|
||||||
|
.trix-content h1 {
|
||||||
|
font-size: 1.2em;
|
||||||
|
line-height: 1.2; }
|
||||||
|
.trix-content blockquote {
|
||||||
|
border: 0 solid #ccc;
|
||||||
|
border-left-width: 0.3em;
|
||||||
|
margin-left: 0.3em;
|
||||||
|
padding-left: 0.6em; }
|
||||||
|
.trix-content [dir=rtl] blockquote,
|
||||||
|
.trix-content blockquote[dir=rtl] {
|
||||||
|
border-width: 0;
|
||||||
|
border-right-width: 0.3em;
|
||||||
|
margin-right: 0.3em;
|
||||||
|
padding-right: 0.6em; }
|
||||||
|
.trix-content li {
|
||||||
|
margin-left: 1em; }
|
||||||
|
.trix-content [dir=rtl] li {
|
||||||
|
margin-right: 1em; }
|
||||||
|
.trix-content pre {
|
||||||
|
display: inline-block;
|
||||||
|
width: 100%;
|
||||||
|
vertical-align: top;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 0.9em;
|
||||||
|
padding: 0.5em;
|
||||||
|
white-space: pre;
|
||||||
|
background-color: #eee;
|
||||||
|
overflow-x: auto; }
|
||||||
|
.trix-content img {
|
||||||
|
max-width: 100%;
|
||||||
|
height: auto; }
|
||||||
|
.trix-content .attachment {
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
max-width: 100%; }
|
||||||
|
.trix-content .attachment a {
|
||||||
|
color: inherit;
|
||||||
|
text-decoration: none; }
|
||||||
|
.trix-content .attachment a:hover, .trix-content .attachment a:visited:hover {
|
||||||
|
color: inherit; }
|
||||||
|
.trix-content .attachment__caption {
|
||||||
|
text-align: center; }
|
||||||
|
.trix-content .attachment__caption .attachment__name + .attachment__size::before {
|
||||||
|
content: ' \2022 '; }
|
||||||
|
.trix-content .attachment--preview {
|
||||||
|
width: 100%;
|
||||||
|
text-align: center; }
|
||||||
|
.trix-content .attachment--preview .attachment__caption {
|
||||||
|
color: #666;
|
||||||
|
font-size: 0.9em;
|
||||||
|
line-height: 1.2; }
|
||||||
|
.trix-content .attachment--file {
|
||||||
|
color: #333;
|
||||||
|
line-height: 1;
|
||||||
|
margin: 0 2px 2px 2px;
|
||||||
|
padding: 0.4em 1em;
|
||||||
|
border: 1px solid #bbb;
|
||||||
|
border-radius: 5px; }
|
||||||
|
.trix-content .attachment-gallery {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
position: relative; }
|
||||||
|
.trix-content .attachment-gallery .attachment {
|
||||||
|
flex: 1 0 33%;
|
||||||
|
padding: 0 0.5em;
|
||||||
|
max-width: 33%; }
|
||||||
|
.trix-content .attachment-gallery.attachment-gallery--2 .attachment, .trix-content .attachment-gallery.attachment-gallery--4 .attachment {
|
||||||
|
flex-basis: 50%;
|
||||||
|
max-width: 50%; }
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We need to override trix.css’s image gallery styles to accommodate the
|
||||||
|
* <action-text-attachment> element we wrap around attachments. Otherwise,
|
||||||
|
* images in galleries will be squished by the max-width: 33%; rule.
|
||||||
|
*/
|
||||||
|
.trix-content .attachment-gallery > action-text-attachment,
|
||||||
|
.trix-content .attachment-gallery > .attachment {
|
||||||
|
flex: 1 0 33%;
|
||||||
|
padding: 0 0.5em;
|
||||||
|
max-width: 33%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.trix-content .attachment-gallery.attachment-gallery--2 > action-text-attachment,
|
||||||
|
.trix-content .attachment-gallery.attachment-gallery--2 > .attachment, .trix-content .attachment-gallery.attachment-gallery--4 > action-text-attachment,
|
||||||
|
.trix-content .attachment-gallery.attachment-gallery--4 > .attachment {
|
||||||
|
flex-basis: 50%;
|
||||||
|
max-width: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.trix-content action-text-attachment .attachment {
|
||||||
|
padding: 0 !important;
|
||||||
|
max-width: 100% !important;
|
||||||
|
}
|
||||||
292
app/assets/stylesheets/application.css
Normal file
292
app/assets/stylesheets/application.css
Normal file
|
|
@ -0,0 +1,292 @@
|
||||||
|
/*
|
||||||
|
* This is a manifest file that'll be compiled into application.css.
|
||||||
|
*
|
||||||
|
* With Propshaft, assets are served efficiently without preprocessing steps. You can still include
|
||||||
|
* application-wide styles in this file, but keep in mind that CSS precedence will follow the standard
|
||||||
|
* cascading order, meaning styles declared later in the document or manifest will override earlier ones,
|
||||||
|
* depending on specificity.
|
||||||
|
*
|
||||||
|
* Consider organizing styles into separate files for maintainability.
|
||||||
|
*/
|
||||||
|
// <weight>: Use a value from 300 to 900
|
||||||
|
// <uniquifier>: Use a unique and descriptive class name
|
||||||
|
|
||||||
|
.winky-sans-all{
|
||||||
|
font-family: "Winky Sans", sans-serif;
|
||||||
|
font-optical-sizing: auto;
|
||||||
|
font-weight: 600;
|
||||||
|
font-style: normal;
|
||||||
|
}
|
||||||
|
table.intereststable {
|
||||||
|
|
||||||
|
padding: 4px;
|
||||||
|
}
|
||||||
|
section.bottomindex {
|
||||||
|
border: 1px solid black;
|
||||||
|
width: 990px;
|
||||||
|
height: 240px;
|
||||||
|
position: absolute;
|
||||||
|
overflow: scroll;
|
||||||
|
margin-top: 400px;
|
||||||
|
}
|
||||||
|
section.etc {
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 10px;
|
||||||
|
width:260px;
|
||||||
|
height: 350px;
|
||||||
|
overflow: scroll;
|
||||||
|
position: absolute;
|
||||||
|
margin-left: 730px;
|
||||||
|
}
|
||||||
|
section.blogdesc {
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
#minty {
|
||||||
|
padding: .5em;
|
||||||
|
background-color: #f0faf5;
|
||||||
|
border: 1px solid black;
|
||||||
|
}
|
||||||
|
#minty-username {
|
||||||
|
margin-bottom: .5em;
|
||||||
|
}
|
||||||
|
.marquee {
|
||||||
|
position: absolute;
|
||||||
|
width:100%;
|
||||||
|
margin-top: 25px;
|
||||||
|
}
|
||||||
|
#minty-content {
|
||||||
|
margin: 0 1em 0.5em 1em;
|
||||||
|
}
|
||||||
|
section.aboutus {
|
||||||
|
border: 1px solid black;
|
||||||
|
padding:10px;
|
||||||
|
width: 470px;
|
||||||
|
height: 350px;
|
||||||
|
text-align: center;
|
||||||
|
position: absolute;
|
||||||
|
margin-left:230px;
|
||||||
|
overflow: scroll;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
section.latestpost {
|
||||||
|
position: absolute;
|
||||||
|
width: 215px;
|
||||||
|
height: 350px;
|
||||||
|
overflow: scroll;
|
||||||
|
margin-left: -15px;
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
|
section.posts {
|
||||||
|
margin: 0 auto;
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 4px;
|
||||||
|
}
|
||||||
|
section.interests {
|
||||||
|
float: right;
|
||||||
|
width: 250px;
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 1px;
|
||||||
|
}
|
||||||
|
section.createmember {
|
||||||
|
float: right;
|
||||||
|
top: 0px;
|
||||||
|
}
|
||||||
|
section.membertable {
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 3%;
|
||||||
|
}
|
||||||
|
section.memberspecific {
|
||||||
|
border: 1px solid black;
|
||||||
|
}
|
||||||
|
table {
|
||||||
|
padding: 3%;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
tr.member {
|
||||||
|
border: 1px solid black;
|
||||||
|
}
|
||||||
|
td.member {
|
||||||
|
padding: 25px;
|
||||||
|
|
||||||
|
}
|
||||||
|
td.memberinterests {
|
||||||
|
border: 1px solid black;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
section.memberlist {
|
||||||
|
width: 850px;
|
||||||
|
height: auto;
|
||||||
|
border: 1px solid black;
|
||||||
|
text-align: center;
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: 9px;
|
||||||
|
}
|
||||||
|
section.avatarimg {
|
||||||
|
border: 3px double black;
|
||||||
|
width: 100px;
|
||||||
|
height: 100px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
section.memberinner {
|
||||||
|
border:1px solid black;
|
||||||
|
width: 250px;
|
||||||
|
height: auto;
|
||||||
|
float: left;
|
||||||
|
padding: 4px;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
color: #000;
|
||||||
|
font-family: 'Comic Sans MS';
|
||||||
|
|
||||||
|
background: #ff597d;
|
||||||
|
background: linear-gradient(90deg, rgba(255, 89, 125, 1) 0%, rgba(97, 255, 136, 1) 50%, rgba(237, 83, 129, 1) 100%);
|
||||||
|
font-size:17px;
|
||||||
|
}
|
||||||
|
section.inner {
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
section.about {
|
||||||
|
|
||||||
|
text-align: center;
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 450px;
|
||||||
|
overflow: scroll;
|
||||||
|
border:2px outset #000;
|
||||||
|
}
|
||||||
|
section.footer {
|
||||||
|
background: #ffe0e8;
|
||||||
|
background: radial-gradient(circle, rgba(255, 224, 232, 1) 0%, rgba(180, 250, 197, 1) 50%, rgba(255, 207, 219, 1) 100%);
|
||||||
|
width: 1000px;
|
||||||
|
color: #000;
|
||||||
|
height: 30px;
|
||||||
|
text-align: center;
|
||||||
|
padding-left: 2%;
|
||||||
|
padding-right: 2%;
|
||||||
|
padding-bottom: 1%;
|
||||||
|
margin: auto;
|
||||||
|
border:3px outset #000;
|
||||||
|
}
|
||||||
|
h1 {
|
||||||
|
font-family: "Comic Sans MS", cursive, sans-serif;
|
||||||
|
background-color: pink;
|
||||||
|
border: 5px outset #42f548;
|
||||||
|
color: #000;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.profilenews {
|
||||||
|
background: #fff;
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
main {
|
||||||
|
width: 1000px;
|
||||||
|
overflow: scroll;
|
||||||
|
height: 1000px;
|
||||||
|
margin: 0 auto;
|
||||||
|
background: #ffe0e8;
|
||||||
|
background: radial-gradient(circle, rgba(255, 224, 232, 1) 0%, rgba(180, 250, 197, 1) 50%, rgba(255, 207, 219, 1) 100%);
|
||||||
|
border: 8px groove #ff5975;
|
||||||
|
box-shadow: 8px 2px 5px 2px #a3ffc3;
|
||||||
|
padding-left: 2%;
|
||||||
|
padding-right: 2%;
|
||||||
|
|
||||||
|
}
|
||||||
|
.header {
|
||||||
|
text-align: center;
|
||||||
|
background: #ffe0e8;
|
||||||
|
background: radial-gradient(circle, rgba(255, 224, 232, 1) 0%, rgba(180, 250, 197, 1) 50%, rgba(255, 207, 219, 1) 100%);
|
||||||
|
}
|
||||||
|
.headerall {
|
||||||
|
color: #000;
|
||||||
|
background: #ffe0e8;
|
||||||
|
background: radial-gradient(circle, rgba(255, 224, 232, 1) 0%, rgba(180, 250, 197, 1) 50%, rgba(255, 207, 219, 1) 100%);
|
||||||
|
}
|
||||||
|
a {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
section.profileinner {
|
||||||
|
text-align: left;
|
||||||
|
padding: 15px;
|
||||||
|
}
|
||||||
|
section.article {
|
||||||
|
text-align: center;
|
||||||
|
background-color: #fff;
|
||||||
|
padding: 5px;
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
section.profile {
|
||||||
|
text-align: center;
|
||||||
|
background-color: #fff;
|
||||||
|
padding: 15px;
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
h1.h1bio {
|
||||||
|
width: 350px;
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
section.homepage {
|
||||||
|
|
||||||
|
}
|
||||||
|
a:visited {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
a:hover {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
section.auth {
|
||||||
|
float:right;
|
||||||
|
box-shadow: 3px 3px 3px gray;
|
||||||
|
background: #fff;
|
||||||
|
color: #000;
|
||||||
|
|
||||||
|
}
|
||||||
|
section.newstuff {
|
||||||
|
width:200px;
|
||||||
|
border:5px outset #42f548;
|
||||||
|
background: pink;
|
||||||
|
color: #000;
|
||||||
|
text-align: center;
|
||||||
|
margin: auto;
|
||||||
|
padding: 3px;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
section.articleshow {
|
||||||
|
align-content: center;
|
||||||
|
background: pink;
|
||||||
|
color: #000;
|
||||||
|
border:5px outset #42f548;
|
||||||
|
}
|
||||||
|
section.newstuff a, a:visited, a:hover {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
table.articleshowing {
|
||||||
|
margin: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.articleshowing a {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.articleshowing a:visited {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
table.articleshowing a:hover {
|
||||||
|
color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
section.profilelist {
|
||||||
|
margin: auto;
|
||||||
|
width: 300px;
|
||||||
|
align-content: center;
|
||||||
|
padding: 10px;
|
||||||
|
overflow: scroll;
|
||||||
|
background-color: pink;
|
||||||
|
color: #000;
|
||||||
|
border:5px outset #42f548;
|
||||||
|
}
|
||||||
16
app/channels/application_cable/connection.rb
Normal file
16
app/channels/application_cable/connection.rb
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
module ApplicationCable
|
||||||
|
class Connection < ActionCable::Connection::Base
|
||||||
|
identified_by :current_user
|
||||||
|
|
||||||
|
def connect
|
||||||
|
set_current_user || reject_unauthorized_connection
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def set_current_user
|
||||||
|
if session = Session.find_by(id: cookies.signed[:session_id])
|
||||||
|
self.current_user = session.user
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
22
app/controllers/application_controller.rb
Normal file
22
app/controllers/application_controller.rb
Normal file
|
|
@ -0,0 +1,22 @@
|
||||||
|
class ApplicationController < ActionController::Base
|
||||||
|
include Authentication
|
||||||
|
# Only allow modern browsers supporting webp images, web push, badges, import maps, CSS nesting, and CSS :has.
|
||||||
|
allow_browser versions: :modern
|
||||||
|
|
||||||
|
# Changes to the importmap will invalidate the etag for HTML responses
|
||||||
|
stale_when_importmap_changes
|
||||||
|
helper_method :current_user
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def current_user
|
||||||
|
|
||||||
|
@current_user ||= User.find_by(id: session[:user_id]) if session[:user_id]
|
||||||
|
end
|
||||||
|
|
||||||
|
def authenticate_user!
|
||||||
|
unless current_user
|
||||||
|
redirect_to login_path, alert: "You must be logged in to access this page."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
74
app/controllers/articles_controller.rb
Normal file
74
app/controllers/articles_controller.rb
Normal file
|
|
@ -0,0 +1,74 @@
|
||||||
|
class ArticlesController < ApplicationController
|
||||||
|
before_action :authenticate_user!, only: [:new, :edit, :update, :create]
|
||||||
|
allow_unauthenticated_access(only: %i[index show])
|
||||||
|
|
||||||
|
def feed
|
||||||
|
@articles = Article.order(created_at: :asc)
|
||||||
|
response.headers['Content-Type'] = 'application/rss+xml'
|
||||||
|
render 'feed', formats: :xml
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def index
|
||||||
|
@articles = Article.order(created_at: :desc)
|
||||||
|
@members = Member.all
|
||||||
|
@blogs = Blog.all
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def show
|
||||||
|
@article = Article.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def new
|
||||||
|
@article = Article.new
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def create
|
||||||
|
@article = Article.new(article_params)
|
||||||
|
if @article.save
|
||||||
|
redirect_to @article
|
||||||
|
else
|
||||||
|
render 'new'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def edit
|
||||||
|
@article = Article.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def update
|
||||||
|
@article = Article.find(params[:id])
|
||||||
|
if @article.update(article_params)
|
||||||
|
redirect_to @article
|
||||||
|
else
|
||||||
|
render :edit
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@article = Article.find(params[:id])
|
||||||
|
@article.destroy
|
||||||
|
redirect_to articles_path
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
|
||||||
|
def article_params
|
||||||
|
params.require(:article).permit(:title, :text, :mood, :music, :icon_image)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
20
app/controllers/blogmap_controller.rb
Normal file
20
app/controllers/blogmap_controller.rb
Normal file
|
|
@ -0,0 +1,20 @@
|
||||||
|
class BlogmapController < ApplicationController
|
||||||
|
allow_unauthenticated_access(only: %i[index show])
|
||||||
|
before_action :authenticate_user!, only: [:new, :edit, :update, :create]
|
||||||
|
|
||||||
|
def index
|
||||||
|
@blogs = Blog.all
|
||||||
|
end
|
||||||
|
def new
|
||||||
|
end
|
||||||
|
def update
|
||||||
|
end
|
||||||
|
def edit
|
||||||
|
end
|
||||||
|
def create
|
||||||
|
end
|
||||||
|
def destroy
|
||||||
|
end
|
||||||
|
def show
|
||||||
|
end
|
||||||
|
end
|
||||||
67
app/controllers/blogs_controller.rb
Normal file
67
app/controllers/blogs_controller.rb
Normal file
|
|
@ -0,0 +1,67 @@
|
||||||
|
class BlogsController < ApplicationController
|
||||||
|
before_action :authenticate_user!, only: [:destroy, :new, :create, :edit, :update]
|
||||||
|
allow_unauthenticated_access(only: %i[index show])
|
||||||
|
before_action :set_member
|
||||||
|
before_action :set_blog, only: [:show, :edit, :update, :destroy]
|
||||||
|
def index
|
||||||
|
@posts = Post.order(created_at: :asc)
|
||||||
|
end
|
||||||
|
|
||||||
|
def edit
|
||||||
|
@blog = Blog.find(params[:id])
|
||||||
|
end
|
||||||
|
def new
|
||||||
|
@blog = @member.blogs.new
|
||||||
|
@blog = @member.blogs.build
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def create
|
||||||
|
@blog = @member.blogs.new(blog_params)
|
||||||
|
|
||||||
|
if @blog.save
|
||||||
|
|
||||||
|
redirect_to root_path, notice: 'Blog was successfully created.'
|
||||||
|
else
|
||||||
|
|
||||||
|
render :new, notice: 'Could not save blog.'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
def destroy
|
||||||
|
@member = Member.find(params[:member_id])
|
||||||
|
@blog = @member.blogs.find(params[:id])
|
||||||
|
@blog.destroy
|
||||||
|
redirect_to member_path(@member), notice: "Blog deleted."
|
||||||
|
end
|
||||||
|
def show
|
||||||
|
@member = Member.find(params[:member_id])
|
||||||
|
@blog = @member.blogs.find(params[:id])
|
||||||
|
@posts = @blog.posts.order(created_at: :desc)
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
@blog = Blog.find(params[:id])
|
||||||
|
|
||||||
|
if @blog.update(blog_params)
|
||||||
|
redirect_to member_blog_path(@member)
|
||||||
|
else
|
||||||
|
render :edit
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
private
|
||||||
|
|
||||||
|
def blog_params
|
||||||
|
params.require(:blog).permit(:title, :links, :description)
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def set_member
|
||||||
|
@member = Member.find(params[:member_id])
|
||||||
|
end
|
||||||
|
def set_blog
|
||||||
|
@blog = @member.blogs.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
42
app/controllers/comments_controller.rb
Normal file
42
app/controllers/comments_controller.rb
Normal file
|
|
@ -0,0 +1,42 @@
|
||||||
|
class CommentsController < ApplicationController
|
||||||
|
allow_unauthenticated_access
|
||||||
|
before_action :set_member
|
||||||
|
before_action :set_blog
|
||||||
|
before_action :set_post
|
||||||
|
|
||||||
|
def create
|
||||||
|
@comment = @post.comments.build(comment_params)
|
||||||
|
if @comment.save
|
||||||
|
redirect_to member_blog_post_path(@member, @blog, @post), notice: "Comment added!"
|
||||||
|
else
|
||||||
|
redirect_to member_blog_post_path(@member, @blog, @post), alert: "Could not save comment."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@comment = @post.comments.find(params[:id])
|
||||||
|
@comment.destroy
|
||||||
|
|
||||||
|
redirect_to member_blog_post_path(@member, @blog, @post),
|
||||||
|
notice: "Comment deleted."
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_member
|
||||||
|
@member = Member.find(params[:member_id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_blog
|
||||||
|
@blog = @member.blogs.find(params[:blog_id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_post
|
||||||
|
@post = @blog.posts.find(params[:post_id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def comment_params
|
||||||
|
params.require(:comment).permit(:commenter, :website, :body)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
0
app/controllers/concerns/.keep
Normal file
0
app/controllers/concerns/.keep
Normal file
54
app/controllers/concerns/authentication.rb
Normal file
54
app/controllers/concerns/authentication.rb
Normal file
|
|
@ -0,0 +1,54 @@
|
||||||
|
module Authentication
|
||||||
|
extend ActiveSupport::Concern
|
||||||
|
|
||||||
|
included do
|
||||||
|
before_action :require_authentication, except: [:new, :create, :destroy]
|
||||||
|
helper_method :authenticated?
|
||||||
|
end
|
||||||
|
|
||||||
|
class_methods do
|
||||||
|
def allow_unauthenticated_access(**options)
|
||||||
|
skip_before_action :require_authentication, **options
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def authenticated?
|
||||||
|
resume_session
|
||||||
|
end
|
||||||
|
|
||||||
|
def require_authentication
|
||||||
|
resume_session || request_authentication
|
||||||
|
end
|
||||||
|
|
||||||
|
def resume_session
|
||||||
|
Current.session ||= find_session_by_cookie
|
||||||
|
end
|
||||||
|
|
||||||
|
def find_session_by_cookie
|
||||||
|
Session.find_by(id: cookies.signed[:session_id]) if cookies.signed[:session_id]
|
||||||
|
end
|
||||||
|
|
||||||
|
def request_authentication
|
||||||
|
session[:return_to_after_authenticating] = request.url
|
||||||
|
redirect_to new_session_path
|
||||||
|
end
|
||||||
|
|
||||||
|
def after_authentication_url
|
||||||
|
session.delete(:return_to_after_authenticating) || root_url
|
||||||
|
end
|
||||||
|
|
||||||
|
def start_new_session_for(user)
|
||||||
|
user.sessions.create!(user_agent: request.user_agent, ip_address: request.remote_ip).tap do |session|
|
||||||
|
Current.session = session
|
||||||
|
cookies.signed.permanent[:session_id] = { value: session.id, httponly: true, same_site: :lax }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def terminate_session
|
||||||
|
Current.session.destroy
|
||||||
|
cookies.delete(:session_id)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
2
app/controllers/lists_controller.rb
Normal file
2
app/controllers/lists_controller.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
class ListsController < ApplicationController
|
||||||
|
end
|
||||||
57
app/controllers/members_controller.rb
Normal file
57
app/controllers/members_controller.rb
Normal file
|
|
@ -0,0 +1,57 @@
|
||||||
|
class MembersController < ApplicationController
|
||||||
|
before_action :authenticate_user!, only: [:new, :edit, :update, :create, :destroy]
|
||||||
|
allow_unauthenticated_access(only: %i[index show])
|
||||||
|
def index
|
||||||
|
@members = Member.all
|
||||||
|
end
|
||||||
|
|
||||||
|
def blogs
|
||||||
|
@blog = Blog.all
|
||||||
|
end
|
||||||
|
def edit
|
||||||
|
@member = Member.find(params[:id])
|
||||||
|
end
|
||||||
|
def new
|
||||||
|
@member = current_user.members.new
|
||||||
|
end
|
||||||
|
def update
|
||||||
|
@member = Member.find(params[:id])
|
||||||
|
if @member.update(member_params)
|
||||||
|
redirect_to @member
|
||||||
|
else
|
||||||
|
render :edit
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
@member = current_user.members.new(member_params)
|
||||||
|
|
||||||
|
if @member.save
|
||||||
|
redirect_to @member
|
||||||
|
else
|
||||||
|
redirect_to root_path
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@member = Member.find(params[:id])
|
||||||
|
@member.destroy
|
||||||
|
redirect_to member_path(@member), notice: "Member deleted."
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
def show
|
||||||
|
@member = Member.find_by(id: params[:id])
|
||||||
|
if @member.nil?
|
||||||
|
flash[:alert] = "Member not found"
|
||||||
|
redirect_to root_path
|
||||||
|
return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
private
|
||||||
|
|
||||||
|
def member_params
|
||||||
|
params.require(:member).permit(:name, :avatar_image, :bio, :links, :birthday, :pronouns, :system_role, :age, :tv_interests, :music_interests, :other_interests)
|
||||||
|
end
|
||||||
|
|
||||||
2
app/controllers/pages_controller.rb
Normal file
2
app/controllers/pages_controller.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
class PagesController < ApplicationController
|
||||||
|
end
|
||||||
35
app/controllers/passwords_controller.rb
Normal file
35
app/controllers/passwords_controller.rb
Normal file
|
|
@ -0,0 +1,35 @@
|
||||||
|
class PasswordsController < ApplicationController
|
||||||
|
allow_unauthenticated_access
|
||||||
|
before_action :set_user_by_token, only: %i[ edit update ]
|
||||||
|
rate_limit to: 10, within: 3.minutes, only: :create, with: -> { redirect_to new_password_path, alert: "Try again later." }
|
||||||
|
|
||||||
|
def new
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
if user = User.find_by(email_address: params[:email_address])
|
||||||
|
PasswordsMailer.reset(user).deliver_later
|
||||||
|
end
|
||||||
|
|
||||||
|
redirect_to new_session_path, notice: "Password reset instructions sent (if user with that email address exists)."
|
||||||
|
end
|
||||||
|
|
||||||
|
def edit
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
if @user.update(params.permit(:password, :password_confirmation))
|
||||||
|
@user.sessions.destroy_all
|
||||||
|
redirect_to new_session_path, notice: "Password has been reset."
|
||||||
|
else
|
||||||
|
redirect_to edit_password_path(params[:token]), alert: "Passwords did not match."
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
def set_user_by_token
|
||||||
|
@user = User.find_by_password_reset_token!(params[:token])
|
||||||
|
rescue ActiveSupport::MessageVerifier::InvalidSignature
|
||||||
|
redirect_to new_password_path, alert: "Password reset link is invalid or has expired."
|
||||||
|
end
|
||||||
|
end
|
||||||
64
app/controllers/posts_controller.rb
Normal file
64
app/controllers/posts_controller.rb
Normal file
|
|
@ -0,0 +1,64 @@
|
||||||
|
class PostsController < ApplicationController
|
||||||
|
before_action :authenticate_user!, only: [:new, :create, :edit, :update, :destroy]
|
||||||
|
allow_unauthenticated_access(only: [:index, :show])
|
||||||
|
|
||||||
|
before_action :set_member
|
||||||
|
before_action :set_blog
|
||||||
|
before_action :set_post, only: [:show, :edit, :update, :destroy]
|
||||||
|
|
||||||
|
def index
|
||||||
|
@posts = @blog.posts
|
||||||
|
end
|
||||||
|
|
||||||
|
def show
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
def new
|
||||||
|
@post = @blog.posts.build
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
@post = @blog.posts.build(post_params)
|
||||||
|
if @post.save
|
||||||
|
redirect_to member_blog_post_path(@member, @blog, @post), notice: "Post created."
|
||||||
|
else
|
||||||
|
render :new, status: :unprocessable_entity
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def edit
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
if @post.update(post_params)
|
||||||
|
redirect_to member_blog_post_path(@member, @blog, @post), notice: "Post updated."
|
||||||
|
else
|
||||||
|
render :edit, status: :unprocessable_entity
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@post.destroy
|
||||||
|
redirect_to member_blog_path(@member, @blog), notice: "Post deleted."
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_member
|
||||||
|
@member = Member.find(params[:member_id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_blog
|
||||||
|
@blog = @member.blogs.find(params[:blog_id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_post
|
||||||
|
@post = @blog.posts.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def post_params
|
||||||
|
params.require(:post).permit(:post_title, :text, :mood, :music, :icon_image)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
64
app/controllers/profiles_controller.rb
Normal file
64
app/controllers/profiles_controller.rb
Normal file
|
|
@ -0,0 +1,64 @@
|
||||||
|
class ProfilesController < ApplicationController
|
||||||
|
allow_unauthenticated_access only: %i[ index show ]
|
||||||
|
before_action :authenticate_user!, only: [:new, :create, :edit, :update]
|
||||||
|
before_action :set_profile, only: [:edit, :update, :update]
|
||||||
|
|
||||||
|
def index
|
||||||
|
@profiles = Profile.all
|
||||||
|
end
|
||||||
|
|
||||||
|
def new
|
||||||
|
@profile = Profile.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
@profile = Profile.create(profile_params)
|
||||||
|
if @profile.save
|
||||||
|
|
||||||
|
redirect_to @profile, notice: 'Profile was successfully created.'
|
||||||
|
else
|
||||||
|
|
||||||
|
render :new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
def edit
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
if @profile.update(profile_params)
|
||||||
|
redirect_to @profile, notice: 'Profile was successfully updated!'
|
||||||
|
else
|
||||||
|
render :edit
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def show
|
||||||
|
@profile = Profile.find_by(id: params[:id])
|
||||||
|
|
||||||
|
if @profile.nil?
|
||||||
|
flash[:alert] = "Profile not found"
|
||||||
|
redirect_to root_path
|
||||||
|
end
|
||||||
|
end
|
||||||
|
def destroy
|
||||||
|
@profile = Profile.find(params[:id])
|
||||||
|
@profile.destroy
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def set_profile
|
||||||
|
@profile = Profile.find_by(id: params[:id]) # Find profile by ID passed in the URL
|
||||||
|
if @profile.nil?
|
||||||
|
redirect_to profiles_path, alert: 'Profile not found.'
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def profile_params
|
||||||
|
params.require(:profile).permit(:bio, :name, :avatar_image, :age, :location, :website, :pronouns)
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
31
app/controllers/sessions_controller.rb
Normal file
31
app/controllers/sessions_controller.rb
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
class SessionsController < ApplicationController
|
||||||
|
|
||||||
|
allow_unauthenticated_access only: %i[new create]
|
||||||
|
rate_limit to: 10, within: 3.minutes, only: :create, with: -> { redirect_to new_session_path, alert: "Try again later." }
|
||||||
|
|
||||||
|
|
||||||
|
def new
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def create
|
||||||
|
|
||||||
|
user = User.find_by(email_address: params[:email_address])
|
||||||
|
|
||||||
|
if user&.authenticate(params[:password])
|
||||||
|
session[:user_id] = user.id
|
||||||
|
start_new_session_for(user)
|
||||||
|
redirect_to root_path, notice: "Signed in!"
|
||||||
|
else
|
||||||
|
flash[:alert] = "Invalid email or password"
|
||||||
|
render :new, status: :unprocessable_entity
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
session.delete(:user_id)
|
||||||
|
reset_session
|
||||||
|
redirect_to root_path, notice: "Signed out!"
|
||||||
|
end
|
||||||
|
end
|
||||||
51
app/controllers/sitemaps_controller.rb
Normal file
51
app/controllers/sitemaps_controller.rb
Normal file
|
|
@ -0,0 +1,51 @@
|
||||||
|
class SitemapsController < ApplicationController
|
||||||
|
before_action :authenticate_user!, only: %i[new edit update create destroy]
|
||||||
|
allow_unauthenticated_access only: %i[index show]
|
||||||
|
|
||||||
|
def index
|
||||||
|
@sitemap = Sitemap.all
|
||||||
|
end
|
||||||
|
|
||||||
|
def new
|
||||||
|
@sitemap = Sitemap.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def create
|
||||||
|
@sitemap = Sitemap.new(sitemap_params)
|
||||||
|
if @sitemap.save
|
||||||
|
redirect_to @sitemap
|
||||||
|
else
|
||||||
|
render :new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def show
|
||||||
|
@sitemap = Sitemap.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def edit
|
||||||
|
@sitemap = Sitemap.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
|
def update
|
||||||
|
@sitemap = Sitemap.find(params[:id])
|
||||||
|
if @sitemap.update(sitemap_params)
|
||||||
|
redirect_to @sitemap
|
||||||
|
else
|
||||||
|
render :edit
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
@sitemap = Sitemap.find(params[:id])
|
||||||
|
@sitemap.destroy
|
||||||
|
redirect_to sitemaps_path
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def sitemap_params
|
||||||
|
params.require(:sitemap).permit(:text)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
2
app/controllers/updates_controller.rb
Normal file
2
app/controllers/updates_controller.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
class UpdatesController < ApplicationController
|
||||||
|
end
|
||||||
19
app/controllers/welcome_controller.rb
Normal file
19
app/controllers/welcome_controller.rb
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
class WelcomeController < ApplicationController
|
||||||
|
allow_unauthenticated_access(only: %i[index show])
|
||||||
|
def index
|
||||||
|
@latest_post = Post.order(created_at: :desc).first
|
||||||
|
|
||||||
|
end
|
||||||
|
def new
|
||||||
|
end
|
||||||
|
def edit
|
||||||
|
end
|
||||||
|
def destroy
|
||||||
|
end
|
||||||
|
def update
|
||||||
|
end
|
||||||
|
def show
|
||||||
|
end
|
||||||
|
def create
|
||||||
|
end
|
||||||
|
end
|
||||||
2
app/helpers/application_helper.rb
Normal file
2
app/helpers/application_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module ApplicationHelper
|
||||||
|
end
|
||||||
2
app/helpers/articles_helper.rb
Normal file
2
app/helpers/articles_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module ArticlesHelper
|
||||||
|
end
|
||||||
2
app/helpers/blogmap_helper.rb
Normal file
2
app/helpers/blogmap_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module BlogmapHelper
|
||||||
|
end
|
||||||
2
app/helpers/blogs_helper.rb
Normal file
2
app/helpers/blogs_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module BlogsHelper
|
||||||
|
end
|
||||||
2
app/helpers/comments_helper.rb
Normal file
2
app/helpers/comments_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module CommentsHelper
|
||||||
|
end
|
||||||
2
app/helpers/lists_helper.rb
Normal file
2
app/helpers/lists_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module ListsHelper
|
||||||
|
end
|
||||||
2
app/helpers/members_helper.rb
Normal file
2
app/helpers/members_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module MembersHelper
|
||||||
|
end
|
||||||
2
app/helpers/pages_helper.rb
Normal file
2
app/helpers/pages_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module PagesHelper
|
||||||
|
end
|
||||||
2
app/helpers/posts_helper.rb
Normal file
2
app/helpers/posts_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module PostsHelper
|
||||||
|
end
|
||||||
2
app/helpers/profiles_helper.rb
Normal file
2
app/helpers/profiles_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module ProfilesHelper
|
||||||
|
end
|
||||||
2
app/helpers/sitemap_helper.rb
Normal file
2
app/helpers/sitemap_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module SitemapHelper
|
||||||
|
end
|
||||||
2
app/helpers/updates_helper.rb
Normal file
2
app/helpers/updates_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module UpdatesHelper
|
||||||
|
end
|
||||||
2
app/helpers/welcome_helper.rb
Normal file
2
app/helpers/welcome_helper.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
module WelcomeHelper
|
||||||
|
end
|
||||||
6
app/javascript/application.js
Normal file
6
app/javascript/application.js
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
// Configure your import map in config/importmap.rb. Read more: https://github.com/rails/importmap-rails
|
||||||
|
import "@hotwired/turbo-rails"
|
||||||
|
import "controllers"
|
||||||
|
|
||||||
|
import "trix"
|
||||||
|
import "@rails/actiontext"
|
||||||
9
app/javascript/controllers/application.js
Normal file
9
app/javascript/controllers/application.js
Normal file
|
|
@ -0,0 +1,9 @@
|
||||||
|
import { Application } from "@hotwired/stimulus"
|
||||||
|
|
||||||
|
const application = Application.start()
|
||||||
|
|
||||||
|
// Configure Stimulus development experience
|
||||||
|
application.debug = false
|
||||||
|
window.Stimulus = application
|
||||||
|
|
||||||
|
export { application }
|
||||||
7
app/javascript/controllers/hello_controller.js
Normal file
7
app/javascript/controllers/hello_controller.js
Normal file
|
|
@ -0,0 +1,7 @@
|
||||||
|
import { Controller } from "@hotwired/stimulus"
|
||||||
|
|
||||||
|
export default class extends Controller {
|
||||||
|
connect() {
|
||||||
|
this.element.textContent = "Hello World!"
|
||||||
|
}
|
||||||
|
}
|
||||||
4
app/javascript/controllers/index.js
Normal file
4
app/javascript/controllers/index.js
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
// Import and register all your controllers from the importmap via controllers/**/*_controller
|
||||||
|
import { application } from "controllers/application"
|
||||||
|
import { eagerLoadControllersFrom } from "@hotwired/stimulus-loading"
|
||||||
|
eagerLoadControllersFrom("controllers", application)
|
||||||
7
app/jobs/application_job.rb
Normal file
7
app/jobs/application_job.rb
Normal file
|
|
@ -0,0 +1,7 @@
|
||||||
|
class ApplicationJob < ActiveJob::Base
|
||||||
|
# Automatically retry jobs that encountered a deadlock
|
||||||
|
# retry_on ActiveRecord::Deadlocked
|
||||||
|
|
||||||
|
# Most jobs are safe to ignore if the underlying records are no longer available
|
||||||
|
# discard_on ActiveJob::DeserializationError
|
||||||
|
end
|
||||||
4
app/mailers/application_mailer.rb
Normal file
4
app/mailers/application_mailer.rb
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
class ApplicationMailer < ActionMailer::Base
|
||||||
|
default from: "from@example.com"
|
||||||
|
layout "mailer"
|
||||||
|
end
|
||||||
6
app/mailers/passwords_mailer.rb
Normal file
6
app/mailers/passwords_mailer.rb
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
class PasswordsMailer < ApplicationMailer
|
||||||
|
def reset(user)
|
||||||
|
@user = user
|
||||||
|
mail subject: "Reset your password", to: user.email_address
|
||||||
|
end
|
||||||
|
end
|
||||||
3
app/models/application_record.rb
Normal file
3
app/models/application_record.rb
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
class ApplicationRecord < ActiveRecord::Base
|
||||||
|
primary_abstract_class
|
||||||
|
end
|
||||||
15
app/models/article.rb
Normal file
15
app/models/article.rb
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
class Article < ApplicationRecord
|
||||||
|
has_rich_text :text
|
||||||
|
has_rich_text :mood
|
||||||
|
has_rich_text :music
|
||||||
|
has_one_attached :icon_image
|
||||||
|
|
||||||
|
|
||||||
|
validates :title, presence: true,
|
||||||
|
length: { minimum: 5 }
|
||||||
|
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
7
app/models/blog.rb
Normal file
7
app/models/blog.rb
Normal file
|
|
@ -0,0 +1,7 @@
|
||||||
|
class Blog < ApplicationRecord
|
||||||
|
has_rich_text :description
|
||||||
|
has_many :posts, dependent: :destroy
|
||||||
|
belongs_to :member
|
||||||
|
validates :description, presence: true,
|
||||||
|
length: { minimum: 5 }
|
||||||
|
end
|
||||||
2
app/models/blogmap.rb
Normal file
2
app/models/blogmap.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
class Blogmap < ApplicationRecord
|
||||||
|
end
|
||||||
6
app/models/comment.rb
Normal file
6
app/models/comment.rb
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
class Comment < ApplicationRecord
|
||||||
|
belongs_to :post
|
||||||
|
belongs_to :user, optional: true
|
||||||
|
has_rich_text :body
|
||||||
|
end
|
||||||
|
|
||||||
0
app/models/concerns/.keep
Normal file
0
app/models/concerns/.keep
Normal file
4
app/models/current.rb
Normal file
4
app/models/current.rb
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
class Current < ActiveSupport::CurrentAttributes
|
||||||
|
attribute :session
|
||||||
|
delegate :user, to: :session, allow_nil: true
|
||||||
|
end
|
||||||
2
app/models/list.rb
Normal file
2
app/models/list.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
class List < ApplicationRecord
|
||||||
|
end
|
||||||
12
app/models/member.rb
Normal file
12
app/models/member.rb
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
class Member < ApplicationRecord
|
||||||
|
belongs_to :user
|
||||||
|
has_one_attached :avatar_image
|
||||||
|
has_rich_text :bio
|
||||||
|
|
||||||
|
has_many :blogs
|
||||||
|
has_many :posts
|
||||||
|
|
||||||
|
validates :name, presence: true,
|
||||||
|
length: { minimum: 1 }
|
||||||
|
validates :avatar_image, presence:true
|
||||||
|
end
|
||||||
10
app/models/page.rb
Normal file
10
app/models/page.rb
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
class Page < ApplicationRecord
|
||||||
|
belongs_to :member
|
||||||
|
has_rich_text :body
|
||||||
|
validates :title, presence: true,
|
||||||
|
length: { minimum: 3 }
|
||||||
|
validates :body, presence: true,
|
||||||
|
length: { minimum: 5 }
|
||||||
|
|
||||||
|
|
||||||
|
end
|
||||||
13
app/models/post.rb
Normal file
13
app/models/post.rb
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
class Post < ApplicationRecord
|
||||||
|
belongs_to :blog
|
||||||
|
has_one_attached :icon_image
|
||||||
|
has_many :comments
|
||||||
|
has_rich_text :text
|
||||||
|
has_rich_text :mood
|
||||||
|
has_rich_text :music
|
||||||
|
validates :post_title, presence: true,
|
||||||
|
length: { minimum: 5 }
|
||||||
|
validates :text, presence: true,
|
||||||
|
length: { minimum: 5 }
|
||||||
|
validates :icon_image, presence:true
|
||||||
|
end
|
||||||
5
app/models/profile.rb
Normal file
5
app/models/profile.rb
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
class Profile < ApplicationRecord
|
||||||
|
belongs_to :user
|
||||||
|
has_one_attached :avatar_image
|
||||||
|
has_rich_text :bio
|
||||||
|
end
|
||||||
3
app/models/session.rb
Normal file
3
app/models/session.rb
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
class Session < ApplicationRecord
|
||||||
|
belongs_to :user
|
||||||
|
end
|
||||||
5
app/models/sitemap.rb
Normal file
5
app/models/sitemap.rb
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
class Sitemap < ApplicationRecord
|
||||||
|
has_rich_text :text
|
||||||
|
has_many :sitemaps, dependent: :destroy
|
||||||
|
|
||||||
|
end
|
||||||
2
app/models/update.rb
Normal file
2
app/models/update.rb
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
class Update < ApplicationRecord
|
||||||
|
end
|
||||||
10
app/models/user.rb
Normal file
10
app/models/user.rb
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
class User < ApplicationRecord
|
||||||
|
has_secure_password
|
||||||
|
has_many :sessions, dependent: :destroy
|
||||||
|
has_many :members
|
||||||
|
normalizes :email_address, with: ->(e) { e.strip.downcase }
|
||||||
|
end
|
||||||
|
|
||||||
|
def build_default_profile
|
||||||
|
create_profile # Creates an empty profile for the new user
|
||||||
|
end
|
||||||
28
app/views/articles/_form.html.erb
Normal file
28
app/views/articles/_form.html.erb
Normal file
|
|
@ -0,0 +1,28 @@
|
||||||
|
<%= form_with model: @article do |form| %>
|
||||||
|
<% if form.object.errors.any? %>
|
||||||
|
<p class="error"><%= form.object.errors.full_messages.first %></p>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :title %><br>
|
||||||
|
<%= form.text_field :title %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :text %><br>
|
||||||
|
<%= form.rich_textarea :text %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :mood %><br>
|
||||||
|
<%= form.rich_textarea :mood %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :music %><br>
|
||||||
|
<%= form.rich_textarea :music %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.submit %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<% end %>
|
||||||
28
app/views/articles/edit.html.erb
Normal file
28
app/views/articles/edit.html.erb
Normal file
|
|
@ -0,0 +1,28 @@
|
||||||
|
<section class="articleshow">
|
||||||
|
<%= link_to 'Back', articles_path %>
|
||||||
|
<%= form_with(model: @article, local: true) do |form| %>
|
||||||
|
<%= form.label :icon_image, style: "display: block" %>
|
||||||
|
<%= form.file_field :icon_image, accept: "image/*" %>
|
||||||
|
<p>
|
||||||
|
<%= form.label :title %><br>
|
||||||
|
<%= form.text_field :title %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :text %><br>
|
||||||
|
<%= form.rich_textarea :text %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :mood %><br>
|
||||||
|
<%= form.rich_textarea :mood %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :music %><br>
|
||||||
|
<%= form.rich_textarea :music %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.submit %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<% end %>
|
||||||
|
</section>
|
||||||
19
app/views/articles/feed.xml.builder
Normal file
19
app/views/articles/feed.xml.builder
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
xml.instruct! :xml, version: "1.0"
|
||||||
|
xml.rss :version => "2.0" do
|
||||||
|
xml.channel do
|
||||||
|
xml.title "AgnestheAlien"
|
||||||
|
xml.description "Agnes the Alien's blog"
|
||||||
|
xml.link root_url
|
||||||
|
|
||||||
|
Article.order('created_at DESC').each do |article|
|
||||||
|
xml.item do
|
||||||
|
xml.title article.title
|
||||||
|
xml.description article.text.to_plain_text[0,150]
|
||||||
|
xml.pubDate article.created_at.rfc2822
|
||||||
|
xml.link article_url(article)
|
||||||
|
xml.guid article_url(article)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
20
app/views/articles/index.html.erb
Normal file
20
app/views/articles/index.html.erb
Normal file
|
|
@ -0,0 +1,20 @@
|
||||||
|
<% if current_user %>
|
||||||
|
|
||||||
|
<%= link_to 'New Poast', new_article_path(@article) %> <hr><br><% end %><br>
|
||||||
|
|
||||||
|
<% @articles.each do |article| %>
|
||||||
|
<tr>
|
||||||
|
<td><%= article.title %></td>
|
||||||
|
<br>
|
||||||
|
<td><%= link_to 'View', article_path(article) %></td> <br>
|
||||||
|
<% if current_user %>
|
||||||
|
|
|
||||||
|
|
||||||
|
<td><%= link_to 'Edit', edit_article_path(article) %></td>
|
||||||
|
<td><%= link_to 'Destroy', article_path(article), data: {
|
||||||
|
turbo_method: :delete,
|
||||||
|
turbo_confirm: "Are you sure?"
|
||||||
|
} %>
|
||||||
|
<% end %>
|
||||||
|
<br> <br><hr>
|
||||||
|
<% end %>
|
||||||
32
app/views/articles/new.html.erb
Normal file
32
app/views/articles/new.html.erb
Normal file
|
|
@ -0,0 +1,32 @@
|
||||||
|
<% if current_user %>
|
||||||
|
<section class="articleshow">
|
||||||
|
<h1>New Article</h1>
|
||||||
|
<%= link_to 'Back', articles_path %>
|
||||||
|
<%= form_with scope: :article, url: articles_path, local: true do |form| %>
|
||||||
|
<%= form.label :icon_image, style: "display: block" %>
|
||||||
|
<%= form.file_field :icon_image, accept: "image/*" %>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label 'Name' %><br>
|
||||||
|
<%= form.text_field :title %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label 'Bio' %><br>
|
||||||
|
<%= form.rich_text_area :text %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label 'Pronouns' %><br>
|
||||||
|
<%= form.rich_text_area :mood %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label 'Current Mood' %><br>
|
||||||
|
<%= form.rich_text_area :music %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.submit %>
|
||||||
|
</p>
|
||||||
|
<% end %> <!-- Closing the form_with block -->
|
||||||
|
</section>
|
||||||
|
<% end %> <!-- Closing the if current_user block -->
|
||||||
|
|
||||||
23
app/views/articles/rss.rss.builder
Normal file
23
app/views/articles/rss.rss.builder
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
|
||||||
|
xml.instruct! :xml, version: "1.0", encoding: "UTF-8"
|
||||||
|
|
||||||
|
xml.rss(version: "2.0", "xmlns:atom" => "http://www.w3.org/2005/Atom") do
|
||||||
|
xml.channel do
|
||||||
|
xml.title "Agnes the Alien's blog"
|
||||||
|
xml.link articles_url
|
||||||
|
xml.description "Agnes the Alien's blog at galaxy"
|
||||||
|
xml.language "en-us"
|
||||||
|
|
||||||
|
@articles.each do |article|
|
||||||
|
xml.item do
|
||||||
|
xml.title article.title
|
||||||
|
xml.link article_url(article)
|
||||||
|
xml.guid article_url(article)
|
||||||
|
xml.tag!('description', article.text.to_s)
|
||||||
|
xml.tag!('pubDate', article.created_at.rfc2822)
|
||||||
|
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
37
app/views/articles/show.html.erb
Normal file
37
app/views/articles/show.html.erb
Normal file
|
|
@ -0,0 +1,37 @@
|
||||||
|
<section class="article">
|
||||||
|
|
||||||
|
<%= link_to 'Back', articles_path %><br>
|
||||||
|
<% if current_user %>
|
||||||
|
<%= link_to 'Edit', edit_article_path(@article) %>
|
||||||
|
<% end %>
|
||||||
|
<br>
|
||||||
|
|
||||||
|
<%= image_tag @article.icon_image if @article.icon_image.attached? %>
|
||||||
|
<section class="inner"><p>
|
||||||
|
<strong>Name:</strong>
|
||||||
|
<%= @article.title %>
|
||||||
|
</p>
|
||||||
|
<hr>
|
||||||
|
<p>
|
||||||
|
|
||||||
|
<%= @article.text %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<% if @article.music.present? %>
|
||||||
|
<strong>Currently Listening To...:</strong>
|
||||||
|
<%= @article.music %>
|
||||||
|
</p>
|
||||||
|
<% end %>
|
||||||
|
<% if @article.mood.present? %>
|
||||||
|
<p>
|
||||||
|
<strong>Current Mood:</strong>
|
||||||
|
<%= @article.mood %>
|
||||||
|
</p>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<hr>
|
||||||
|
|
||||||
|
|
||||||
|
<%= link_to 'Back', articles_path %>
|
||||||
9
app/views/blogmap/index.html.erb
Normal file
9
app/views/blogmap/index.html.erb
Normal file
|
|
@ -0,0 +1,9 @@
|
||||||
|
<ul>
|
||||||
|
<% @blogs.each do |blog| %>
|
||||||
|
<section class="blogdesc"><li>
|
||||||
|
<%= link_to blog.member.name, member_path(blog.member) %> —
|
||||||
|
<%= link_to blog.title, member_blog_path(blog.member, blog) %> <br><br>Description: <%= truncate(blog.description.to_plain_text, length: 400) %>
|
||||||
|
</section>
|
||||||
|
</li>
|
||||||
|
<br><% end %>
|
||||||
|
</ul>
|
||||||
21
app/views/blogs/edit.html.erb
Normal file
21
app/views/blogs/edit.html.erb
Normal file
|
|
@ -0,0 +1,21 @@
|
||||||
|
<main><h3>Edit your blog!</h3>
|
||||||
|
|
||||||
|
<%= form_with model: [@member, @blog] do |form| %>
|
||||||
|
<%= form.label :title %><br />
|
||||||
|
<%= form.text_area :title %>
|
||||||
|
<p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :links %><br>
|
||||||
|
<%= form.text_area :links %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :description %><br>
|
||||||
|
<%= form.rich_text_area :description %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
|
||||||
|
<%= form.submit %>
|
||||||
|
<% end %>
|
||||||
|
</main>
|
||||||
9
app/views/blogs/index.html.erb
Normal file
9
app/views/blogs/index.html.erb
Normal file
|
|
@ -0,0 +1,9 @@
|
||||||
|
<ul>
|
||||||
|
<% @blogs.each do |blog| %>
|
||||||
|
<section class="blogdesc"><li>
|
||||||
|
<%= link_to blog.member.name, member_path(blog.member) %> —
|
||||||
|
<%= link_to blog.title, member_blog_path(blog.member, blog) %> <br><br>Description: <%= truncate(blog.description.to_plain_text, length: 400) %>
|
||||||
|
</section>
|
||||||
|
</li>
|
||||||
|
<br><% end %>
|
||||||
|
</ul>
|
||||||
9
app/views/blogs/list.html.erb
Normal file
9
app/views/blogs/list.html.erb
Normal file
|
|
@ -0,0 +1,9 @@
|
||||||
|
<ul>
|
||||||
|
<% @blogs.each do |blog| %>
|
||||||
|
<section class="blogdesc"><li>
|
||||||
|
<%= link_to blog.member.name, member_path(blog.member) %> —
|
||||||
|
<%= link_to blog.title, member_blog_path(blog.member, blog) %> <br><br>Description: <%= truncate(blog.description.to_plain_text, length: 400) %>
|
||||||
|
</section>
|
||||||
|
</li>
|
||||||
|
<br><% end %>
|
||||||
|
</ul>
|
||||||
23
app/views/blogs/new.html.erb
Normal file
23
app/views/blogs/new.html.erb
Normal file
|
|
@ -0,0 +1,23 @@
|
||||||
|
<main><h3>Create your blog!</h3>
|
||||||
|
|
||||||
|
<%= form_with model: [@member, @blog] do |form| %>
|
||||||
|
<%= form.label :title %><br />
|
||||||
|
<%= form.text_area :title %>
|
||||||
|
<p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :links %><br>
|
||||||
|
<%= form.text_area :links %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :description %><br>
|
||||||
|
<%= form.rich_text_area :description %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
|
||||||
|
<%= form.submit %>
|
||||||
|
<% end %>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
|
||||||
39
app/views/blogs/show.html.erb
Normal file
39
app/views/blogs/show.html.erb
Normal file
|
|
@ -0,0 +1,39 @@
|
||||||
|
<section class="blog">
|
||||||
|
<% if current_user %>
|
||||||
|
<%= link_to "Edit", edit_member_blog_path %> | <%= link_to "New Post", new_member_blog_post_path(@blog.member, @blog) %> <br><br><br><small><%= button_to "Destroy",
|
||||||
|
member_blog_path(@blog.member, @blog),
|
||||||
|
method: :delete,
|
||||||
|
data: { confirm: "Are you sure?" } %>
|
||||||
|
<% end %>
|
||||||
|
</small>
|
||||||
|
|
||||||
|
<% if @blog.title.present? %>
|
||||||
|
<h2><%= @blog.title %></h2>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<%= @blog.description %>
|
||||||
|
<hr>
|
||||||
|
<% @posts.each do |post| %>
|
||||||
|
<section class="posts"><%= post.created_at %> |
|
||||||
|
Title: <%= post.post_title %><br><br>
|
||||||
|
<% if post.icon_image.attached? %>
|
||||||
|
<%= image_tag post.icon_image, width: "100", height: "100" %>
|
||||||
|
<% else %>
|
||||||
|
<p>No avatar image available.</p>
|
||||||
|
<% end %><br>
|
||||||
|
Content:
|
||||||
|
<%= truncate(post.text.to_plain_text, length: 400) %>
|
||||||
|
<br>
|
||||||
|
<%= link_to 'Show', member_blog_post_path(@member, @blog, post) %>
|
||||||
|
<% if current_user %>
|
||||||
|
|
||||||
|
<%= link_to 'Edit', edit_member_blog_post_path(@member, @blog, post) %>
|
||||||
|
<%= link_to 'Destroy', member_blog_post_path(@member, @blog, post), data: {
|
||||||
|
turbo_method: :delete
|
||||||
|
} %>
|
||||||
|
<% end %>
|
||||||
|
</section>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
</section>
|
||||||
|
|
||||||
21
app/views/comments/_comment.html.erb
Normal file
21
app/views/comments/_comment.html.erb
Normal file
|
|
@ -0,0 +1,21 @@
|
||||||
|
<hr><p>
|
||||||
|
<strong>Name:</strong>
|
||||||
|
<%= comment.commenter %>
|
||||||
|
</p>
|
||||||
|
<% if comment.website.present? %>
|
||||||
|
<p> <strong>Website:</strong>
|
||||||
|
<%= comment.website %>
|
||||||
|
</p>
|
||||||
|
<% end %>
|
||||||
|
<p>
|
||||||
|
<strong>Comment:</strong>
|
||||||
|
<%= comment.body %> </p>
|
||||||
|
|
||||||
|
<% if current_user %>
|
||||||
|
<%= link_to "Destroy Comment",
|
||||||
|
member_blog_post_comment_path(@member, @blog, @post, comment),
|
||||||
|
data: { turbo_method: :delete, turbo_confirm: "Are you sure?" } %>
|
||||||
|
|
||||||
|
|
||||||
|
<% end %>
|
||||||
|
<hr>
|
||||||
3
app/views/layouts/action_text/contents/_content.html.erb
Normal file
3
app/views/layouts/action_text/contents/_content.html.erb
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
<div class="trix-content">
|
||||||
|
<%= yield -%>
|
||||||
|
</div>
|
||||||
60
app/views/layouts/application.html.erb
Normal file
60
app/views/layouts/application.html.erb
Normal file
|
|
@ -0,0 +1,60 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title><%= content_for(:title) || "KissingComputer" %></title>
|
||||||
|
<meta name="viewport" content="width=device-width,initial-scale=1">
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
|
<meta name="application-name" content="KissingComputer">
|
||||||
|
<meta name="mobile-web-app-capable" content="yes">
|
||||||
|
<%= csrf_meta_tags %>
|
||||||
|
<%= csp_meta_tag %>
|
||||||
|
|
||||||
|
|
||||||
|
<%= yield :head %>
|
||||||
|
|
||||||
|
<%# Enable PWA manifest for installable apps (make sure to enable in config/routes.rb too!) %>
|
||||||
|
<%#= tag.link rel: "manifest", href: pwa_manifest_path(format: :json) %>
|
||||||
|
|
||||||
|
<link rel="icon" href="/icon.png" type="image/png">
|
||||||
|
<link rel="icon" href="/icon.svg" type="image/svg+xml">
|
||||||
|
<link rel="apple-touch-icon" href="/icon.png">
|
||||||
|
<link rel="alternate" type="application/rss+xml" title="RSS" href="/feed.xml" /><link rel="preconnect" href="https://fonts.googleapis.com">
|
||||||
|
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
|
||||||
|
<link href="https://fonts.googleapis.com/css2?family=Winky+Sans:ital,wght@0,300..900;1,300..900&display=swap" rel="stylesheet">
|
||||||
|
|
||||||
|
<%# Includes all stylesheet files in app/assets/stylesheets %>
|
||||||
|
<%= stylesheet_link_tag :app, "data-turbo-track": "reload" %>
|
||||||
|
<%= javascript_importmap_tags %>
|
||||||
|
<%= javascript_include_tag "application", "data-turbolinks-track" => true %>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<main><br>
|
||||||
|
<div class="headerall"><h3 class="float"><img src="/kissingcomputer.png" width="80px" alt="An emoji of a computer with kiss marks."> K I S S I N G C O M P U T E R</h3><div class="header"> <a href="/">Home</a> | <a href="/members/1">About Me</a> | <a href="/blogmap">Pages</a> | <a href="/sitemaps">Sitemap</a> | <a href="https://lesbian.alien.dentist/@feed.atom"><img src="https://file.garden/aJzQmzrHVB4BLKwu/46.png" alt="pink rss feed logo"></a></div></div>
|
||||||
|
<br><br>
|
||||||
|
<section class="article">
|
||||||
|
<section class="auth">
|
||||||
|
<% if current_user %>
|
||||||
|
<%= link_to 'Sign Out', sign_out_path, data: { turbo_method: :delete } %>
|
||||||
|
|
||||||
|
<% else %>
|
||||||
|
<%= link_to "Sign In", new_session_path %>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
</section></section>
|
||||||
|
|
||||||
|
<br><br>
|
||||||
|
|
||||||
|
<br>
|
||||||
|
<%= yield %>
|
||||||
|
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<section class="footer">
|
||||||
|
v. 1.0 | <a href="https://treasurechest.alien.town/agnes/kissingcomputer">source code</a><br>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
|
||||||
13
app/views/layouts/mailer.html.erb
Normal file
13
app/views/layouts/mailer.html.erb
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
|
||||||
|
<style>
|
||||||
|
/* Email styles need to be inline */
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<%= yield %>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
1
app/views/layouts/mailer.text.erb
Normal file
1
app/views/layouts/mailer.text.erb
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
<%= yield %>
|
||||||
56
app/views/members/edit.html.erb
Normal file
56
app/views/members/edit.html.erb
Normal file
|
|
@ -0,0 +1,56 @@
|
||||||
|
EDIT MEMBER!
|
||||||
|
|
||||||
|
<div class="member-new">
|
||||||
|
<%= form_with model: (@member) do |form| %>
|
||||||
|
|
||||||
|
<%= form.label :avatar_image %><br />
|
||||||
|
<%= form.file_field :avatar_image %>
|
||||||
|
<p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :name %><br>
|
||||||
|
<%= form.text_area :name %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :age %><br>
|
||||||
|
<%= form.text_area :age %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :birthday %><br>
|
||||||
|
<%= form.text_area :birthday %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :bio %><br>
|
||||||
|
<%= form.rich_text_area :bio %>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :links %><br>
|
||||||
|
<%= form.rich_text_area :links %>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<%= form.label :pronouns %><br>
|
||||||
|
<%= form.text_area :pronouns %><br>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<%= form.label :system_role %><br>
|
||||||
|
<%= form.text_area :system_role %><br>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<h3> Interests</h3>
|
||||||
|
|
||||||
|
<p><%= form.label :tv_interests %>
|
||||||
|
<%= form.text_area :tv_interests %>
|
||||||
|
</p><br>
|
||||||
|
<p><%= form.label :music_interests %>
|
||||||
|
<%= form.text_area :music_interests %>
|
||||||
|
</p><br>
|
||||||
|
<%= form.label :other_interests %>
|
||||||
|
<%= form.text_area :other_interests %>
|
||||||
|
</p><br>
|
||||||
|
<%= form.submit %>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
42
app/views/members/index.html.erb
Normal file
42
app/views/members/index.html.erb
Normal file
|
|
@ -0,0 +1,42 @@
|
||||||
|
<section class="members"><br>
|
||||||
|
<% if current_user %> <section class="createmember"><a href="/members/new">Create a new profile?</a></section> <br><br><% end %>
|
||||||
|
<section class="memberlist">
|
||||||
|
<table class="member">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="member">Name</th>
|
||||||
|
<th class="member">Avatar</th>
|
||||||
|
<th class="member"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<% @members.each do |member| %>
|
||||||
|
<section class="member-specific">
|
||||||
|
<tr>
|
||||||
|
<td class="member"><%=link_to member.name, member_path(member) %></td>
|
||||||
|
<td class="member">
|
||||||
|
<% if member.avatar_image.attached? %>
|
||||||
|
<%= image_tag member.avatar_image, alt: "#{member.name}'s avatar", width: "100", height: "100" %>
|
||||||
|
<% else %>
|
||||||
|
<p>No avatar image available.</p>
|
||||||
|
<% end %>
|
||||||
|
</td>
|
||||||
|
<td class="member"><p><%= truncate(member.bio.to_plain_text, length: 100) %></p>
|
||||||
|
<% if current_user %>
|
||||||
|
<td ><%= link_to 'Edit', edit_member_path(member) %></td>
|
||||||
|
<td><%= link_to 'Destroy', member_path(member), data: {
|
||||||
|
turbo_method: :delete
|
||||||
|
} %>
|
||||||
|
<% end %>
|
||||||
|
</td>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</tr>
|
||||||
|
</section>
|
||||||
|
<% end %>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue