remove compile time postgres dependency, add docker build

This commit is contained in:
neri 2020-07-12 02:26:11 +02:00
parent deb99942d3
commit e110378ba6
8 changed files with 112 additions and 74 deletions

11
Cargo.lock generated
View file

@ -688,6 +688,7 @@ dependencies = [
"futures",
"log",
"mime",
"openssl-sys",
"rand",
"sqlx",
]
@ -1454,6 +1455,15 @@ version = "0.1.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "77af24da69f9d9341038eba93a073b1fdaaa1b788221b00a69bce9e762cb32de"
[[package]]
name = "openssl-src"
version = "111.10.1+1.1.1g"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "375f12316ddf0762f7cf1e2890a0a857954b96851b47b5af7fc06940c9e12f83"
dependencies = [
"cc",
]
[[package]]
name = "openssl-sys"
version = "0.9.58"
@ -1463,6 +1473,7 @@ dependencies = [
"autocfg",
"cc",
"libc",
"openssl-src",
"pkg-config",
"vcpkg",
]

View file

@ -19,3 +19,7 @@ futures = "0.3.5"
mime = "0.3.16"
rand = "0.7.3"
chrono = "0.4.13"
openssl-sys = "*"
[features]
vendored = ["openssl-sys/vendored"]

View file

@ -1,30 +1,31 @@
FROM postgres as builder
FROM ekidd/rust-musl-builder as build
ENV POSTGRES_USER "datatrash"
ENV POSTGRES_PASSWORD "secure"
USER rust
WORKDIR /home/rust/src/
RUN USER=rust cargo new datatrash
RUN apt-get update
RUN apt-get install --yes curl build-essential
RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs -o rustup-install.sh
RUN sh rustup-install.sh -y
ENV PATH="/root/.cargo/bin:${PATH}"
RUN rustup target add x86_64-unknown-linux-musl
WORKDIR /home/rust/src/datatrash
COPY --chown=rust Cargo.toml Cargo.lock ./
RUN cargo build --release --features vendored
ENV USER rust
WORKDIR /
RUN cargo new --bin datatrash
WORKDIR /datatrash
COPY Cargo.lock Cargo.lock
COPY Cargo.toml Cargo.toml
RUN cargo build --release --target=x86_64-unknown-linux-musl --features vendored
RUN rm src/*.rs
COPY --chown=rust src ./src
COPY --chown=rust static ./static
COPY --chown=rust template ./template
COPY --chown=rust init-db.sql ./init-db.sql
RUN touch src/main.rs
RUN cargo install --path . --features vendored
COPY src src
ENV DATABASE_URL "postgresql://datatrash:secure@localhost"
RUN rm target/release/deps/datatrash*
RUN cargo build --release --target=x86_64-unknown-linux-musl --features vendored
RUN strip target/release/horrible
FROM alpine
FROM SCRATCH
COPY --from=builder /datatrash/target/release/datatrash /datatrash
ENTRYPOINT ["/datatrash"]
ENV DATABASE_URL "postresql://localhost"
ENV SERVER_URL "http://localhost:8000"
ENV FILES_DIR "./files"
ENV UPLOAD_MAX_BYTES "8388608"
ENV BIND_ADDRESS "0.0.0.0:8000"
ENV RUST_BACKTRACE "1"
COPY --from=build /home/rust/.cargo/bin/datatrash .
COPY static ./static
RUN mkdir ./files
EXPOSE 8000
ENTRYPOINT ["./datatrash"]

View file

@ -6,16 +6,16 @@ A file and text uploading service with configurable time limit
## compiling
Compiling is a little strange.
The SQL-statements are checked for correctness at compile-time, unfortunately this means that the
database needs to be running at compile-time too.
```sh
docker build -t datatrash .
docker cp datatrash:/home/rust/.cargo/bin/datatrash datatrash
```
To get set up:
or, to just run it in docker
- Start a postgresql somewhere
- Set its connection url in the `.env` file
- Run the `init-db.sql` script in the database (`cat init-db.sql | psql`)
- Build the project `cargo build --release`
```sh
docker-compose up -d --build
```
## running & config
@ -27,6 +27,4 @@ To get set up:
| UPLOAD_MAX_BYTES | 8388608 (8MiB) |
| BIND_ADDRESS | 0.0.0.0:8000 |
Other things are not configurable yet.
- The maximum filename length is 255
The maximum filename length is 255

15
docker-compose.yml Normal file
View file

@ -0,0 +1,15 @@
version: "3.3"
services:
datatrash:
build: .
environment:
DATABASE_URL: 'postgresql://admin:secure@postgres'
ports:
- '8000:8000'
postgres:
image: postgres
environment:
POSTGRES_USER: admin
POSTGRES_PASSWORD: secure

View file

@ -1,26 +1,26 @@
use async_std::{fs, path::PathBuf, sync::Receiver, task};
use chrono::{prelude::*, Duration};
use futures::future::FutureExt;
use sqlx::postgres::PgPool;
use sqlx::{postgres::PgPool, Cursor, Row};
pub(crate) async fn delete_old_files(receiver: Receiver<()>, db: PgPool, files_dir: PathBuf) {
loop {
wait_for_file_expiry(&receiver, &db).await;
let now = Local::now().naive_local();
let expired_files =
sqlx::query!("SELECT file_id FROM files WHERE files.valid_till < $1", now)
.fetch_all(&db)
.await
.unwrap();
for expired_file in expired_files {
let mut cursor = sqlx::query("SELECT file_id FROM files WHERE files.valid_till < $1")
.bind(Local::now().naive_local())
.fetch(&db);
while let Some(row) = cursor.next().await.expect("could not load expired files") {
let file_id: String = row.get("file_id");
let mut path = files_dir.clone();
path.push(&expired_file.file_id);
path.push(&file_id);
if path.exists().await {
log::info!("delete file {}", expired_file.file_id);
log::info!("delete file {}", file_id);
fs::remove_file(&path).await.expect("could not delete file");
}
}
sqlx::query!("DELETE FROM files WHERE valid_till < $1", now)
sqlx::query("DELETE FROM files WHERE valid_till < $1")
.bind(Local::now().naive_local())
.execute(&db)
.await
.expect("could not delete expired files from database");
@ -28,12 +28,15 @@ pub(crate) async fn delete_old_files(receiver: Receiver<()>, db: PgPool, files_d
}
async fn wait_for_file_expiry(receiver: &Receiver<()>, db: &PgPool) {
let row = sqlx::query!("SELECT MIN(valid_till) as min from files")
.fetch_one(db)
let mut cursor = sqlx::query("SELECT MIN(valid_till) as min from files").fetch(db);
let row = cursor
.next()
.await
.expect("could not fetch expiring file from database");
let next_timeout = match row.min {
Some(min) => min.signed_duration_since(Local::now().naive_local()),
.expect("could not fetch expiring files from database")
.expect("postgres min did not return any row");
let valid_till: Option<NaiveDateTime> = row.get("min");
let next_timeout = match valid_till {
Some(valid_till) => valid_till.signed_duration_since(Local::now().naive_local()),
None => Duration::days(1),
};
let positive_timeout = next_timeout

View file

@ -13,7 +13,10 @@ use actix_web::{
};
use async_std::{fs, path::PathBuf, sync::Sender, task};
use file_kind::FileKind;
use sqlx::postgres::PgPool;
use sqlx::{
postgres::{PgPool, PgRow},
Cursor, Row,
};
use std::env;
const INDEX_HTML: &str = include_str!("../template/index.html");
@ -49,16 +52,14 @@ async fn upload(
}
};
sqlx::query!(
"INSERT INTO Files (file_id, file_name, valid_till, kind) VALUES ($1, $2, $3, $4)",
file_id,
original_name.unwrap_or_else(|| file_id.clone()),
valid_till.naive_local(),
kind.to_string()
)
.execute(db.as_ref())
.await
.map_err(|_| error::ErrorInternalServerError("could not insert file into database"))?;
sqlx::query("INSERT INTO Files (file_id, file_name, valid_till, kind) VALUES ($1, $2, $3, $4)")
.bind(&file_id)
.bind(original_name.unwrap_or_else(|| file_id.clone()))
.bind(valid_till.naive_local())
.bind(kind.to_string())
.execute(db.as_ref())
.await
.map_err(|_| error::ErrorInternalServerError("could not insert file into database"))?;
log::info!(
"create new file {} (valid_till: {}, kind: {})",
@ -89,17 +90,21 @@ async fn download(
db: web::Data<PgPool>,
config: web::Data<Config>,
) -> Result<HttpResponse, Error> {
let row = sqlx::query!(
"SELECT file_id, file_name, kind from files WHERE file_id = $1",
*id
)
.fetch_one(db.as_ref())
.await
.map_err(|_| error::ErrorNotFound("could not find file"))?;
let mut cursor = sqlx::query("SELECT file_id, file_name, kind from files WHERE file_id = $1")
.bind(id.as_ref())
.fetch(db.as_ref());
let row: PgRow = cursor
.next()
.await
.map_err(|_| error::ErrorInternalServerError("could not run select statement"))?
.ok_or_else(|| error::ErrorNotFound("could not find file"))?;
let file_id: String = row.get("file_id");
let file_name: String = row.get("file_name");
let kind: String = row.get("kind");
let mut path = config.files_dir.clone();
path.push(&row.file_id);
path.push(&file_id);
if row.kind == FileKind::TEXT.to_string() {
if kind == FileKind::TEXT.to_string() {
let content = fs::read_to_string(path).await?;
let view_html = VIEW_HTML.replace("{text}", &content);
let response = HttpResponse::Ok().content_type("text/html").body(view_html);
@ -107,7 +112,7 @@ async fn download(
} else {
let file = NamedFile::open(path)?.set_content_disposition(ContentDisposition {
disposition: DispositionType::Attachment,
parameters: vec![DispositionParam::Filename(row.file_name)],
parameters: vec![DispositionParam::Filename(file_name)],
});
file.into_response(&req)
}
@ -120,7 +125,7 @@ async fn setup_db() -> PgPool {
.await
.expect("could not create db pool");
sqlx::query_file!("./init-db.sql")
sqlx::query(include_str!("../init-db.sql"))
.execute(&pool)
.await
.expect("could not create table Files");

View file

@ -20,6 +20,7 @@
<br />
Gültig für
<select name="validity_secs">
<option value="10">10 sekunden</option>
<option value="1800">30 minuten</option>
<option value="3600">60 minuten</option>
<option value="43200">12 stunden</option>