mirror of
https://github.com/kieraneglin/pinchflat.git
synced 2026-01-23 02:24:24 +00:00
* Updated yt-dlp runner to take an action type * Added actions to all callers of the yt-dlp runner * [SQUASH] updated test files to use new mocking strategy * Removed unneeded alias
290 lines
9.5 KiB
Elixir
290 lines
9.5 KiB
Elixir
defmodule PinchflatWeb.SourceControllerTest do
|
|
use PinchflatWeb.ConnCase
|
|
|
|
import Pinchflat.MediaFixtures
|
|
import Pinchflat.SourcesFixtures
|
|
import Pinchflat.ProfilesFixtures
|
|
|
|
alias Pinchflat.Repo
|
|
alias Pinchflat.Settings
|
|
alias Pinchflat.Media.FileSyncingWorker
|
|
alias Pinchflat.Sources.SourceDeletionWorker
|
|
alias Pinchflat.Downloading.MediaDownloadWorker
|
|
alias Pinchflat.Metadata.SourceMetadataStorageWorker
|
|
alias Pinchflat.SlowIndexing.MediaCollectionIndexingWorker
|
|
|
|
setup do
|
|
media_profile = media_profile_fixture()
|
|
Settings.set(onboarding: false)
|
|
|
|
{
|
|
:ok,
|
|
%{
|
|
create_attrs: %{
|
|
media_profile_id: media_profile.id,
|
|
collection_type: "channel",
|
|
original_url: "https://www.youtube.com/source/abc123"
|
|
},
|
|
update_attrs: %{
|
|
original_url: "https://www.youtube.com/source/321xyz"
|
|
},
|
|
invalid_attrs: %{original_url: nil, media_profile_id: nil}
|
|
}
|
|
}
|
|
end
|
|
|
|
describe "index" do
|
|
# Most of the tests are in `index_table_list_test.exs`
|
|
test "returns 200", %{conn: conn} do
|
|
conn = get(conn, ~p"/sources")
|
|
assert html_response(conn, 200) =~ "Sources"
|
|
end
|
|
end
|
|
|
|
describe "new source" do
|
|
test "renders form", %{conn: conn} do
|
|
conn = get(conn, ~p"/sources/new")
|
|
assert html_response(conn, 200) =~ "New Source"
|
|
end
|
|
|
|
test "renders correct layout when onboarding", %{conn: conn} do
|
|
Settings.set(onboarding: true)
|
|
conn = get(conn, ~p"/sources/new")
|
|
|
|
refute html_response(conn, 200) =~ "MENU"
|
|
end
|
|
|
|
test "preloads some attributes when using a template", %{conn: conn} do
|
|
source = source_fixture(custom_name: "My first source", download_cutoff_date: "2021-01-01")
|
|
|
|
conn = get(conn, ~p"/sources/new", %{"template_id" => source.id})
|
|
assert html_response(conn, 200) =~ "New Source"
|
|
assert html_response(conn, 200) =~ "2021-01-01"
|
|
refute html_response(conn, 200) =~ source.custom_name
|
|
end
|
|
end
|
|
|
|
describe "create source" do
|
|
test "redirects to show when data is valid", %{conn: conn, create_attrs: create_attrs} do
|
|
expect(YtDlpRunnerMock, :run, 1, &runner_function_mock/5)
|
|
conn = post(conn, ~p"/sources", source: create_attrs)
|
|
|
|
assert %{id: id} = redirected_params(conn)
|
|
assert redirected_to(conn) == ~p"/sources/#{id}"
|
|
|
|
conn = get(conn, ~p"/sources/#{id}")
|
|
assert html_response(conn, 200) =~ "Source"
|
|
end
|
|
|
|
test "renders errors when data is invalid", %{conn: conn, invalid_attrs: invalid_attrs} do
|
|
conn = post(conn, ~p"/sources", source: invalid_attrs)
|
|
assert html_response(conn, 200) =~ "New Source"
|
|
end
|
|
|
|
test "redirects to onboarding when onboarding", %{conn: conn, create_attrs: create_attrs} do
|
|
expect(YtDlpRunnerMock, :run, 1, &runner_function_mock/5)
|
|
|
|
Settings.set(onboarding: true)
|
|
conn = post(conn, ~p"/sources", source: create_attrs)
|
|
|
|
assert redirected_to(conn) == ~p"/?onboarding=1"
|
|
end
|
|
|
|
test "renders correct layout on error when onboarding", %{conn: conn, invalid_attrs: invalid_attrs} do
|
|
Settings.set(onboarding: true)
|
|
conn = post(conn, ~p"/sources", source: invalid_attrs)
|
|
|
|
refute html_response(conn, 200) =~ "MENU"
|
|
end
|
|
end
|
|
|
|
describe "edit source" do
|
|
setup [:create_source]
|
|
|
|
test "renders form for editing chosen source", %{conn: conn, source: source} do
|
|
conn = get(conn, ~p"/sources/#{source}/edit")
|
|
assert html_response(conn, 200) =~ "Editing \"#{source.custom_name}\""
|
|
end
|
|
end
|
|
|
|
describe "update source" do
|
|
setup [:create_source]
|
|
|
|
test "redirects when data is valid", %{conn: conn, source: source, update_attrs: update_attrs} do
|
|
expect(YtDlpRunnerMock, :run, 1, &runner_function_mock/5)
|
|
|
|
conn = put(conn, ~p"/sources/#{source}", source: update_attrs)
|
|
assert redirected_to(conn) == ~p"/sources/#{source}"
|
|
|
|
conn = get(conn, ~p"/sources/#{source}")
|
|
assert html_response(conn, 200) =~ "https://www.youtube.com/source/321xyz"
|
|
end
|
|
|
|
test "renders errors when data is invalid", %{
|
|
conn: conn,
|
|
source: source,
|
|
invalid_attrs: invalid_attrs
|
|
} do
|
|
conn = put(conn, ~p"/sources/#{source}", source: invalid_attrs)
|
|
assert html_response(conn, 200) =~ "Editing \"#{source.custom_name}\""
|
|
end
|
|
end
|
|
|
|
describe "delete source in all cases" do
|
|
setup [:create_source]
|
|
|
|
test "redirects to the sources page", %{conn: conn, source: source} do
|
|
conn = delete(conn, ~p"/sources/#{source}")
|
|
assert redirected_to(conn) == ~p"/sources"
|
|
end
|
|
|
|
test "sets marked_for_deletion_at", %{conn: conn, source: source} do
|
|
delete(conn, ~p"/sources/#{source}")
|
|
assert Repo.reload!(source).marked_for_deletion_at
|
|
end
|
|
end
|
|
|
|
describe "delete source when just deleting the records" do
|
|
setup [:create_source]
|
|
|
|
test "enqueues a job without the delete_files arg", %{conn: conn, source: source} do
|
|
delete(conn, ~p"/sources/#{source}")
|
|
|
|
assert [%{args: %{"delete_files" => false}}] = all_enqueued(worker: SourceDeletionWorker)
|
|
end
|
|
end
|
|
|
|
describe "delete source when deleting the records and files" do
|
|
setup [:create_source]
|
|
|
|
test "enqueues a job without the delete_files arg", %{conn: conn, source: source} do
|
|
delete(conn, ~p"/sources/#{source}?delete_files=true")
|
|
|
|
assert [%{args: %{"delete_files" => true}}] = all_enqueued(worker: SourceDeletionWorker)
|
|
end
|
|
end
|
|
|
|
describe "force_download_pending" do
|
|
test "enqueues pending download tasks", %{conn: conn} do
|
|
source = source_fixture()
|
|
_media_item = media_item_fixture(%{source_id: source.id, media_filepath: nil})
|
|
|
|
assert [] = all_enqueued(worker: MediaDownloadWorker)
|
|
post(conn, ~p"/sources/#{source.id}/force_download_pending")
|
|
assert [_] = all_enqueued(worker: MediaDownloadWorker)
|
|
end
|
|
|
|
test "redirects to the source page", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
conn = post(conn, ~p"/sources/#{source.id}/force_download_pending")
|
|
assert redirected_to(conn) == ~p"/sources/#{source.id}"
|
|
end
|
|
end
|
|
|
|
describe "force_redownload" do
|
|
test "enqueues re-download tasks", %{conn: conn} do
|
|
source = source_fixture()
|
|
_media_item = media_item_fixture(source_id: source.id, media_downloaded_at: now())
|
|
|
|
assert [] = all_enqueued(worker: MediaDownloadWorker)
|
|
post(conn, ~p"/sources/#{source.id}/force_redownload")
|
|
assert [_] = all_enqueued(worker: MediaDownloadWorker)
|
|
end
|
|
|
|
test "redirects to the source page", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
conn = post(conn, ~p"/sources/#{source.id}/force_redownload")
|
|
assert redirected_to(conn) == ~p"/sources/#{source.id}"
|
|
end
|
|
end
|
|
|
|
describe "force_index" do
|
|
test "forces an index", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
assert [] = all_enqueued(worker: MediaCollectionIndexingWorker)
|
|
post(conn, ~p"/sources/#{source.id}/force_index")
|
|
assert [_] = all_enqueued(worker: MediaCollectionIndexingWorker)
|
|
end
|
|
|
|
test "forces an index even if one wouldn't normally run", %{conn: conn} do
|
|
source = source_fixture(index_frequency_minutes: 0, last_indexed_at: DateTime.utc_now())
|
|
|
|
post(conn, ~p"/sources/#{source.id}/force_index")
|
|
assert [job] = all_enqueued(worker: MediaCollectionIndexingWorker)
|
|
assert job.args == %{"id" => source.id, "force" => true}
|
|
end
|
|
|
|
test "deletes pending indexing tasks", %{conn: conn} do
|
|
source = source_fixture()
|
|
{:ok, task} = MediaCollectionIndexingWorker.kickoff_with_task(source)
|
|
job = Repo.preload(task, :job).job
|
|
|
|
assert job.state == "available"
|
|
post(conn, ~p"/sources/#{source.id}/force_index")
|
|
assert Repo.reload!(job).state == "cancelled"
|
|
end
|
|
|
|
test "redirects to the source page", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
conn = post(conn, ~p"/sources/#{source.id}/force_index")
|
|
assert redirected_to(conn) == ~p"/sources/#{source.id}"
|
|
end
|
|
end
|
|
|
|
describe "force_metadata_refresh" do
|
|
test "forces a metadata refresh", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
assert [] = all_enqueued(worker: SourceMetadataStorageWorker)
|
|
post(conn, ~p"/sources/#{source.id}/force_metadata_refresh")
|
|
assert [_] = all_enqueued(worker: SourceMetadataStorageWorker)
|
|
end
|
|
|
|
test "redirects to the source page", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
conn = post(conn, ~p"/sources/#{source.id}/force_metadata_refresh")
|
|
assert redirected_to(conn) == ~p"/sources/#{source.id}"
|
|
end
|
|
end
|
|
|
|
describe "sync_files_on_disk" do
|
|
test "forces a file sync", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
assert [] = all_enqueued(worker: FileSyncingWorker)
|
|
post(conn, ~p"/sources/#{source.id}/sync_files_on_disk")
|
|
assert [_] = all_enqueued(worker: FileSyncingWorker)
|
|
end
|
|
|
|
test "redirects to the source page", %{conn: conn} do
|
|
source = source_fixture()
|
|
|
|
conn = post(conn, ~p"/sources/#{source.id}/sync_files_on_disk")
|
|
assert redirected_to(conn) == ~p"/sources/#{source.id}"
|
|
end
|
|
end
|
|
|
|
defp create_source(_) do
|
|
source = source_fixture()
|
|
media_item = media_item_with_attachments(%{source_id: source.id})
|
|
|
|
%{source: source, media_item: media_item}
|
|
end
|
|
|
|
defp runner_function_mock(_url, :get_source_details, _opts, _ot, _addl) do
|
|
{
|
|
:ok,
|
|
Phoenix.json_library().encode!(%{
|
|
channel: "some channel name",
|
|
channel_id: "some_channel_id_#{:rand.uniform(1_000_000)}",
|
|
playlist_id: "some_playlist_id_#{:rand.uniform(1_000_000)}",
|
|
playlist_title: "some playlist name"
|
|
})
|
|
}
|
|
end
|
|
end
|