2020-04-09 17:24:50 -07:00
|
|
|
// The Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) The Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2020 Jason Volk <jason@zemos.net>
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
|
|
|
|
2021-02-01 22:38:03 -08:00
|
|
|
namespace ircd::m::bridge
|
|
|
|
{
|
2021-02-04 17:46:50 -08:00
|
|
|
static thread_local char tmp[4][512];
|
2021-02-01 22:38:03 -08:00
|
|
|
}
|
|
|
|
|
2020-04-09 17:24:50 -07:00
|
|
|
decltype(ircd::m::bridge::log)
|
|
|
|
ircd::m::bridge::log
|
|
|
|
{
|
|
|
|
"m.bridge"
|
|
|
|
};
|
|
|
|
|
2021-02-04 19:51:36 -08:00
|
|
|
ircd::json::object
|
|
|
|
ircd::m::bridge::protocol(const mutable_buffer &buf,
|
|
|
|
const config &config,
|
|
|
|
const string_view &name)
|
|
|
|
{
|
|
|
|
const string_view &uri
|
|
|
|
{
|
|
|
|
make_uri(tmp[0], config, fmt::sprintf
|
|
|
|
{
|
|
|
|
tmp[1], "thirdparty/protocol/%s",
|
|
|
|
url::encode(tmp[2], name),
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
|
|
|
switch(const query query{config, uri, buf}; query.code)
|
|
|
|
{
|
|
|
|
case http::OK:
|
|
|
|
return json::object
|
|
|
|
{
|
|
|
|
query.request.in.content
|
|
|
|
};
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
query.code
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-04 17:46:50 -08:00
|
|
|
bool
|
|
|
|
ircd::m::bridge::exists(const config &config,
|
|
|
|
const m::user::id &user_id)
|
|
|
|
{
|
|
|
|
const string_view &uri
|
|
|
|
{
|
|
|
|
make_uri(tmp[0], config, fmt::sprintf
|
|
|
|
{
|
|
|
|
tmp[1], "users/%s",
|
|
|
|
url::encode(tmp[2], user_id),
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
|
|
|
switch(const query query{config, uri}; query.code)
|
|
|
|
{
|
|
|
|
case http::OK: return true;
|
|
|
|
case http::NOT_FOUND: return false;
|
|
|
|
default:
|
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
query.code
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ircd::m::bridge::exists(const config &config,
|
|
|
|
const m::room::alias &room_alias)
|
|
|
|
{
|
|
|
|
const string_view &uri
|
|
|
|
{
|
|
|
|
make_uri(tmp[0], config, fmt::sprintf
|
|
|
|
{
|
|
|
|
tmp[1], "rooms/%s",
|
|
|
|
url::encode(tmp[2], room_alias),
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
|
|
|
switch(const query query{config, uri}; query.code)
|
|
|
|
{
|
|
|
|
case http::OK: return true;
|
|
|
|
case http::NOT_FOUND: return false;
|
|
|
|
default:
|
|
|
|
throw http::error
|
|
|
|
{
|
|
|
|
query.code
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-01 22:38:03 -08:00
|
|
|
ircd::string_view
|
|
|
|
ircd::m::bridge::make_uri(const mutable_buffer &buf,
|
|
|
|
const config &config,
|
|
|
|
const string_view &path)
|
|
|
|
{
|
|
|
|
const rfc3986::uri base_url
|
|
|
|
{
|
|
|
|
at<"url"_>(config)
|
|
|
|
};
|
|
|
|
|
|
|
|
char hs_token[256];
|
|
|
|
return fmt::sprintf
|
|
|
|
{
|
|
|
|
buf, "%s/_matrix/app/v1/%s?access_token=%s",
|
|
|
|
base_url.path,
|
|
|
|
path,
|
|
|
|
url::encode(hs_token, at<"hs_token"_>(config)),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-04-09 17:24:50 -07:00
|
|
|
//
|
|
|
|
// query
|
|
|
|
//
|
|
|
|
|
2020-04-11 20:12:42 -07:00
|
|
|
decltype(ircd::m::bridge::query::timeout)
|
|
|
|
ircd::m::bridge::query::timeout
|
|
|
|
{
|
|
|
|
{ "name", "ircd.m.bridge.query.timeout" },
|
2021-02-04 17:46:50 -08:00
|
|
|
{ "default", 10L },
|
2020-04-11 20:12:42 -07:00
|
|
|
};
|
|
|
|
|
2020-04-09 17:24:50 -07:00
|
|
|
ircd::m::bridge::query::query(const config &config,
|
2021-02-04 17:46:50 -08:00
|
|
|
const string_view &uri,
|
|
|
|
const mutable_buffer &in_body)
|
2020-04-09 17:24:50 -07:00
|
|
|
:base_url
|
|
|
|
{
|
|
|
|
at<"url"_>(config)
|
|
|
|
}
|
|
|
|
,buf
|
|
|
|
{
|
|
|
|
8_KiB
|
|
|
|
}
|
|
|
|
,uri
|
|
|
|
{
|
2021-02-04 17:46:50 -08:00
|
|
|
uri
|
2020-04-11 20:12:42 -07:00
|
|
|
}
|
|
|
|
,wb
|
|
|
|
{
|
2021-02-04 17:46:50 -08:00
|
|
|
buf
|
2020-04-09 17:24:50 -07:00
|
|
|
}
|
|
|
|
,hypertext
|
|
|
|
{
|
|
|
|
wb,
|
|
|
|
base_url.remote,
|
|
|
|
"GET",
|
|
|
|
uri,
|
|
|
|
}
|
2021-02-04 17:46:50 -08:00
|
|
|
,sopts
|
2020-04-09 17:24:50 -07:00
|
|
|
{
|
2021-02-04 17:46:50 -08:00
|
|
|
false, // http_exceptions
|
2020-04-09 17:24:50 -07:00
|
|
|
}
|
|
|
|
,request
|
|
|
|
{
|
2021-02-04 17:46:50 -08:00
|
|
|
net::hostport { base_url.remote },
|
|
|
|
server::out { wb.completed(), {} },
|
|
|
|
server::in { wb.remains(), !empty(in_body)? in_body: wb.remains() },
|
|
|
|
&sopts,
|
2020-04-09 17:24:50 -07:00
|
|
|
}
|
|
|
|
,code
|
|
|
|
{
|
2020-04-11 20:12:42 -07:00
|
|
|
request.get(seconds(timeout))
|
2020-04-09 17:24:50 -07:00
|
|
|
}
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// config
|
|
|
|
//
|
|
|
|
|
|
|
|
bool
|
2021-02-03 02:49:07 -08:00
|
|
|
ircd::m::bridge::config::exists(const string_view &id)
|
2020-04-09 17:24:50 -07:00
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
return get(std::nothrow, id, []
|
|
|
|
(const auto &, const auto &, const auto &)
|
2020-04-11 18:36:33 -07:00
|
|
|
{
|
|
|
|
});
|
2020-04-09 17:24:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ircd::m::bridge::config::get(const string_view &id,
|
|
|
|
const closure &closure)
|
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
if(unlikely(!get(std::nothrow, id, closure)))
|
2020-04-09 17:24:50 -07:00
|
|
|
throw m::NOT_FOUND
|
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
"No bridge config found for '%s'",
|
|
|
|
id,
|
2020-04-09 17:24:50 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ircd::m::bridge::config::get(std::nothrow_t,
|
|
|
|
const string_view &id,
|
|
|
|
const closure &closure)
|
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
return !config::for_each([&id, &closure]
|
|
|
|
(const auto &event_idx, const auto &event, const config &config)
|
2020-04-09 17:24:50 -07:00
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
if(json::get<"id"_>(config) != id)
|
|
|
|
return true;
|
2020-04-09 17:24:50 -07:00
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
closure(event_idx, event, config);
|
|
|
|
return false;
|
2020-04-09 17:24:50 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
bool
|
|
|
|
ircd::m::bridge::config::for_each(const closure_bool &closure)
|
2020-04-09 17:24:50 -07:00
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
return events::type::for_each_in("ircd.bridge", [&closure]
|
|
|
|
(const string_view &, const event::idx &event_idx)
|
2020-04-09 17:24:50 -07:00
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
const m::event::fetch event
|
|
|
|
{
|
|
|
|
std::nothrow, event_idx
|
|
|
|
};
|
2020-04-09 17:24:50 -07:00
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
if(unlikely(!event.valid))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if(!my(event))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if(!defined(json::get<"state_key"_>(event)))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
const bridge::config config
|
2020-04-09 17:24:50 -07:00
|
|
|
{
|
2021-02-03 02:49:07 -08:00
|
|
|
json::get<"content"_>(event)
|
2020-04-09 17:24:50 -07:00
|
|
|
};
|
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
if(!json::get<"id"_>(config))
|
|
|
|
return true;
|
2020-04-09 17:24:50 -07:00
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
// the state_key has to match the id for now
|
|
|
|
if(json::get<"id"_>(config) != json::get<"state_key"_>(event))
|
|
|
|
return true;
|
2020-04-09 17:24:50 -07:00
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
// filter replaced state
|
|
|
|
if(room::state::next(event_idx))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// filter redacted state
|
|
|
|
if(redacted(event_idx))
|
|
|
|
return true;
|
2020-04-09 17:24:50 -07:00
|
|
|
|
2021-02-03 02:49:07 -08:00
|
|
|
const user::id sender
|
|
|
|
{
|
|
|
|
json::get<"sender"_>(event)
|
|
|
|
};
|
|
|
|
|
|
|
|
if(!is_oper(sender))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if(likely(closure(event_idx, event, config)))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
});
|
2020-04-09 17:24:50 -07:00
|
|
|
}
|