use std::{convert::TryFrom, env}; use matrix_sdk_appservice::{ matrix_sdk::{ async_trait, room::Room, ruma::{ events::{ room::member::{MemberEventContent, MembershipState}, SyncStateEvent, }, UserId, }, EventHandler, }, AppService, AppServiceRegistration, }; use tracing::{error, trace}; struct AppServiceEventHandler { appservice: AppService, } impl AppServiceEventHandler { pub fn new(appservice: AppService) -> Self { Self { appservice } } pub async fn handle_room_member( &self, room: Room, event: &SyncStateEvent<MemberEventContent>, ) -> Result<(), Box<dyn std::error::Error>> { if !self.appservice.user_id_is_in_namespace(&event.state_key)? { trace!("not an appservice user: {}", event.state_key); } else if let MembershipState::Invite = event.content.membership { let user_id = UserId::try_from(event.state_key.clone())?; let appservice = self.appservice.clone(); appservice.register_virtual_user(user_id.localpart()).await?; let client = appservice.virtual_user_client(user_id.localpart()).await?; client.join_room_by_id(room.room_id()).await?; } Ok(()) } } #[async_trait] impl EventHandler for AppServiceEventHandler { async fn on_room_member(&self, room: Room, event: &SyncStateEvent<MemberEventContent>) { match self.handle_room_member(room, event).await { Ok(_) => (), Err(error) => error!("{:?}", error), } } } #[tokio::main] pub async fn main() -> Result<(), Box<dyn std::error::Error>> { env::set_var("RUST_LOG", "matrix_sdk=debug,matrix_sdk_appservice=debug"); tracing_subscriber::fmt::init(); let homeserver_url = "http://localhost:8008"; let server_name = "localhost"; let registration = AppServiceRegistration::try_from_yaml_file("./tests/registration.yaml")?; let mut appservice = AppService::new(homeserver_url, server_name, registration).await?; appservice.set_event_handler(Box::new(AppServiceEventHandler::new(appservice.clone()))).await?; let (host, port) = appservice.registration().get_host_and_port()?; appservice.run(host, port).await?; Ok(()) }