mirror of
https://git.pleroma.social/pleroma/pleroma.git
synced 2025-01-03 13:58:41 +00:00
Basic status creation and retrieval.
This commit is contained in:
parent
8de523c8ae
commit
9a8850eb9e
14 changed files with 272 additions and 9 deletions
|
@ -1,5 +1,6 @@
|
||||||
defmodule Pleroma.User do
|
defmodule Pleroma.User do
|
||||||
use Ecto.Schema
|
use Ecto.Schema
|
||||||
|
alias Pleroma.User
|
||||||
|
|
||||||
schema "users" do
|
schema "users" do
|
||||||
field :bio, :string
|
field :bio, :string
|
||||||
|
@ -7,7 +8,22 @@ defmodule Pleroma.User do
|
||||||
field :name, :string
|
field :name, :string
|
||||||
field :nickname, :string
|
field :nickname, :string
|
||||||
field :password_hash, :string
|
field :password_hash, :string
|
||||||
|
field :following, { :array, :string }, default: []
|
||||||
|
field :ap_id, :string
|
||||||
|
|
||||||
timestamps()
|
timestamps()
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def ap_id(%User{nickname: nickname}) do
|
||||||
|
host =
|
||||||
|
Application.get_env(:pleroma, Pleroma.Web.Endpoint)
|
||||||
|
|> Keyword.fetch!(:url)
|
||||||
|
|> Keyword.fetch!(:host)
|
||||||
|
|
||||||
|
"https://#{host}/users/#{nickname}"
|
||||||
|
end
|
||||||
|
|
||||||
|
def ap_followers(%User{} = user) do
|
||||||
|
"#{ap_id(user)}/followers"
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,8 +1,16 @@
|
||||||
defmodule Pleroma.Web.ActivityPub.ActivityPub do
|
defmodule Pleroma.Web.ActivityPub.ActivityPub do
|
||||||
alias Pleroma.Repo
|
alias Pleroma.Repo
|
||||||
alias Pleroma.Activity
|
alias Pleroma.Activity
|
||||||
|
import Ecto.Query
|
||||||
|
|
||||||
def insert(map) when is_map(map) do
|
def insert(map) when is_map(map) do
|
||||||
Repo.insert(%Activity{data: map})
|
Repo.insert(%Activity{data: map})
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def fetch_public_activities do
|
||||||
|
query = from activity in Activity,
|
||||||
|
where: fragment(~s(? @> '{"to": ["https://www.w3.org/ns/activitystreams#Public"]}'), activity.data)
|
||||||
|
|
||||||
|
Repo.all(query)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
defmodule Pleroma.Web.TwitterAPI.Representers.ActivityRepresenter do
|
||||||
|
use Pleroma.Web.TwitterAPI.Representers.BaseRepresenter
|
||||||
|
alias Pleroma.Web.TwitterAPI.Representers.UserRepresenter
|
||||||
|
|
||||||
|
def to_map(activity, %{user: user}) do
|
||||||
|
content = get_in(activity.data, ["object", "content"])
|
||||||
|
%{
|
||||||
|
"id" => activity.id,
|
||||||
|
"user" => UserRepresenter.to_map(user),
|
||||||
|
"attentions" => [],
|
||||||
|
"statusnet_html" => content,
|
||||||
|
"text" => content,
|
||||||
|
"is_local" => true,
|
||||||
|
"is_post_verb" => true
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
33
lib/pleroma/web/twitter_api/twitter_api.ex
Normal file
33
lib/pleroma/web/twitter_api/twitter_api.ex
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
defmodule Pleroma.Web.TwitterAPI.TwitterAPI do
|
||||||
|
alias Pleroma.User
|
||||||
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
|
alias Pleroma.Repo
|
||||||
|
alias Pleroma.Web.TwitterAPI.Representers.ActivityRepresenter
|
||||||
|
|
||||||
|
def create_status(user = %User{}, data = %{}) do
|
||||||
|
activity = %{
|
||||||
|
type: "Create",
|
||||||
|
to: [
|
||||||
|
User.ap_followers(user),
|
||||||
|
"https://www.w3.org/ns/activitystreams#Public"
|
||||||
|
],
|
||||||
|
actor: User.ap_id(user),
|
||||||
|
object: %{
|
||||||
|
type: "Note",
|
||||||
|
content: data.status
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ActivityPub.insert(activity)
|
||||||
|
end
|
||||||
|
|
||||||
|
def fetch_public_statuses do
|
||||||
|
activities = ActivityPub.fetch_public_activities
|
||||||
|
|
||||||
|
Enum.map(activities, fn(activity) ->
|
||||||
|
actor = get_in(activity.data, ["actor"])
|
||||||
|
user = Repo.get_by!(User, ap_id: actor)
|
||||||
|
ActivityRepresenter.to_map(activity, %{user: user})
|
||||||
|
end)
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,9 @@
|
||||||
|
defmodule Pleroma.Repo.Migrations.AddFollowingListToUsers do
|
||||||
|
use Ecto.Migration
|
||||||
|
|
||||||
|
def change do
|
||||||
|
alter table(:users) do
|
||||||
|
add :following, :map
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -0,0 +1,9 @@
|
||||||
|
defmodule Pleroma.Repo.Migrations.AddApIdToUsers do
|
||||||
|
use Ecto.Migration
|
||||||
|
|
||||||
|
def change do
|
||||||
|
alter table(:users) do
|
||||||
|
add :ap_id, :string
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
17
test/builders/user_builder.ex
Normal file
17
test/builders/user_builder.ex
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
defmodule Pleroma.Builders.UserBuilder do
|
||||||
|
alias Pleroma.{User, Repo}
|
||||||
|
|
||||||
|
def build do
|
||||||
|
%User{
|
||||||
|
email: "test@example.org",
|
||||||
|
name: "Test Name",
|
||||||
|
nickname: "testname",
|
||||||
|
password_hash: Comeonin.Pbkdf2.hashpwsalt("test"),
|
||||||
|
bio: "A tester.",
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def insert do
|
||||||
|
Repo.insert(build())
|
||||||
|
end
|
||||||
|
end
|
37
test/support/builders/activity_builder.ex
Normal file
37
test/support/builders/activity_builder.ex
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
defmodule Pleroma.Builders.ActivityBuilder do
|
||||||
|
alias Pleroma.Builders.UserBuilder
|
||||||
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
|
alias Pleroma.User
|
||||||
|
|
||||||
|
def public_and_non_public do
|
||||||
|
{:ok, user} = UserBuilder.insert
|
||||||
|
public = %{
|
||||||
|
"id" => 1,
|
||||||
|
"actor" => user.ap_id,
|
||||||
|
"to" => ["https://www.w3.org/ns/activitystreams#Public"],
|
||||||
|
"object" => %{
|
||||||
|
"type" => "Note",
|
||||||
|
"content" => "test"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
non_public = %{
|
||||||
|
"id" => 2,
|
||||||
|
"actor" => user.ap_id,
|
||||||
|
"to" => [],
|
||||||
|
"object" => %{
|
||||||
|
"type" => "Note",
|
||||||
|
"content" => "test"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{:ok, public} = ActivityPub.insert(public)
|
||||||
|
{:ok, non_public} = ActivityPub.insert(non_public)
|
||||||
|
|
||||||
|
%{
|
||||||
|
public: public,
|
||||||
|
non_public: non_public,
|
||||||
|
user: user
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
18
test/support/builders/user_builder.ex
Normal file
18
test/support/builders/user_builder.ex
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
defmodule Pleroma.Builders.UserBuilder do
|
||||||
|
alias Pleroma.{User, Repo}
|
||||||
|
|
||||||
|
def build do
|
||||||
|
%User{
|
||||||
|
email: "test@example.org",
|
||||||
|
name: "Test Name",
|
||||||
|
nickname: "testname",
|
||||||
|
password_hash: Comeonin.Pbkdf2.hashpwsalt("test"),
|
||||||
|
bio: "A tester.",
|
||||||
|
ap_id: "some id"
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def insert do
|
||||||
|
Repo.insert(build())
|
||||||
|
end
|
||||||
|
end
|
26
test/user_test.exs
Normal file
26
test/user_test.exs
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
defmodule Pleroma.UserTest do
|
||||||
|
alias Pleroma.Builders.UserBuilder
|
||||||
|
alias Pleroma.User
|
||||||
|
use Pleroma.DataCase
|
||||||
|
|
||||||
|
test "ap_id returns the activity pub id for the user" do
|
||||||
|
host =
|
||||||
|
Application.get_env(:pleroma, Pleroma.Web.Endpoint)
|
||||||
|
|> Keyword.fetch!(:url)
|
||||||
|
|> Keyword.fetch!(:host)
|
||||||
|
|
||||||
|
user = UserBuilder.build
|
||||||
|
|
||||||
|
expected_ap_id = "https://#{host}/users/#{user.nickname}"
|
||||||
|
|
||||||
|
assert expected_ap_id == User.ap_id(user)
|
||||||
|
end
|
||||||
|
|
||||||
|
test "ap_followers returns the followers collection for the user" do
|
||||||
|
user = UserBuilder.build
|
||||||
|
|
||||||
|
expected_followers_collection = "#{User.ap_id(user)}/followers"
|
||||||
|
|
||||||
|
assert expected_followers_collection == User.ap_followers(user)
|
||||||
|
end
|
||||||
|
end
|
|
@ -2,6 +2,7 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubTest do
|
||||||
use Pleroma.DataCase
|
use Pleroma.DataCase
|
||||||
alias Pleroma.Web.ActivityPub.ActivityPub
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
alias Pleroma.Activity
|
alias Pleroma.Activity
|
||||||
|
alias Pleroma.Builders.{UserBuilder, ActivityBuilder}
|
||||||
|
|
||||||
describe "insertion" do
|
describe "insertion" do
|
||||||
test "inserts a given map into the activity database" do
|
test "inserts a given map into the activity database" do
|
||||||
|
@ -14,4 +15,13 @@ defmodule Pleroma.Web.ActivityPub.ActivityPubTest do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe "fetch activities" do
|
||||||
|
test "retrieves all public activities" do
|
||||||
|
%{user: user, public: public} = ActivityBuilder.public_and_non_public
|
||||||
|
|
||||||
|
activities = ActivityPub.fetch_public_activities
|
||||||
|
assert length(activities) == 1
|
||||||
|
assert Enum.at(activities, 0) == public
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
defmodule Pleroma.Web.TwitterAPI.Representers.ActivityRepresenterTest do
|
||||||
|
use Pleroma.DataCase
|
||||||
|
alias Pleroma.{User, Activity}
|
||||||
|
alias Pleroma.Web.TwitterAPI.Representers.{UserRepresenter, ActivityRepresenter}
|
||||||
|
alias Pleroma.Builders.UserBuilder
|
||||||
|
|
||||||
|
test "an activity" do
|
||||||
|
{:ok, user} = UserBuilder.insert
|
||||||
|
content = "Some content"
|
||||||
|
activity = %Activity{
|
||||||
|
id: 1,
|
||||||
|
data: %{
|
||||||
|
"type" => "Create",
|
||||||
|
"to" => [
|
||||||
|
User.ap_followers(user),
|
||||||
|
"https://www.w3.org/ns/activitystreams#Public"
|
||||||
|
],
|
||||||
|
"actor" => User.ap_id(user),
|
||||||
|
"object" => %{
|
||||||
|
"type" => "Note",
|
||||||
|
"content" => content
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
expected_status = %{
|
||||||
|
"id" => activity.id,
|
||||||
|
"user" => UserRepresenter.to_map(user),
|
||||||
|
"is_local" => true,
|
||||||
|
"attentions" => [],
|
||||||
|
"statusnet_html" => content,
|
||||||
|
"text" => content,
|
||||||
|
"is_post_verb" => true
|
||||||
|
}
|
||||||
|
|
||||||
|
assert ActivityRepresenter.to_map(activity, %{user: user}) == expected_status
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,7 +1,7 @@
|
||||||
defmodule Pleroma.Web.TwitterAPI.ControllerTest do
|
defmodule Pleroma.Web.TwitterAPI.ControllerTest do
|
||||||
use Pleroma.Web.ConnCase
|
use Pleroma.Web.ConnCase
|
||||||
alias Pleroma.{User, Repo}
|
|
||||||
alias Pleroma.Web.TwitterAPI.Representers.UserRepresenter
|
alias Pleroma.Web.TwitterAPI.Representers.UserRepresenter
|
||||||
|
alias Pleroma.Builders.UserBuilder
|
||||||
|
|
||||||
describe "POST /api/account/verify_credentials" do
|
describe "POST /api/account/verify_credentials" do
|
||||||
setup [:valid_user]
|
setup [:valid_user]
|
||||||
|
@ -20,14 +20,7 @@ defmodule Pleroma.Web.TwitterAPI.ControllerTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
defp valid_user(_context) do
|
defp valid_user(_context) do
|
||||||
user = %User{
|
{ :ok, user } = UserBuilder.insert
|
||||||
email: "test@example.org",
|
|
||||||
name: "Test Name",
|
|
||||||
nickname: "testname",
|
|
||||||
password_hash: Comeonin.Pbkdf2.hashpwsalt("test"),
|
|
||||||
bio: "A tester."
|
|
||||||
}
|
|
||||||
user = Repo.insert!(user)
|
|
||||||
[user: user]
|
[user: user]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
31
test/web/twitter_api/twitter_api_test.exs
Normal file
31
test/web/twitter_api/twitter_api_test.exs
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
defmodule Pleroma.Web.TwitterAPI.TwitterAPITest do
|
||||||
|
use Pleroma.DataCase
|
||||||
|
alias Pleroma.Builders.{UserBuilder, ActivityBuilder}
|
||||||
|
alias Pleroma.Web.TwitterAPI.TwitterAPI
|
||||||
|
alias Pleroma.{Activity, User}
|
||||||
|
alias Pleroma.Web.TwitterAPI.Representers.{UserRepresenter, ActivityRepresenter}
|
||||||
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
|
|
||||||
|
test "create a status" do
|
||||||
|
user = UserBuilder.build
|
||||||
|
input = %{
|
||||||
|
status: "Hello again."
|
||||||
|
}
|
||||||
|
|
||||||
|
{ :ok, activity = %Activity{} } = TwitterAPI.create_status(user, input)
|
||||||
|
|
||||||
|
assert get_in(activity.data, [:object, :content]) == "Hello again."
|
||||||
|
assert get_in(activity.data, [:object, :type]) == "Note"
|
||||||
|
assert get_in(activity.data, [:actor]) == User.ap_id(user)
|
||||||
|
assert Enum.member?(get_in(activity.data, [:to]), User.ap_followers(user))
|
||||||
|
assert Enum.member?(get_in(activity.data, [:to]), "https://www.w3.org/ns/activitystreams#Public")
|
||||||
|
end
|
||||||
|
|
||||||
|
test "fetch public activities" do
|
||||||
|
%{ public: activity, user: user } = ActivityBuilder.public_and_non_public
|
||||||
|
statuses = TwitterAPI.fetch_public_statuses()
|
||||||
|
|
||||||
|
assert length(statuses) == 1
|
||||||
|
assert Enum.at(statuses, 0) == ActivityRepresenter.to_map(activity, %{user: user})
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue