2018-02-14 21:23:20 +01:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 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.
|
|
|
|
|
|
|
|
#include "rooms.h"
|
|
|
|
|
2018-02-23 05:16:12 +01:00
|
|
|
using namespace ircd::m;
|
2018-02-14 21:23:20 +01:00
|
|
|
using namespace ircd;
|
|
|
|
|
2019-04-22 22:21:55 +02:00
|
|
|
static resource::response
|
|
|
|
get__state(client &client,
|
|
|
|
const resource::request &request,
|
|
|
|
const m::room::state &state);
|
|
|
|
|
2018-02-14 21:23:20 +01:00
|
|
|
resource::response
|
2019-03-14 22:25:37 +01:00
|
|
|
put__state(client &client,
|
2018-02-14 21:23:20 +01:00
|
|
|
const resource::request &request,
|
2018-02-23 05:16:12 +01:00
|
|
|
const room::id &room_id)
|
2018-02-14 21:23:20 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
if(request.parv.size() < 3)
|
|
|
|
throw m::NEED_MORE_PARAMS
|
|
|
|
{
|
|
|
|
"'type' path parameter required."
|
|
|
|
};
|
|
|
|
|
2019-02-18 21:25:00 +01:00
|
|
|
char type_buf[m::event::TYPE_MAX_SIZE];
|
2018-02-14 21:23:20 +01:00
|
|
|
const string_view &type
|
|
|
|
{
|
2018-12-07 01:41:47 +01:00
|
|
|
url::decode(type_buf, request.parv[2])
|
2018-02-14 21:23:20 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
if(request.parv.size() < 3)
|
|
|
|
throw m::NEED_MORE_PARAMS
|
|
|
|
{
|
|
|
|
"'state_key' path parameter required."
|
|
|
|
};
|
|
|
|
|
2019-02-18 21:25:00 +01:00
|
|
|
char skey_buf[m::event::STATE_KEY_MAX_SIZE];
|
2018-02-14 21:23:20 +01:00
|
|
|
const string_view &state_key
|
|
|
|
{
|
2018-12-07 01:41:47 +01:00
|
|
|
url::decode(skey_buf, request.parv[3])
|
2018-02-14 21:23:20 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
const json::object &content
|
2018-02-14 21:23:20 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
request.content
|
2018-02-14 21:23:20 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
const auto event_id
|
|
|
|
{
|
|
|
|
send(room_id, request.user_id, type, state_key, content)
|
|
|
|
};
|
2018-02-14 21:23:20 +01:00
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
return resource::response
|
|
|
|
{
|
|
|
|
client, json::members
|
|
|
|
{
|
|
|
|
{ "event_id", event_id }
|
|
|
|
}
|
|
|
|
};
|
2018-02-14 21:23:20 +01:00
|
|
|
}
|
2018-02-16 21:43:30 +01:00
|
|
|
|
|
|
|
resource::response
|
2019-03-14 22:25:37 +01:00
|
|
|
get__state(client &client,
|
2018-02-16 21:43:30 +01:00
|
|
|
const resource::request &request,
|
2018-02-23 05:16:12 +01:00
|
|
|
const room::id &room_id)
|
2018-02-16 21:43:30 +01:00
|
|
|
{
|
2019-02-18 21:25:00 +01:00
|
|
|
char type_buf[m::event::TYPE_MAX_SIZE];
|
2018-02-16 21:43:30 +01:00
|
|
|
const string_view &type
|
|
|
|
{
|
2018-12-07 01:41:47 +01:00
|
|
|
url::decode(type_buf, request.parv[2])
|
2018-02-16 21:43:30 +01:00
|
|
|
};
|
|
|
|
|
2019-02-18 21:25:00 +01:00
|
|
|
char skey_buf[m::event::STATE_KEY_MAX_SIZE];
|
2018-02-16 21:43:30 +01:00
|
|
|
const string_view &state_key
|
|
|
|
{
|
2018-12-07 01:41:47 +01:00
|
|
|
url::decode(skey_buf, request.parv[3])
|
2018-02-16 21:43:30 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
// (non-standard) Allow an event_id to be passed in the query string
|
|
|
|
// for reference framing.
|
|
|
|
m::event::id::buf event_id;
|
|
|
|
if(request.query["event_id"])
|
|
|
|
url::decode(event_id, request.query["event_id"]);
|
2018-02-16 21:43:30 +01:00
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
const m::room room
|
2018-02-16 21:43:30 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
room_id, event_id
|
2018-02-16 21:43:30 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
if(!exists(room))
|
|
|
|
throw m::NOT_FOUND
|
2018-02-16 21:43:30 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
"Room %s not found.",
|
|
|
|
string_view{room_id}
|
|
|
|
};
|
2018-02-23 05:16:12 +01:00
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
if(!room.visible(request.user_id))
|
|
|
|
throw m::ACCESS_DENIED
|
|
|
|
{
|
|
|
|
"You are not permitted to view %s",
|
|
|
|
string_view{room_id}
|
|
|
|
};
|
|
|
|
|
|
|
|
const m::room::state state
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
room
|
2018-02-23 05:16:12 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
if(!type)
|
|
|
|
return get__state(client, request, state);
|
2018-02-23 05:16:12 +01:00
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
m::event::fetch::opts fopts;
|
|
|
|
fopts.query_json_force = true;
|
|
|
|
const m::event::fetch event
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
state.get(type, state_key), fopts
|
|
|
|
};
|
|
|
|
|
|
|
|
if(!visible(event, request.user_id))
|
|
|
|
throw m::ACCESS_DENIED
|
|
|
|
{
|
|
|
|
"You are not permitted to view this state in %s",
|
|
|
|
string_view{room_id}
|
|
|
|
};
|
2018-02-23 05:16:12 +01:00
|
|
|
|
|
|
|
return resource::response
|
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
client, event.source
|
2018-02-23 05:16:12 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
resource::response
|
|
|
|
get__state(client &client,
|
|
|
|
const resource::request &request,
|
2019-03-14 22:25:37 +01:00
|
|
|
const m::room::state &state)
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
resource::response::chunked response
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
client, http::OK
|
2018-02-23 05:16:12 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
json::stack out
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
response.buf, response.flusher()
|
2018-02-23 05:16:12 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
json::stack::array top
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
out
|
2018-02-23 05:16:12 +01:00
|
|
|
};
|
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
state.for_each([&request, &top]
|
|
|
|
(const m::event &event)
|
2018-02-23 05:16:12 +01:00
|
|
|
{
|
2019-03-14 22:25:37 +01:00
|
|
|
if(!visible(event, request.user_id))
|
|
|
|
return;
|
2018-02-23 05:16:12 +01:00
|
|
|
|
2019-03-14 22:25:37 +01:00
|
|
|
top.append(event);
|
|
|
|
});
|
|
|
|
|
2019-06-24 07:09:41 +02:00
|
|
|
return std::move(response);
|
2018-02-23 05:16:12 +01:00
|
|
|
}
|
2019-04-22 22:21:55 +02:00
|
|
|
|
|
|
|
m::event::id::buf
|
|
|
|
IRCD_MODULE_EXPORT
|
|
|
|
ircd::m::send(const m::room &room,
|
|
|
|
const m::id::user &sender,
|
|
|
|
const string_view &type,
|
|
|
|
const string_view &state_key,
|
|
|
|
const json::iov &content)
|
|
|
|
{
|
|
|
|
json::iov event;
|
|
|
|
const json::iov::push push[]
|
|
|
|
{
|
|
|
|
{ event, { "sender", sender }},
|
|
|
|
{ event, { "type", type }},
|
|
|
|
{ event, { "state_key", state_key }},
|
|
|
|
};
|
|
|
|
|
|
|
|
return commit(room, event, content);
|
|
|
|
}
|