use flowsnet_platform_sdk::write_error_log;
use http_req::{
request::{self, Method, Request},
uri::Uri,
};
use lazy_static::lazy_static;
use rand::{distributions::Alphanumeric, Rng};
use serde::{Deserialize, Serialize};
use std::future::Future;
use std::io::{self, Write};
lazy_static! {
static ref SLACK_API_PREFIX: String = String::from(
std::option_env!("SLACK_API_PREFIX")
.unwrap_or("https://slack-flows-extension.vercel.app/api")
);
}
extern "C" {
fn is_listening() -> i32;
fn get_flows_user(p: *mut u8) -> i32;
fn get_flow_id(p: *mut u8) -> i32;
fn get_event_body_length() -> i32;
fn get_event_body(p: *mut u8) -> i32;
fn set_output(p: *const u8, len: i32);
fn set_error_code(code: i16);
}
#[derive(Debug, Serialize, Deserialize)]
pub struct SlackMessage {
#[serde(rename = "type")]
pub event_type: String,
pub channel: String,
pub user: String,
pub text: String,
pub channel_type: String,
}
#[derive(Debug, Serialize, Deserialize)]
pub struct Event {
pub challenge: Option<String>,
pub team_id: Option<String>,
pub event: Option<SlackMessage>,
}
pub async fn revoke_listeners() {
unsafe {
let mut flows_user = Vec::<u8>::with_capacity(100);
let c = get_flows_user(flows_user.as_mut_ptr());
flows_user.set_len(c as usize);
let flows_user = String::from_utf8(flows_user).unwrap();
let mut flow_id = Vec::<u8>::with_capacity(100);
let c = get_flow_id(flow_id.as_mut_ptr());
if c == 0 {
panic!("Failed to get flow id");
}
flow_id.set_len(c as usize);
let flow_id = String::from_utf8(flow_id).unwrap();
let mut writer = Vec::new();
let res = request::get(
format!(
"{}/{}/{}/revoke",
SLACK_API_PREFIX.as_str(),
flows_user,
flow_id
),
&mut writer,
)
.unwrap();
match res.status_code().is_success() {
true => (),
false => {
write_error_log!(String::from_utf8_lossy(&writer));
set_error_code(format!("{}", res.status_code()).parse::<i16>().unwrap_or(0));
}
}
}
}
pub async fn listen_to_channel<F, Fut>(team_name: &str, channel_name: &str, cb: F)
where
F: FnOnce(SlackMessage) -> Fut,
Fut: Future<Output = ()>,
{
unsafe {
match is_listening() {
1 => {
let mut flows_user = Vec::<u8>::with_capacity(100);
let c = get_flows_user(flows_user.as_mut_ptr());
flows_user.set_len(c as usize);
let flows_user = String::from_utf8(flows_user).unwrap();
let mut flow_id = Vec::<u8>::with_capacity(100);
let c = get_flow_id(flow_id.as_mut_ptr());
if c == 0 {
panic!("Failed to get flow id");
}
flow_id.set_len(c as usize);
let flow_id = String::from_utf8(flow_id).unwrap();
let mut writer = Vec::new();
let res = request::get(
format!(
"{}/{}/{}/listen?team={}&channel={}",
SLACK_API_PREFIX.as_str(),
flows_user,
flow_id,
team_name,
channel_name
),
&mut writer,
)
.unwrap();
match res.status_code().is_success() {
true => {
let output = format!(
"[{}] Listening to channel `{}` on workspace `{}`",
std::env!("CARGO_CRATE_NAME"),
channel_name,
team_name
);
set_output(output.as_ptr(), output.len() as i32);
if let Ok(sm) = serde_json::from_slice::<SlackMessage>(&writer) {
cb(sm);
}
}
false => {
write_error_log!(String::from_utf8_lossy(&writer));
set_error_code(
format!("{}", res.status_code()).parse::<i16>().unwrap_or(0),
);
}
}
}
_ => {
if let Some(sm) = message_from_channel() {
cb(sm).await;
}
}
}
}
}
fn message_from_channel() -> Option<SlackMessage> {
unsafe {
let l = get_event_body_length();
let mut event_body = Vec::<u8>::with_capacity(l as usize);
let c = get_event_body(event_body.as_mut_ptr());
assert!(c == l);
event_body.set_len(c as usize);
match serde_json::from_slice::<Event>(&event_body) {
Ok(e) => e.event,
Err(_) => None,
}
}
}
pub async fn send_message_to_channel(team_name: &str, channel_name: &str, text: String) {
unsafe {
let mut flows_user = Vec::<u8>::with_capacity(100);
let c = get_flows_user(flows_user.as_mut_ptr());
flows_user.set_len(c as usize);
let flows_user = String::from_utf8(flows_user).unwrap();
let mut writer = Vec::new();
if let Ok(res) = request::post(
format!(
"{}/{}/send?team={}&channel={}",
SLACK_API_PREFIX.as_str(),
flows_user,
team_name,
channel_name
),
text.as_bytes(),
&mut writer,
) {
if !res.status_code().is_success() {
write_error_log!(String::from_utf8_lossy(&writer));
set_error_code(format!("{}", res.status_code()).parse::<i16>().unwrap_or(0));
}
}
}
}
pub async fn upload_file(
team_name: &str,
channel_name: &str,
file_name: &str,
file_type: &str,
file_bytes: Vec<u8>,
) {
unsafe {
let mut flows_user = Vec::<u8>::with_capacity(100);
let c = get_flows_user(flows_user.as_mut_ptr());
flows_user.set_len(c as usize);
let flows_user = String::from_utf8(flows_user).unwrap();
let boundary: String = rand::thread_rng()
.sample_iter(&Alphanumeric)
.take(15)
.map(char::from)
.collect();
let boundary = format!("------------------------{}", boundary);
if let Ok(file_part) =
compose_file_part(&boundary, channel_name, file_name, file_type, file_bytes)
{
let mut writer = Vec::new();
let uri = format!(
"{}/{}/upload?team={}&channel={}",
SLACK_API_PREFIX.as_str(),
flows_user,
team_name,
channel_name
);
let uri = Uri::try_from(uri.as_str()).unwrap();
if let Ok(res) = Request::new(&uri)
.method(Method::POST)
.header(
"Content-Type",
&format!("multipart/form-data; boundary={}", boundary),
)
.header("Content-Length", &file_part.len())
.body(&file_part)
.send(&mut writer)
{
if !res.status_code().is_success() {
write_error_log!(String::from_utf8_lossy(&writer));
set_error_code(format!("{}", res.status_code()).parse::<i16>().unwrap_or(0));
}
}
}
}
}
fn compose_file_part(
boundary: &str,
channel: &str,
file_name: &str,
file_type: &str,
file_bytes: Vec<u8>,
) -> io::Result<Vec<u8>> {
let mut data = Vec::new();
write!(data, "--{}\r\n", boundary)?;
write!(data, "Content-Disposition: form-data; name=\"channel\"\r\n")?;
write!(data, "\r\n{}\r\n", channel)?;
write!(data, "--{}\r\n", boundary)?;
write!(
data,
"Content-Disposition: form-data; name=\"file\"; filename=\"{}\"\r\n",
file_name
)?;
write!(data, "Content-Type: {}\r\n\r\n", file_type)?;
data.extend_from_slice(&file_bytes);
write!(data, "\r\n")?; write!(data, "--{}--\r\n", boundary)?;
Ok(data)
}