2018-09-14 05:08:08 -07: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.
|
|
|
|
|
2019-08-11 22:22:57 -07:00
|
|
|
namespace ircd::m
|
|
|
|
{
|
2019-09-26 12:16:22 -07:00
|
|
|
static string_view gen_password_hash(const mutable_buffer &, const string_view &);
|
2019-08-11 22:22:57 -07:00
|
|
|
static room create_user_room(const user::id &, const room::id &, const json::members &contents);
|
|
|
|
}
|
|
|
|
|
2019-09-26 19:28:25 -07:00
|
|
|
bool
|
|
|
|
ircd::m::exists(const user::id &user_id)
|
|
|
|
{
|
|
|
|
// The way we know a user exists is testing if their room exists.
|
|
|
|
const m::user::room user_room
|
|
|
|
{
|
|
|
|
user_id
|
|
|
|
};
|
|
|
|
|
|
|
|
return m::exists(user_room);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ircd::m::exists(const user &user)
|
|
|
|
{
|
|
|
|
return exists(user.user_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ircd::m::my(const user &user)
|
|
|
|
{
|
|
|
|
return my(user.user_id);
|
|
|
|
}
|
|
|
|
|
2019-07-15 10:09:26 -07:00
|
|
|
ircd::m::user
|
2019-08-11 22:22:57 -07:00
|
|
|
IRCD_MODULE_EXPORT
|
2019-07-15 10:09:26 -07:00
|
|
|
ircd::m::create(const m::user::id &user_id,
|
|
|
|
const json::members &contents)
|
2018-10-23 13:22:08 -07:00
|
|
|
{
|
|
|
|
const m::user user
|
|
|
|
{
|
|
|
|
user_id
|
|
|
|
};
|
|
|
|
|
2019-08-11 22:22:57 -07:00
|
|
|
const m::room::id::buf room_id
|
2018-10-23 13:22:08 -07:00
|
|
|
{
|
|
|
|
user.room_id()
|
|
|
|
};
|
|
|
|
|
2019-08-11 22:22:57 -07:00
|
|
|
const m::room room
|
2018-10-23 13:22:08 -07:00
|
|
|
{
|
2019-08-11 22:22:57 -07:00
|
|
|
create_user_room(user_id, room_id, contents)
|
2018-10-23 13:22:08 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
return user;
|
|
|
|
}
|
2019-08-11 22:22:57 -07:00
|
|
|
|
|
|
|
ircd::m::room
|
|
|
|
ircd::m::create_user_room(const user::id &user_id,
|
|
|
|
const room::id &room_id,
|
|
|
|
const json::members &contents)
|
|
|
|
try
|
|
|
|
{
|
2019-09-30 20:50:58 -07:00
|
|
|
return create(room_id, me(), "user");
|
2019-08-11 22:22:57 -07:00
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
if(m::exists(room_id))
|
|
|
|
return room_id;
|
|
|
|
|
|
|
|
log::error
|
|
|
|
{
|
|
|
|
log, "Failed to create user %s room %s :%s",
|
|
|
|
string_view{user_id},
|
|
|
|
string_view{room_id},
|
|
|
|
e.what()
|
|
|
|
};
|
|
|
|
|
|
|
|
throw;
|
|
|
|
}
|
2019-09-16 11:02:50 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// user::user
|
|
|
|
//
|
|
|
|
|
2019-09-26 19:28:25 -07:00
|
|
|
/// Generates a user-room ID into buffer; see room_id() overload.
|
|
|
|
ircd::m::id::room::buf
|
|
|
|
ircd::m::user::room_id()
|
|
|
|
const
|
|
|
|
{
|
|
|
|
ircd::m::id::room::buf buf;
|
|
|
|
return buf.assigned(room_id(buf));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This generates a room mxid for the "user's room" essentially serving as
|
|
|
|
/// a database mechanism for this specific user. This room_id is a hash of
|
|
|
|
/// the user's full mxid.
|
|
|
|
///
|
|
|
|
ircd::m::id::room
|
|
|
|
ircd::m::user::room_id(const mutable_buffer &buf)
|
|
|
|
const
|
|
|
|
{
|
|
|
|
assert(!empty(user_id));
|
|
|
|
const ripemd160::buf hash
|
|
|
|
{
|
|
|
|
ripemd160{user_id}
|
|
|
|
};
|
|
|
|
|
|
|
|
char b58[size(hash) * 2];
|
|
|
|
return
|
|
|
|
{
|
2019-09-30 20:50:58 -07:00
|
|
|
buf, b58encode(b58, hash), origin(my())
|
2019-09-26 19:28:25 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
ircd::m::device::id::buf
|
|
|
|
ircd::m::user::get_device_from_access_token(const string_view &token)
|
|
|
|
{
|
2019-09-30 20:50:58 -07:00
|
|
|
const m::room::id::buf tokens_room_id
|
|
|
|
{
|
|
|
|
"tokens", origin(my())
|
|
|
|
};
|
|
|
|
|
|
|
|
const m::room tokens
|
|
|
|
{
|
|
|
|
tokens_room_id
|
|
|
|
};
|
|
|
|
|
2019-09-26 19:28:25 -07:00
|
|
|
const event::idx event_idx
|
|
|
|
{
|
2019-09-30 20:50:58 -07:00
|
|
|
tokens.get("ircd.access_token", token)
|
2019-09-26 19:28:25 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
device::id::buf ret;
|
|
|
|
m::get(event_idx, "content", [&ret]
|
|
|
|
(const json::object &content)
|
|
|
|
{
|
|
|
|
ret = unquote(content.at("device_id"));
|
|
|
|
});
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ircd::string_view
|
|
|
|
ircd::m::user::gen_access_token(const mutable_buffer &buf)
|
|
|
|
{
|
|
|
|
static const size_t token_max{32};
|
2019-09-28 01:19:25 -07:00
|
|
|
static const auto &token_dict
|
|
|
|
{
|
|
|
|
rand::dict::alpha
|
|
|
|
};
|
2019-09-26 19:28:25 -07:00
|
|
|
|
|
|
|
const mutable_buffer out
|
|
|
|
{
|
|
|
|
data(buf), std::min(token_max, size(buf))
|
|
|
|
};
|
|
|
|
|
|
|
|
return rand::string(token_dict, out);
|
|
|
|
}
|
|
|
|
|
|
|
|
ircd::m::event::id::buf
|
|
|
|
ircd::m::user::activate()
|
|
|
|
{
|
2019-10-02 18:30:14 -07:00
|
|
|
const m::user::room user_room
|
2019-09-26 19:28:25 -07:00
|
|
|
{
|
2019-10-02 18:30:14 -07:00
|
|
|
user_id
|
2019-09-26 19:28:25 -07:00
|
|
|
};
|
|
|
|
|
2019-10-02 18:30:14 -07:00
|
|
|
return send(user_room, m::me(), "ircd.account", "active", json::members
|
|
|
|
{
|
|
|
|
{ "value", true }
|
|
|
|
});
|
2019-09-26 19:28:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ircd::m::event::id::buf
|
|
|
|
ircd::m::user::deactivate()
|
|
|
|
{
|
2019-10-02 18:30:14 -07:00
|
|
|
const m::user::room user_room
|
2019-09-26 19:28:25 -07:00
|
|
|
{
|
2019-10-02 18:30:14 -07:00
|
|
|
user_id
|
2019-09-26 19:28:25 -07:00
|
|
|
};
|
|
|
|
|
2019-10-02 18:30:14 -07:00
|
|
|
return send(user_room, m::me(), "ircd.account", "active", json::members
|
|
|
|
{
|
|
|
|
{ "value", false }
|
|
|
|
});
|
2019-09-26 19:28:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ircd::m::user::is_active()
|
|
|
|
const
|
|
|
|
{
|
2019-10-02 18:30:14 -07:00
|
|
|
const m::user::room user_room
|
|
|
|
{
|
|
|
|
user_id
|
|
|
|
};
|
2019-09-26 19:28:25 -07:00
|
|
|
|
2019-10-02 18:30:14 -07:00
|
|
|
const m::event::idx &event_idx
|
2019-09-26 19:28:25 -07:00
|
|
|
{
|
2019-10-02 18:30:14 -07:00
|
|
|
user_room.get(std::nothrow, "ircd.account", "active")
|
2019-09-26 19:28:25 -07:00
|
|
|
};
|
|
|
|
|
2019-10-02 18:30:14 -07:00
|
|
|
return m::query(event_idx, "content", []
|
|
|
|
(const json::object &content)
|
|
|
|
{
|
|
|
|
return content.get<bool>("value", false);
|
|
|
|
});
|
2019-09-26 19:28:25 -07:00
|
|
|
}
|
|
|
|
|
2019-09-16 11:02:50 -07:00
|
|
|
ircd::m::event::id::buf
|
|
|
|
IRCD_MODULE_EXPORT
|
|
|
|
ircd::m::user::password(const string_view &password)
|
|
|
|
{
|
|
|
|
char buf[64];
|
|
|
|
const auto supplied
|
|
|
|
{
|
|
|
|
gen_password_hash(buf, password)
|
|
|
|
};
|
|
|
|
|
|
|
|
const m::user::room user_room
|
|
|
|
{
|
|
|
|
user_id
|
|
|
|
};
|
|
|
|
|
|
|
|
return send(user_room, user_id, "ircd.password", user_id,
|
|
|
|
{
|
|
|
|
{ "sha256", supplied }
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
IRCD_MODULE_EXPORT
|
|
|
|
ircd::m::user::is_password(const string_view &password)
|
2019-09-28 01:19:25 -07:00
|
|
|
const try
|
2019-09-16 11:02:50 -07:00
|
|
|
{
|
|
|
|
char buf[64];
|
|
|
|
const auto supplied
|
|
|
|
{
|
|
|
|
gen_password_hash(buf, password)
|
|
|
|
};
|
|
|
|
|
|
|
|
bool ret{false};
|
|
|
|
const m::user::room user_room
|
|
|
|
{
|
|
|
|
user_id
|
|
|
|
};
|
|
|
|
|
|
|
|
const ctx::uninterruptible::nothrow ui;
|
|
|
|
user_room.get("ircd.password", user_id, [&supplied, &ret]
|
|
|
|
(const m::event &event)
|
|
|
|
{
|
|
|
|
const json::object &content
|
|
|
|
{
|
|
|
|
json::at<"content"_>(event)
|
|
|
|
};
|
|
|
|
|
|
|
|
const auto &correct
|
|
|
|
{
|
|
|
|
unquote(content.at("sha256"))
|
|
|
|
};
|
|
|
|
|
|
|
|
ret = supplied == correct;
|
|
|
|
});
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
catch(const m::NOT_FOUND &e)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
log::critical
|
|
|
|
{
|
|
|
|
"is_password__user(): %s %s",
|
|
|
|
string_view{user_id},
|
|
|
|
e.what()
|
|
|
|
};
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ircd::string_view
|
|
|
|
ircd::m::gen_password_hash(const mutable_buffer &out,
|
|
|
|
const string_view &supplied_password)
|
|
|
|
{
|
|
|
|
//TODO: ADD SALT
|
|
|
|
const sha256::buf hash
|
|
|
|
{
|
|
|
|
sha256{supplied_password}
|
|
|
|
};
|
|
|
|
|
|
|
|
return b64encode_unpadded(out, hash);
|
|
|
|
}
|
2019-09-26 19:28:25 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// user::room
|
|
|
|
//
|
|
|
|
|
|
|
|
ircd::m::user::room::room(const m::user::id &user_id,
|
|
|
|
const vm::copts *const &copts,
|
|
|
|
const event::fetch::opts *const &fopts)
|
|
|
|
:room
|
|
|
|
{
|
|
|
|
m::user{user_id}, copts, fopts
|
|
|
|
}
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ircd::m::user::room::room(const m::user &user,
|
|
|
|
const vm::copts *const &copts,
|
|
|
|
const event::fetch::opts *const &fopts)
|
|
|
|
:user{user}
|
|
|
|
,room_id{user.room_id()}
|
|
|
|
{
|
|
|
|
static_cast<m::room &>(*this) =
|
|
|
|
{
|
|
|
|
room_id, copts, fopts
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ircd::m::user::room::is(const room::id &room_id,
|
|
|
|
const user::id &user_id)
|
|
|
|
{
|
|
|
|
const user::room user_room{user_id};
|
|
|
|
return user_room.room_id == room_id;
|
|
|
|
}
|