Merge remote-tracking branch 'origin/main' into ap07
This commit is contained in:
commit
e8dc0942e5
@ -11,7 +11,7 @@ executors:
|
|||||||
default: false
|
default: false
|
||||||
docker:
|
docker:
|
||||||
- image: plumeorg/plume-buildenv:v0.4.0
|
- image: plumeorg/plume-buildenv:v0.4.0
|
||||||
- image: <<#parameters.postgres>>circleci/postgres:9.6-alpine<</parameters.postgres>><<^parameters.postgres>>alpine:latest<</parameters.postgres>>
|
- image: <<#parameters.postgres>>cimg/postgres:14.2<</parameters.postgres>><<^parameters.postgres>>alpine:latest<</parameters.postgres>>
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_USER: postgres
|
POSTGRES_USER: postgres
|
||||||
POSTGRES_DB: plume
|
POSTGRES_DB: plume
|
||||||
|
@ -20,7 +20,7 @@ rocket_contrib = { version = "0.4.5", features = ["json"] }
|
|||||||
rocket_i18n = "0.4.1"
|
rocket_i18n = "0.4.1"
|
||||||
scheduled-thread-pool = "0.2.2"
|
scheduled-thread-pool = "0.2.2"
|
||||||
serde = "1.0"
|
serde = "1.0"
|
||||||
serde_json = "1.0.80"
|
serde_json = "1.0.81"
|
||||||
shrinkwraprs = "0.3.0"
|
shrinkwraprs = "0.3.0"
|
||||||
validator = { version = "0.15", features = ["derive"] }
|
validator = { version = "0.15", features = ["derive"] }
|
||||||
webfinger = "0.4.1"
|
webfinger = "0.4.1"
|
||||||
|
@ -507,6 +507,7 @@ figure {
|
|||||||
margin: auto $horizontal-margin 2em;
|
margin: auto $horizontal-margin 2em;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
a {
|
a {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
@ -9,12 +9,12 @@ array_tool = "1.0"
|
|||||||
base64 = "0.13"
|
base64 = "0.13"
|
||||||
heck = "0.4.0"
|
heck = "0.4.0"
|
||||||
hex = "0.4"
|
hex = "0.4"
|
||||||
openssl = "0.10.22"
|
openssl = "0.10.40"
|
||||||
rocket = "0.4.6"
|
rocket = "0.4.6"
|
||||||
reqwest = { version = "0.11.10", features = ["blocking", "json", "socks"] }
|
reqwest = { version = "0.11.10", features = ["blocking", "json", "socks"] }
|
||||||
serde = "1.0"
|
serde = "1.0"
|
||||||
serde_derive = "1.0"
|
serde_derive = "1.0"
|
||||||
serde_json = "1.0.80"
|
serde_json = "1.0.81"
|
||||||
shrinkwraprs = "0.3.0"
|
shrinkwraprs = "0.3.0"
|
||||||
syntect = "4.5.0"
|
syntect = "4.5.0"
|
||||||
regex-syntax = { version = "0.6.17", default-features = false, features = ["unicode-perl"] }
|
regex-syntax = { version = "0.6.17", default-features = false, features = ["unicode-perl"] }
|
||||||
|
@ -157,6 +157,9 @@ where
|
|||||||
.build()
|
.build()
|
||||||
.expect("Error while initializing tokio runtime for federation");
|
.expect("Error while initializing tokio runtime for federation");
|
||||||
rt.block_on(async {
|
rt.block_on(async {
|
||||||
|
// TODO: should be determined dependent on database connections because
|
||||||
|
// after broadcasting, target instance sends request to this instance,
|
||||||
|
// and Plume accesses database at that time.
|
||||||
let capacity = 6;
|
let capacity = 6;
|
||||||
let (tx, rx) = flume::bounded::<RequestBuilder>(capacity);
|
let (tx, rx) = flume::bounded::<RequestBuilder>(capacity);
|
||||||
let mut handles = Vec::with_capacity(capacity);
|
let mut handles = Vec::with_capacity(capacity);
|
||||||
@ -206,7 +209,7 @@ where
|
|||||||
.expect("activity_pub::broadcast: request signature error"),
|
.expect("activity_pub::broadcast: request signature error"),
|
||||||
);
|
);
|
||||||
let request_builder = client.post(&inbox).headers(headers.clone()).body(body);
|
let request_builder = client.post(&inbox).headers(headers.clone()).body(body);
|
||||||
tx.send_async(request_builder).await.unwrap();
|
let _ = tx.send_async(request_builder).await;
|
||||||
}
|
}
|
||||||
drop(tx);
|
drop(tx);
|
||||||
join_all(handles).await;
|
join_all(handles).await;
|
||||||
|
@ -12,14 +12,14 @@ itertools = "0.10.3"
|
|||||||
lazy_static = "1.0"
|
lazy_static = "1.0"
|
||||||
ldap3 = "0.10.4"
|
ldap3 = "0.10.4"
|
||||||
migrations_internals= "1.4.0"
|
migrations_internals= "1.4.0"
|
||||||
openssl = "0.10.22"
|
openssl = "0.10.40"
|
||||||
rocket = "0.4.6"
|
rocket = "0.4.6"
|
||||||
rocket_i18n = "0.4.1"
|
rocket_i18n = "0.4.1"
|
||||||
reqwest = "0.11.10"
|
reqwest = "0.11.10"
|
||||||
scheduled-thread-pool = "0.2.2"
|
scheduled-thread-pool = "0.2.2"
|
||||||
serde = "1.0"
|
serde = "1.0"
|
||||||
serde_derive = "1.0"
|
serde_derive = "1.0"
|
||||||
serde_json = "1.0.80"
|
serde_json = "1.0.81"
|
||||||
tantivy = "0.13.3"
|
tantivy = "0.13.3"
|
||||||
url = "2.1"
|
url = "2.1"
|
||||||
walkdir = "2.2"
|
walkdir = "2.2"
|
||||||
|
@ -6,6 +6,7 @@ use crate::{
|
|||||||
Connection, Error, Result,
|
Connection, Error, Result,
|
||||||
};
|
};
|
||||||
use diesel::{self, BoolExpressionMethods, ExpressionMethods, QueryDsl, RunQueryDsl};
|
use diesel::{self, BoolExpressionMethods, ExpressionMethods, QueryDsl, RunQueryDsl};
|
||||||
|
use std::cmp::Ordering;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
|
|
||||||
pub(crate) mod query;
|
pub(crate) mod query;
|
||||||
@ -85,6 +86,16 @@ impl Timeline {
|
|||||||
.or(timeline_definition::user_id.is_null()),
|
.or(timeline_definition::user_id.is_null()),
|
||||||
)
|
)
|
||||||
.load::<Self>(conn)
|
.load::<Self>(conn)
|
||||||
|
.map(|mut timelines| {
|
||||||
|
timelines.sort_by(|t1, t2| {
|
||||||
|
if t1.user_id.is_some() && t2.user_id.is_none() {
|
||||||
|
Ordering::Less
|
||||||
|
} else {
|
||||||
|
Ordering::Equal
|
||||||
|
}
|
||||||
|
});
|
||||||
|
timelines
|
||||||
|
})
|
||||||
.map_err(Error::from)
|
.map_err(Error::from)
|
||||||
} else {
|
} else {
|
||||||
timeline_definition::table
|
timeline_definition::table
|
||||||
|
@ -21,8 +21,9 @@ impl<'r> Responder<'r> for ErrorPage {
|
|||||||
warn!("{:?}", self.0);
|
warn!("{:?}", self.0);
|
||||||
|
|
||||||
match self.0 {
|
match self.0 {
|
||||||
Error::NotFound => Err(Status::NotFound),
|
Error::NotFound | Error::Unauthorized | Error::Db(diesel::result::Error::NotFound) => {
|
||||||
Error::Unauthorized => Err(Status::NotFound),
|
Err(Status::NotFound)
|
||||||
|
}
|
||||||
_ => Err(Status::InternalServerError),
|
_ => Err(Status::InternalServerError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -28,26 +28,27 @@ use plume_models::{
|
|||||||
|
|
||||||
#[get("/")]
|
#[get("/")]
|
||||||
pub fn index(conn: DbConn, rockets: PlumeRocket) -> Result<Ructe, ErrorPage> {
|
pub fn index(conn: DbConn, rockets: PlumeRocket) -> Result<Ructe, ErrorPage> {
|
||||||
let inst = Instance::get_local()?;
|
let all_tl = Timeline::list_all_for_user(&conn, rockets.user.clone().map(|u| u.id))?;
|
||||||
let timelines = Timeline::list_all_for_user(&conn, rockets.user.clone().map(|u| u.id))?
|
if all_tl.is_empty() {
|
||||||
.into_iter()
|
Err(Error::NotFound.into())
|
||||||
.filter_map(|t| {
|
|
||||||
if let Ok(latest) = t.get_latest(&conn, 12) {
|
|
||||||
Some((t, latest))
|
|
||||||
} else {
|
} else {
|
||||||
None
|
let inst = Instance::get_local()?;
|
||||||
}
|
let page = Page::default();
|
||||||
})
|
let tl = &all_tl[0];
|
||||||
.collect();
|
let posts = tl.get_page(&conn, page.limits())?;
|
||||||
|
let total_posts = tl.count_posts(&conn)?;
|
||||||
Ok(render!(instance::index(
|
Ok(render!(instance::index(
|
||||||
&(&conn, &rockets).to_context(),
|
&(&conn, &rockets).to_context(),
|
||||||
inst,
|
inst,
|
||||||
User::count_local(&conn)?,
|
User::count_local(&conn)?,
|
||||||
Post::count_local(&conn)?,
|
Post::count_local(&conn)?,
|
||||||
timelines
|
tl.id,
|
||||||
|
posts,
|
||||||
|
all_tl,
|
||||||
|
Page::total(total_posts as i32)
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[get("/admin")]
|
#[get("/admin")]
|
||||||
pub fn admin(_admin: Admin, conn: DbConn, rockets: PlumeRocket) -> Result<Ructe, ErrorPage> {
|
pub fn admin(_admin: Admin, conn: DbConn, rockets: PlumeRocket) -> Result<Ructe, ErrorPage> {
|
||||||
|
6
src/routes/mod.rs
Executable file → Normal file
6
src/routes/mod.rs
Executable file → Normal file
@ -59,6 +59,12 @@ impl From<Flash<Redirect>> for RespondOrRedirect {
|
|||||||
#[derive(Shrinkwrap, Copy, Clone, UriDisplayQuery)]
|
#[derive(Shrinkwrap, Copy, Clone, UriDisplayQuery)]
|
||||||
pub struct Page(i32);
|
pub struct Page(i32);
|
||||||
|
|
||||||
|
impl From<i32> for Page {
|
||||||
|
fn from(page: i32) -> Self {
|
||||||
|
Self(page)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'v> FromFormValue<'v> for Page {
|
impl<'v> FromFormValue<'v> for Page {
|
||||||
type Error = &'v RawStr;
|
type Error = &'v RawStr;
|
||||||
fn from_form_value(form_value: &'v RawStr) -> Result<Page, &'v RawStr> {
|
fn from_form_value(form_value: &'v RawStr) -> Result<Page, &'v RawStr> {
|
||||||
|
@ -4,37 +4,37 @@
|
|||||||
@use crate::templates::{base, partials::*};
|
@use crate::templates::{base, partials::*};
|
||||||
@use crate::template_utils::*;
|
@use crate::template_utils::*;
|
||||||
@use crate::routes::*;
|
@use crate::routes::*;
|
||||||
|
@use rocket::uri;
|
||||||
|
|
||||||
@(ctx: BaseContext, instance: Instance, n_users: i64, n_articles: i64, all_tl: Vec<(Timeline, Vec<Post>)>)
|
@(ctx: BaseContext, instance: Instance, n_users: i64, n_articles: i64, tl_id: i32, articles: Vec<Post>, all_tl: Vec<Timeline>, n_pages: i32)
|
||||||
|
|
||||||
@:base(ctx, instance.name.clone(), {}, {}, {
|
@:base(ctx, instance.name.clone(), {}, {}, {
|
||||||
<h1>@i18n!(ctx.1, "Welcome to {}"; instance.name.as_str())</h1>
|
<section class="flex wrap" dir="auto">
|
||||||
|
<h1 class="grow">@i18n!(ctx.1, "Welcome to {}"; instance.name.as_str())</h1>
|
||||||
|
</section>
|
||||||
|
|
||||||
@tabs(&vec![(format!("{}", uri!(instance::index)), i18n!(ctx.1, "Latest articles"), true)]
|
@tabs(&all_tl
|
||||||
.into_iter().chain(all_tl.clone()
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|(tl, _)| {
|
.map(|t| {
|
||||||
let url = format!("{}", uri!(timelines::details: id = tl.id, page = _));
|
let url = format!("{}", uri!(timelines::details: id = t.id, page = _));
|
||||||
(url, i18n_timeline_name(ctx.1, &tl.name), false)
|
(url, i18n_timeline_name(ctx.1, &t.name), t.id == tl_id)
|
||||||
})
|
})
|
||||||
).collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
)
|
)
|
||||||
|
|
||||||
@for (tl, articles) in all_tl {
|
|
||||||
@if !articles.is_empty() {
|
@if !articles.is_empty() {
|
||||||
<div class="h-feed">
|
|
||||||
<h2 dir="auto">
|
|
||||||
<span class="p-name">@i18n_timeline_name(ctx.1, &tl.name)</span>
|
|
||||||
—
|
|
||||||
<a href="@uri!(timelines::details: id = tl.id, page = _)">@i18n!(ctx.1, "View all")</a>
|
|
||||||
</h2>
|
|
||||||
<div class="cards">
|
<div class="cards">
|
||||||
@for article in articles {
|
@for article in articles {
|
||||||
@:post_card(ctx, article)
|
@:post_card(ctx, article)
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
} else {
|
||||||
|
<p class="center">@i18n!(ctx.1, "Nothing to see here yet.")</p>
|
||||||
}
|
}
|
||||||
|
@if n_pages > 1 {
|
||||||
|
<div class="pagination" dir="auto">
|
||||||
|
<a href="@uri!(timelines::details: id = tl_id, page = Some(2.into()))">@i18n!(ctx.1, "Next page")</a>
|
||||||
|
</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
@:instance_description(ctx, instance, n_users, n_articles)
|
@:instance_description(ctx, instance, n_users, n_articles)
|
||||||
|
@ -12,14 +12,13 @@
|
|||||||
<h1 class="grow">@i18n_timeline_name(ctx.1, &tl.name)</h1>
|
<h1 class="grow">@i18n_timeline_name(ctx.1, &tl.name)</h1>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
@tabs(&vec![(format!("{}", uri!(instance::index)), i18n!(ctx.1, "Latest articles"), false)]
|
@tabs(&all_tl
|
||||||
.into_iter().chain(all_tl
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|t| {
|
.map(|t| {
|
||||||
let url = format!("{}", uri!(timelines::details: id = t.id, page = _));
|
let url = format!("{}", uri!(timelines::details: id = t.id, page = _));
|
||||||
(url, i18n_timeline_name(ctx.1, &t.name), t.id == tl.id)
|
(url, i18n_timeline_name(ctx.1, &t.name), t.id == tl.id)
|
||||||
})
|
})
|
||||||
).collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
)
|
)
|
||||||
|
|
||||||
@if !articles.is_empty() {
|
@if !articles.is_empty() {
|
||||||
|
Loading…
Reference in New Issue
Block a user