Merge remote-tracking branch 'origin/main' into ap07

This commit is contained in:
Kitaiti Makoto 2022-05-06 12:38:47 +09:00
commit e8dc0942e5
12 changed files with 81 additions and 59 deletions

View File

@ -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

View File

@ -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"

View File

@ -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;

View File

@ -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"] }

View File

@ -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;

View File

@ -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"

View File

@ -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

View File

@ -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),
} }
} }

View File

@ -28,25 +28,26 @@ 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")]

6
src/routes/mod.rs Executable file → Normal file
View 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> {

View File

@ -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>
&mdash;
<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)

View File

@ -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() {