2024-09-16 15:18:16 +00:00
|
|
|
use actix_web::{web, Error, HttpResponse, Result};
|
2022-11-28 14:29:33 +00:00
|
|
|
use lemmy_api_common::context::LemmyContext;
|
2023-04-17 19:19:51 +00:00
|
|
|
use lemmy_db_schema::RegistrationMode;
|
2022-05-03 17:44:13 +00:00
|
|
|
use lemmy_db_views::structs::SiteView;
|
2023-07-19 10:09:04 +00:00
|
|
|
use lemmy_utils::{
|
|
|
|
cache_header::{cache_1hour, cache_3days},
|
2024-09-16 15:18:16 +00:00
|
|
|
error::LemmyResult,
|
2024-04-02 15:19:51 +00:00
|
|
|
VERSION,
|
2023-07-19 10:09:04 +00:00
|
|
|
};
|
2020-05-16 14:04:08 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2024-05-31 20:38:46 +00:00
|
|
|
use std::collections::HashMap;
|
2020-05-16 14:04:08 +00:00
|
|
|
use url::Url;
|
2019-11-15 02:08:25 +00:00
|
|
|
|
2024-06-03 21:30:16 +00:00
|
|
|
/// A description of the nodeinfo endpoint is here:
|
|
|
|
/// https://github.com/jhass/nodeinfo/blob/main/PROTOCOL.md
|
2019-12-31 12:55:33 +00:00
|
|
|
pub fn config(cfg: &mut web::ServiceConfig) {
|
|
|
|
cfg
|
2023-07-19 10:09:04 +00:00
|
|
|
.route(
|
2024-06-03 21:30:16 +00:00
|
|
|
"/nodeinfo/2.1",
|
2023-07-19 10:09:04 +00:00
|
|
|
web::get().to(node_info).wrap(cache_1hour()),
|
|
|
|
)
|
2024-06-03 21:30:16 +00:00
|
|
|
.service(web::redirect("/version", "/nodeinfo/2.1"))
|
2024-05-23 14:59:56 +00:00
|
|
|
// For backwards compatibility, can be removed after Lemmy 0.20
|
2024-06-03 21:30:16 +00:00
|
|
|
.service(web::redirect("/nodeinfo/2.0.json", "/nodeinfo/2.1"))
|
|
|
|
.service(web::redirect("/nodeinfo/2.1.json", "/nodeinfo/2.1"))
|
2023-07-19 10:09:04 +00:00
|
|
|
.route(
|
|
|
|
"/.well-known/nodeinfo",
|
|
|
|
web::get().to(node_info_well_known).wrap(cache_3days()),
|
|
|
|
);
|
2019-12-31 12:55:33 +00:00
|
|
|
}
|
|
|
|
|
2024-04-10 14:14:11 +00:00
|
|
|
async fn node_info_well_known(context: web::Data<LemmyContext>) -> LemmyResult<HttpResponse> {
|
2020-01-19 11:32:02 +00:00
|
|
|
let node_info = NodeInfoWellKnown {
|
2021-11-21 17:10:39 +00:00
|
|
|
links: vec![NodeInfoWellKnownLinks {
|
2024-05-23 14:59:56 +00:00
|
|
|
rel: Url::parse("http://nodeinfo.diaspora.software/ns/schema/2.1")?,
|
2020-04-08 12:37:05 +00:00
|
|
|
href: Url::parse(&format!(
|
2024-06-03 21:30:16 +00:00
|
|
|
"{}/nodeinfo/2.1",
|
2021-09-22 15:57:09 +00:00
|
|
|
&context.settings().get_protocol_and_hostname(),
|
2020-04-08 12:37:05 +00:00
|
|
|
))?,
|
2021-11-21 17:10:39 +00:00
|
|
|
}],
|
2020-01-19 11:32:02 +00:00
|
|
|
};
|
2020-04-08 12:37:05 +00:00
|
|
|
Ok(HttpResponse::Ok().json(node_info))
|
2019-11-15 02:08:25 +00:00
|
|
|
}
|
|
|
|
|
2020-08-18 13:43:50 +00:00
|
|
|
async fn node_info(context: web::Data<LemmyContext>) -> Result<HttpResponse, Error> {
|
2024-09-16 15:18:16 +00:00
|
|
|
let site_view = SiteView::read_local(&mut context.pool()).await?;
|
2020-07-01 12:54:29 +00:00
|
|
|
|
2023-08-01 08:40:42 +00:00
|
|
|
// Since there are 3 registration options,
|
|
|
|
// we need to set open_registrations as true if RegistrationMode is not Closed.
|
|
|
|
let open_registrations = Some(site_view.local_site.registration_mode != RegistrationMode::Closed);
|
2020-07-01 12:54:29 +00:00
|
|
|
let json = NodeInfo {
|
2024-05-14 02:53:20 +00:00
|
|
|
version: Some("2.1".to_string()),
|
2023-02-18 14:36:12 +00:00
|
|
|
software: Some(NodeInfoSoftware {
|
|
|
|
name: Some("lemmy".to_string()),
|
2024-04-02 15:19:51 +00:00
|
|
|
version: Some(VERSION.to_string()),
|
2024-05-14 02:53:20 +00:00
|
|
|
repository: Some("https://github.com/LemmyNet/lemmy".to_string()),
|
|
|
|
homepage: Some("https://join-lemmy.org/".to_string()),
|
2023-02-18 14:36:12 +00:00
|
|
|
}),
|
2024-05-14 02:53:20 +00:00
|
|
|
protocols: Some(vec!["activitypub".to_string()]),
|
2023-02-18 14:36:12 +00:00
|
|
|
usage: Some(NodeInfoUsage {
|
|
|
|
users: Some(NodeInfoUsers {
|
|
|
|
total: Some(site_view.counts.users),
|
|
|
|
active_halfyear: Some(site_view.counts.users_active_half_year),
|
|
|
|
active_month: Some(site_view.counts.users_active_month),
|
|
|
|
}),
|
|
|
|
local_posts: Some(site_view.counts.posts),
|
|
|
|
local_comments: Some(site_view.counts.comments),
|
|
|
|
}),
|
2023-01-05 01:42:30 +00:00
|
|
|
open_registrations,
|
2024-05-14 02:53:20 +00:00
|
|
|
services: Some(NodeInfoServices {
|
|
|
|
inbound: Some(vec![]),
|
|
|
|
outbound: Some(vec![]),
|
|
|
|
}),
|
2024-05-31 20:38:46 +00:00
|
|
|
metadata: Some(HashMap::new()),
|
2020-07-01 12:54:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(HttpResponse::Ok().json(json))
|
2019-11-15 02:08:25 +00:00
|
|
|
}
|
2020-01-19 11:32:02 +00:00
|
|
|
|
2020-03-18 15:08:08 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug)]
|
2024-06-03 21:30:00 +00:00
|
|
|
pub struct NodeInfoWellKnown {
|
2021-11-21 17:10:39 +00:00
|
|
|
pub links: Vec<NodeInfoWellKnownLinks>,
|
2020-01-19 11:32:02 +00:00
|
|
|
}
|
|
|
|
|
2020-03-18 15:08:08 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug)]
|
2024-06-03 21:30:00 +00:00
|
|
|
pub struct NodeInfoWellKnownLinks {
|
2020-04-08 12:37:05 +00:00
|
|
|
pub rel: Url,
|
|
|
|
pub href: Url,
|
2020-01-19 11:32:02 +00:00
|
|
|
}
|
|
|
|
|
2024-05-14 02:53:20 +00:00
|
|
|
/// Nodeinfo spec: http://nodeinfo.diaspora.software/docson/index.html#/ns/schema/2.1
|
2023-02-18 14:36:12 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Default)]
|
|
|
|
#[serde(rename_all = "camelCase", default)]
|
|
|
|
pub struct NodeInfo {
|
|
|
|
pub version: Option<String>,
|
|
|
|
pub software: Option<NodeInfoSoftware>,
|
|
|
|
pub protocols: Option<Vec<String>>,
|
|
|
|
pub usage: Option<NodeInfoUsage>,
|
|
|
|
pub open_registrations: Option<bool>,
|
2024-05-14 02:53:20 +00:00
|
|
|
/// These fields are required by the spec for no reason
|
|
|
|
pub services: Option<NodeInfoServices>,
|
2024-05-31 20:38:46 +00:00
|
|
|
pub metadata: Option<HashMap<String, String>>,
|
2020-01-19 11:32:02 +00:00
|
|
|
}
|
|
|
|
|
2023-02-18 14:36:12 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Default)]
|
|
|
|
#[serde(default)]
|
|
|
|
pub struct NodeInfoSoftware {
|
|
|
|
pub name: Option<String>,
|
|
|
|
pub version: Option<String>,
|
2024-05-14 02:53:20 +00:00
|
|
|
pub repository: Option<String>,
|
|
|
|
pub homepage: Option<String>,
|
2020-01-19 11:32:02 +00:00
|
|
|
}
|
|
|
|
|
2023-02-18 14:36:12 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Default)]
|
|
|
|
#[serde(rename_all = "camelCase", default)]
|
|
|
|
pub struct NodeInfoUsage {
|
|
|
|
pub users: Option<NodeInfoUsers>,
|
|
|
|
pub local_posts: Option<i64>,
|
|
|
|
pub local_comments: Option<i64>,
|
2020-01-19 11:32:02 +00:00
|
|
|
}
|
|
|
|
|
2023-02-18 14:36:12 +00:00
|
|
|
#[derive(Serialize, Deserialize, Debug, Default)]
|
|
|
|
#[serde(rename_all = "camelCase", default)]
|
|
|
|
pub struct NodeInfoUsers {
|
|
|
|
pub total: Option<i64>,
|
|
|
|
pub active_halfyear: Option<i64>,
|
|
|
|
pub active_month: Option<i64>,
|
2020-01-19 11:32:02 +00:00
|
|
|
}
|
2024-05-14 02:53:20 +00:00
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize, Debug, Default)]
|
|
|
|
#[serde(rename_all = "camelCase", default)]
|
|
|
|
pub struct NodeInfoServices {
|
|
|
|
pub inbound: Option<Vec<String>>,
|
|
|
|
pub outbound: Option<Vec<String>>,
|
|
|
|
}
|