From 62454de4b9da91a8ca5f77c7d48e6f3077b53772 Mon Sep 17 00:00:00 2001 From: taylordowns2000 Date: Mon, 31 Jul 2023 10:56:46 +0000 Subject: [PATCH] =?UTF-8?q?Deploying=20to=20gh-pages=20from=20@=20OpenFn/L?= =?UTF-8?q?ightning@8f679daa891be473b8f4e9385ba035190a73cab1=20?= =?UTF-8?q?=F0=9F=9A=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .build | 2 +- Lightning.Accounts.html | 126 +++++++++--------- Lightning.AdaptorRegistry.Npm.html | 10 +- Lightning.AdaptorRegistry.html | 12 +- Lightning.AttemptService.html | 8 +- Lightning.AuthProviders.WellKnown.html | 10 +- Lightning.CLI.Result.html | 2 +- Lightning.Credentials.html | 50 +++---- Lightning.Invocation.html | 86 ++++++------ Lightning.InvocationReasons.html | 8 +- Lightning.Jobs.Job.html | 18 +-- Lightning.Jobs.html | 34 ++--- Lightning.Pipeline.StateAssembler.html | 2 +- Lightning.Policies.Permissions.html | 18 +-- Lightning.Projects.html | 74 +++++----- Lightning.Runtime.LogAgent.html | 6 +- Lightning.Scrubber.html | 8 +- Lightning.TaskWorker.html | 2 +- Lightning.Validators.html | 6 +- Lightning.WorkOrderService.html | 8 +- Lightning.Workflows.html | 42 +++--- Lightning.epub | Bin 350241 -> 350322 bytes LightningWeb.Gettext.html | 8 +- LightningWeb.OauthCredentialHelper.html | 8 +- LightningWeb.Pagination.html | 10 +- Mix.Tasks.Lightning.InstallSchemas.html | 10 +- changelog.html | 2 +- ...s-C2951775.js => search_items-01F3CC93.js} | 2 +- readme.html | 8 +- search.html | 2 +- 30 files changed, 291 insertions(+), 291 deletions(-) rename dist/{search_items-C2951775.js => search_items-01F3CC93.js} (93%) diff --git a/.build b/.build index 26d8e89fc3..366bdaac12 100644 --- a/.build +++ b/.build @@ -195,7 +195,7 @@ dist/merriweather-latin-ext-300-normal-K6L27CZ5.woff2 dist/merriweather-vietnamese-300-italic-EHHNZPUO.woff2 dist/merriweather-vietnamese-300-normal-U376L4Z4.woff2 dist/remixicon-NKANDIL5.woff2 -dist/search_items-C2951775.js +dist/search_items-01F3CC93.js dist/sidebar_items-08FEF45F.js index.html provisioning.html diff --git a/Lightning.Accounts.html b/Lightning.Accounts.html index 38ca93dba1..60cedb1e00 100644 --- a/Lightning.Accounts.html +++ b/Lightning.Accounts.html @@ -642,10 +642,10 @@

apply_user_email(user, password, attrs)

Examples -
iex> apply_user_email(user, "valid password", %{email: ...})
-{:ok, %User{}}role: :superuser
-iex> apply_user_email(user, "invalid password", %{email: ...})
-{:error, %Ecto.Changeset{}}
+
iex> apply_user_email(user, "valid password", %{email: ...})
+{:ok, %User{}}role: :superuser
+iex> apply_user_email(user, "invalid password", %{email: ...})
+{:error, %Ecto.Changeset{}}
@@ -697,8 +697,8 @@

change_scheduled_deletion(user, attrs \\ %{ Examples

-
iex> change_scheduled_deletion(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_scheduled_deletion(user)
+%Ecto.Changeset{data: %User{}}
@@ -734,8 +734,8 @@

change_superuser_registration(attrs \\ %{}) Examples

-
iex> change_superuser_registration(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_superuser_registration(user)
+%Ecto.Changeset{data: %User{}}
@@ -789,8 +789,8 @@

change_user_email(user, attrs \\ %{})

Examples -
iex> change_user_email(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_user_email(user)
+%Ecto.Changeset{data: %User{}}
@@ -820,8 +820,8 @@

change_user_password(user, attrs \\ %{}) Examples

-
iex> change_user_password(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_user_password(user)
+%Ecto.Changeset{data: %User{}}
@@ -851,8 +851,8 @@

change_user_registration(attrs \\ %{})

Examples -
iex> change_user_registration(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_user_registration(user)
+%Ecto.Changeset{data: %User{}}
@@ -947,11 +947,11 @@

delete_token(token)

Examples -
iex> delete_token(token)
-{:ok, %UserToken{}}
+
iex> delete_token(token)
+{:ok, %UserToken{}}
 
-iex> delete_token(token)
-{:error, %Ecto.Changeset{}}
+
iex> delete_token(token) +{:error, %Ecto.Changeset{}}
@@ -979,11 +979,11 @@

delete_user(user)

Examples -
iex> delete_user(user)
-{:ok, %User{}}
+
iex> delete_user(user)
+{:ok, %User{}}
 
-iex> delete_user(user)
-{:error, %Ecto.Changeset{}}
+
iex> delete_user(user) +{:error, %Ecto.Changeset{}}
@@ -1040,8 +1040,8 @@

deliver_update_email_instructions(user, cur Examples

-
iex> deliver_update_email_instructions(user, current_email, &Routes.user_update_email_url(conn, :edit, &1))
-{:ok, %{to: ..., body: ...}}
+
iex> deliver_update_email_instructions(user, current_email, &Routes.user_update_email_url(conn, :edit, &1))
+{:ok, %{to: ..., body: ...}}
@@ -1069,11 +1069,11 @@

deliver_user_confirmation_instructions(user Examples

-
iex> deliver_user_confirmation_instructions(user, &Routes.user_confirmation_url(conn, :edit, &1))
-{:ok, %{to: ..., body: ...}}
+
iex> deliver_user_confirmation_instructions(user, &Routes.user_confirmation_url(conn, :edit, &1))
+{:ok, %{to: ..., body: ...}}
 
-iex> deliver_user_confirmation_instructions(confirmed_user, &Routes.user_confirmation_url(conn, :edit, &1))
-{:error, :already_confirmed}
+
iex> deliver_user_confirmation_instructions(confirmed_user, &Routes.user_confirmation_url(conn, :edit, &1)) +{:error, :already_confirmed}
@@ -1123,8 +1123,8 @@

deliver_user_reset_password_instructions(us Examples

-
iex> deliver_user_reset_password_instructions(user, &Routes.user_reset_password_url(conn, :edit, &1))
-{:ok, %{to: ..., body: ...}}
+
iex> deliver_user_reset_password_instructions(user, &Routes.user_reset_password_url(conn, :edit, &1))
+{:ok, %{to: ..., body: ...}}
@@ -1240,10 +1240,10 @@

get_token!(id)

Examples -
iex> get_token!(123)
-%UserToken{}
+
iex> get_token!(123)
+%UserToken{}
 
-iex> get_token!(456)
+iex> get_token!(456)
 ** (Ecto.NoResultsError)
@@ -1272,10 +1272,10 @@

get_user!(id)

Examples -
iex> get_user!(123)
-%User{}
+
iex> get_user!(123)
+%User{}
 
-iex> get_user!(456)
+iex> get_user!(456)
 ** (Ecto.NoResultsError)
@@ -1348,10 +1348,10 @@

get_user_by_email(email)

Examples -
iex> get_user_by_email("foo@example.com")
-%User{}
+
iex> get_user_by_email("foo@example.com")
+%User{}
 
-iex> get_user_by_email("unknown@example.com")
+iex> get_user_by_email("unknown@example.com")
 nil
@@ -1380,10 +1380,10 @@

get_user_by_email_and_password(email, passw Examples

-
iex> get_user_by_email_and_password("foo@example.com", "correct_password")
-%User{}
+
iex> get_user_by_email_and_password("foo@example.com", "correct_password")
+%User{}
 
-iex> get_user_by_email_and_password("foo@example.com", "invalid_password")
+iex> get_user_by_email_and_password("foo@example.com", "invalid_password")
 nil
@@ -1412,10 +1412,10 @@

get_user_by_reset_password_token(token)

Examples -
iex> get_user_by_reset_password_token("validtoken")
-%User{}
+
iex> get_user_by_reset_password_token("validtoken")
+%User{}
 
-iex> get_user_by_reset_password_token("invalidtoken")
+iex> get_user_by_reset_password_token("invalidtoken")
 nil
@@ -1583,8 +1583,8 @@

list_users()

Examples -
iex> list_users()
-[%User{}, ...]
+
iex> list_users()
+[%User{}, ...]
@@ -1662,11 +1662,11 @@

register_superuser(attrs)

Examples -
iex> register_superuser(%{field: value})
-{:ok, %User{}}
+
iex> register_superuser(%{field: value})
+{:ok, %User{}}
 
-iex> register_superuser(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> register_superuser(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -1704,11 +1704,11 @@

register_user(attrs)

Examples -
iex> register_user(%{field: value})
-{:ok, %User{}}
+
iex> register_user(%{field: value})
+{:ok, %User{}}
 
-iex> register_user(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> register_user(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -1736,11 +1736,11 @@

reset_user_password(user, attrs)

Examples -
iex> reset_user_password(user, %{password: "new long password", password_confirmation: "new long password"})
-{:ok, %User{}}
+
iex> reset_user_password(user, %{password: "new long password", password_confirmation: "new long password"})
+{:ok, %User{}}
 
-iex> reset_user_password(user, %{password: "valid", password_confirmation: "not the same"})
-{:error, %Ecto.Changeset{}}
+
iex> reset_user_password(user, %{password: "valid", password_confirmation: "not the same"}) +{:error, %Ecto.Changeset{}}
@@ -1840,11 +1840,11 @@

update_user_password(user, password, attrs) Examples

-
iex> update_user_password(user, "valid password", %{password: ...})
-{:ok, %User{}}
+
iex> update_user_password(user, "valid password", %{password: ...})
+{:ok, %User{}}
 
-iex> update_user_password(user, "invalid password", %{password: ...})
-{:error, %Ecto.Changeset{}}
+
iex> update_user_password(user, "invalid password", %{password: ...}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.AdaptorRegistry.Npm.html b/Lightning.AdaptorRegistry.Npm.html index a91dcf7c3a..3951b1dafd 100644 --- a/Lightning.AdaptorRegistry.Npm.html +++ b/Lightning.AdaptorRegistry.Npm.html @@ -1659,14 +1659,14 @@

request(request)

Examples -
request = %HTTPoison.Request{
+
request = %HTTPoison.Request{
   method: :post,
   url: "https://my.website.com",
   body: "{\"foo\": 3}",
-  headers: [{"Accept", "application/json"}]
-}
+  headers: [{"Accept", "application/json"}]
+}
 
-request(request)
+
request(request)
@@ -1724,7 +1724,7 @@

request(method, url, body \\ "", Examples

-
request(:post, "https://my.website.com", "{\"foo\": 3}", [{"Accept", "application/json"}])
+
request(:post, "https://my.website.com", "{\"foo\": 3}", [{"Accept", "application/json"}])
diff --git a/Lightning.AdaptorRegistry.html b/Lightning.AdaptorRegistry.html index 61e9b0f900..2232a04ad3 100644 --- a/Lightning.AdaptorRegistry.html +++ b/Lightning.AdaptorRegistry.html @@ -117,9 +117,9 @@

Registry process to query and maintain a list of adaptors available for writing jobs.

Currently it queries NPM for all modules in the @openfn organization and filters out modules that are known not to be adaptors.

Usage

# Starting the process
-AdaptorRegistry.start_link()
+AdaptorRegistry.start_link()
 # Getting a list of all adaptors
-Lightning.AdaptorRegistry.AdaptorRegistry.all()

Caching

By default the results are cached to disk, and will be reused every start.

In order to disable or configure caching pass see: start_link/1.

The process uses :continue to return before the adaptors have been queried. +Lightning.AdaptorRegistry.AdaptorRegistry.all()

Caching

By default the results are cached to disk, and will be reused every start.

In order to disable or configure caching pass see: start_link/1.

The process uses :continue to return before the adaptors have been queried. This does mean that the first call to the process will be delayed until the handle_continue/2 has finished.

Timeouts

There is a 'general' timeout of 30s, this is used for GenServer calls like all/1 and also internally when the modules are being queried. NPM can @@ -411,10 +411,10 @@

resolve_package_name(package_name)

-

Destructures an NPM style package name into module name and version.

Example

iex> resolve_package_name("@openfn/language-salesforce@1.2.3")
-{ "@openfn/language-salesforce", "1.2.3" }
-iex> resolve_package_name("@openfn/language-salesforce")
-{ "@openfn/language-salesforce", nil }
+

Destructures an NPM style package name into module name and version.

Example

iex> resolve_package_name("@openfn/language-salesforce@1.2.3")
+{ "@openfn/language-salesforce", "1.2.3" }
+iex> resolve_package_name("@openfn/language-salesforce")
+{ "@openfn/language-salesforce", nil }
diff --git a/Lightning.AttemptService.html b/Lightning.AttemptService.html index 946aee77bc..e93b5812d1 100644 --- a/Lightning.AttemptService.html +++ b/Lightning.AttemptService.html @@ -354,11 +354,11 @@

create_attempt(work_order, job, reason)

Examples -
iex> create_attempt(%{field: value})
-{:ok, %Attempt{}}
+
iex> create_attempt(%{field: value})
+{:ok, %Attempt{}}
 
-iex> create_attempt(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_attempt(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.AuthProviders.WellKnown.html b/Lightning.AuthProviders.WellKnown.html index 630bf7fd4d..2da5311227 100644 --- a/Lightning.AuthProviders.WellKnown.html +++ b/Lightning.AuthProviders.WellKnown.html @@ -1834,14 +1834,14 @@

request(request)

Examples -
request = %HTTPoison.Request{
+
request = %HTTPoison.Request{
   method: :post,
   url: "https://my.website.com",
   body: "{\"foo\": 3}",
-  headers: [{"Accept", "application/json"}]
-}
+  headers: [{"Accept", "application/json"}]
+}
 
-request(request)
+
request(request)
@@ -1899,7 +1899,7 @@

request(method, url, body \\ "", Examples

-
request(:post, "https://my.website.com", "{\"foo\": 3}", [{"Accept", "application/json"}])
+
request(:post, "https://my.website.com", "{\"foo\": 3}", [{"Accept", "application/json"}])
diff --git a/Lightning.CLI.Result.html b/Lightning.CLI.Result.html index 6199064554..8396b2530b 100644 --- a/Lightning.CLI.Result.html +++ b/Lightning.CLI.Result.html @@ -121,7 +121,7 @@

Logs

The OpenFn CLI returns JSON formatted log lines, which are decoded and added -to a Result struct.

There are two kinds of output:

{"level":"<<level>>","name":"<<module>>","message":"..."],"time":<<timestamp>>}

These are usually for general logging, and debugging.

{"message":["<<message|filepath|output>>"]}

The above is the equivalent of the output of a command

+to a Result struct.

There are two kinds of output:

{"level":"<<level>>","name":"<<module>>","message":"..."],"time":<<timestamp>>}

These are usually for general logging, and debugging.

{"message":["<<message|filepath|output>>"]}

The above is the equivalent of the output of a command

diff --git a/Lightning.Credentials.html b/Lightning.Credentials.html index 0afd7fb50f..28ba5b0d44 100644 --- a/Lightning.Credentials.html +++ b/Lightning.Credentials.html @@ -277,8 +277,8 @@

change_credential(credential, attrs \\ %{}) Examples

-
iex> change_credential(credential)
-%Ecto.Changeset{data: %Credential{}}
+
iex> change_credential(credential)
+%Ecto.Changeset{data: %Credential{}}
@@ -308,11 +308,11 @@

create_credential(attrs \\ %{})

Examples -
iex> create_credential(%{field: value})
-{:ok, %Credential{}}
+
iex> create_credential(%{field: value})
+{:ok, %Credential{}}
 
-iex> create_credential(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_credential(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -340,11 +340,11 @@

delete_credential(credential)

Examples -
iex> delete_credential(credential)
-{:ok, %Credential{}}
+
iex> delete_credential(credential)
+{:ok, %Credential{}}
 
-iex> delete_credential(credential)
-{:error, %Ecto.Changeset{}}
+
iex> delete_credential(credential) +{:error, %Ecto.Changeset{}}
@@ -372,10 +372,10 @@

get_credential!(id)

Examples -
iex> get_credential!(123)
-%Credential{}
+
iex> get_credential!(123)
+%Credential{}
 
-iex> get_credential!(456)
+iex> get_credential!(456)
 ** (Ecto.NoResultsError)
@@ -406,11 +406,11 @@

invalid_projects_for_user(credential_id, us Examples

-
iex> can_credential_be_shared_to_user(credential_id, user_id)
-[]
+
iex> can_credential_be_shared_to_user(credential_id, user_id)
+[]
 
-iex> can_credential_be_shared_to_user(credential_id, user_id)
-["52ea8758-6ce5-43d7-912f-6a1e1f11dc55"]
+
iex> can_credential_be_shared_to_user(credential_id, user_id) +["52ea8758-6ce5-43d7-912f-6a1e1f11dc55"]
@@ -438,8 +438,8 @@

list_credentials()

Examples -
iex> list_credentials()
-[%Credential{}, ...]
+
iex> list_credentials()
+[%Credential{}, ...]
@@ -489,8 +489,8 @@

list_credentials_for_user(user_id)

Examples -
iex> list_credentials_for_user(123)
-[%Credential{user_id: 123}, %Credential{user_id: 123},...]
+
iex> list_credentials_for_user(123)
+[%Credential{user_id: 123}, %Credential{user_id: 123},...]
@@ -576,11 +576,11 @@

update_credential(credential, attrs)

Examples -
iex> update_credential(credential, %{field: new_value})
-{:ok, %Credential{}}
+
iex> update_credential(credential, %{field: new_value})
+{:ok, %Credential{}}
 
-iex> update_credential(credential, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_credential(credential, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.Invocation.html b/Lightning.Invocation.html index 9a05017054..c04bc25f75 100644 --- a/Lightning.Invocation.html +++ b/Lightning.Invocation.html @@ -476,8 +476,8 @@

change_dataclip(dataclip, attrs \\ %{})

Examples -
iex> change_dataclip(dataclip)
-%Ecto.Changeset{data: %Dataclip{}}
+
iex> change_dataclip(dataclip)
+%Ecto.Changeset{data: %Dataclip{}}
@@ -507,8 +507,8 @@

change_run(run, attrs \\ %{})

Examples -
iex> change_run(run)
-%Ecto.Changeset{data: %Run{}}
+
iex> change_run(run)
+%Ecto.Changeset{data: %Run{}}
@@ -546,11 +546,11 @@

create_dataclip(attrs \\ %{})

Examples -
iex> create_dataclip(%{field: value})
-{:ok, %Dataclip{}}
+
iex> create_dataclip(%{field: value})
+{:ok, %Dataclip{}}
 
-iex> create_dataclip(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_dataclip(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -602,11 +602,11 @@

create_run(attrs \\ %{})

Examples -
iex> create_run(%{field: value})
-{:ok, %Run{}}
+
iex> create_run(%{field: value})
+{:ok, %Run{}}
 
-iex> create_run(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_run(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -634,11 +634,11 @@

delete_dataclip(dataclip)

Examples -
iex> delete_dataclip(dataclip)
-{:ok, %Dataclip{}}
+
iex> delete_dataclip(dataclip)
+{:ok, %Dataclip{}}
 
-iex> delete_dataclip(dataclip)
-{:error, %Ecto.Changeset{}}
+
iex> delete_dataclip(dataclip) +{:error, %Ecto.Changeset{}}
@@ -666,11 +666,11 @@

delete_run(run)

Examples -
iex> delete_run(run)
-{:ok, %Run{}}
+
iex> delete_run(run)
+{:ok, %Run{}}
 
-iex> delete_run(run)
-{:error, %Ecto.Changeset{}}
+
iex> delete_run(run) +{:error, %Ecto.Changeset{}}
@@ -859,14 +859,14 @@

get_dataclip(run)

Examples -
iex> get_dataclip("27b73932-16c7-4a72-86a3-85d805ccff98")
-%Dataclip{}
+
iex> get_dataclip("27b73932-16c7-4a72-86a3-85d805ccff98")
+%Dataclip{}
 
-iex> get_dataclip("27b73932-16c7-4a72-86a3-85d805ccff98")
+iex> get_dataclip("27b73932-16c7-4a72-86a3-85d805ccff98")
 nil
 
-iex> get_dataclip(%Run{id: "a uuid"})
-%Dataclip{}
+
iex> get_dataclip(%Run{id: "a uuid"}) +%Dataclip{}
@@ -900,10 +900,10 @@

get_dataclip!(id)

Examples -
iex> get_dataclip!(123)
-%Dataclip{}
+
iex> get_dataclip!(123)
+%Dataclip{}
 
-iex> get_dataclip!(456)
+iex> get_dataclip!(456)
 ** (Ecto.NoResultsError)
@@ -982,10 +982,10 @@

get_run!(id)

Examples -
iex> get_run!(123)
-%Run{}
+
iex> get_run!(123)
+%Run{}
 
-iex> get_run!(456)
+iex> get_run!(456)
 ** (Ecto.NoResultsError)
@@ -1064,8 +1064,8 @@

list_dataclips()

Examples -
iex> list_dataclips()
-[%Dataclip{}, ...]
+
iex> list_dataclips()
+[%Dataclip{}, ...]
@@ -1174,8 +1174,8 @@

list_runs()

Examples -
iex> list_runs()
-[%Run{}, ...]
+
iex> list_runs()
+[%Run{}, ...]
@@ -1330,11 +1330,11 @@

update_dataclip(dataclip, attrs)

Examples -
iex> update_dataclip(dataclip, %{field: new_value})
-{:ok, %Dataclip{}}
+
iex> update_dataclip(dataclip, %{field: new_value})
+{:ok, %Dataclip{}}
 
-iex> update_dataclip(dataclip, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_dataclip(dataclip, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -1362,11 +1362,11 @@

update_run(run, attrs)

Examples -
iex> update_run(run, %{field: new_value})
-{:ok, %Run{}}
+
iex> update_run(run, %{field: new_value})
+{:ok, %Run{}}
 
-iex> update_run(run, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_run(run, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.InvocationReasons.html b/Lightning.InvocationReasons.html index cb4c9fadfa..274bc9f5e2 100644 --- a/Lightning.InvocationReasons.html +++ b/Lightning.InvocationReasons.html @@ -273,11 +273,11 @@

create_reason(attrs \\ %{})

Examples -
iex> create_reason(%{field: value})
-{:ok, %InvocationReason{}}
+
iex> create_reason(%{field: value})
+{:ok, %InvocationReason{}}
 
-iex> create_reason(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_reason(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.Jobs.Job.html b/Lightning.Jobs.Job.html index b6381cc257..93ae641f49 100644 --- a/Lightning.Jobs.Job.html +++ b/Lightning.Jobs.Job.html @@ -336,17 +336,17 @@

put_workflow(changeset, workflow)

Attaches a workflow to a job, this is useful when you have an unpersisted Workflow changeset - and want it to be created at the same time as a Job.

Example:

workflow =
-  Ecto.Changeset.cast(
-    %Lightning.Workflows.Workflow{},
-    %{ "project_id" => attrs[:project_id], "id" => Ecto.UUID.generate() },
-    [:project_id, :id]
-  )
+  Ecto.Changeset.cast(
+    %Lightning.Workflows.Workflow{},
+    %{ "project_id" => attrs[:project_id], "id" => Ecto.UUID.generate() },
+    [:project_id, :id]
+  )
 
 job =
-  %Job{}
-  |> Ecto.Changeset.change()
-  |> Job.put_workflow(workflow)
-  |> Job.changeset(attrs)
+
%Job{} + |> Ecto.Changeset.change() + |> Job.put_workflow(workflow) + |> Job.changeset(attrs)
diff --git a/Lightning.Jobs.html b/Lightning.Jobs.html index 3b5b6c8d86..95d6c74b10 100644 --- a/Lightning.Jobs.html +++ b/Lightning.Jobs.html @@ -295,8 +295,8 @@

change_job(job, attrs \\ %{})

Examples -
iex> change_job(job)
-%Ecto.Changeset{data: %Job{}}
+
iex> change_job(job)
+%Ecto.Changeset{data: %Job{}}
@@ -326,11 +326,11 @@

create_job(attrs \\ %{})

Examples -
iex> create_job(%{field: value})
-{:ok, %Job{}}
+
iex> create_job(%{field: value})
+{:ok, %Job{}}
 
-iex> create_job(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_job(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -358,11 +358,11 @@

delete_job(job)

Examples -
iex> delete_job(job)
-{:ok, %Job{}}
+
iex> delete_job(job)
+{:ok, %Job{}}
 
-iex> delete_job(job)
-{:error, %Ecto.Changeset{}}
+
iex> delete_job(job) +{:error, %Ecto.Changeset{}}
@@ -448,10 +448,10 @@

get_job!(id)

Examples -
iex> get_job!(123)
-%Job{}
+
iex> get_job!(123)
+%Job{}
 
-iex> get_job!(456)
+iex> get_job!(456)
 ** (Ecto.NoResultsError)
@@ -637,11 +637,11 @@

update_job(job, attrs)

Examples -
iex> update_job(job, %{field: new_value})
-{:ok, %Job{}}
+
iex> update_job(job, %{field: new_value})
+{:ok, %Job{}}
 
-iex> update_job(job, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_job(job, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.Pipeline.StateAssembler.html b/Lightning.Pipeline.StateAssembler.html index fe1e083b9a..ed155164ec 100644 --- a/Lightning.Pipeline.StateAssembler.html +++ b/Lightning.Pipeline.StateAssembler.html @@ -121,7 +121,7 @@

How state is assembled

For the most common jobs, an inbound webhook will store an :http_request type -dataclip. The reason that is created is associated with the dataclip.

At runtime, the initial state for a Run will be in the shape of:

{ "data": <the dataclip>, "configuration": <the job's credential> }

+dataclip. The reason that is created is associated with the dataclip.

At runtime, the initial state for a Run will be in the shape of:

{ "data": <the dataclip>, "configuration": <the job's credential> }

saved-inputs

diff --git a/Lightning.Policies.Permissions.html b/Lightning.Policies.Permissions.html index 0d272cecf7..5336197548 100644 --- a/Lightning.Policies.Permissions.html +++ b/Lightning.Policies.Permissions.html @@ -115,13 +115,13 @@

This module defines a unique interface managing authorizations in Lightning.

Users in Lightning have instance-wide and project-wide roles which determine their level of access to resources in the application. Fo rmore details see the documentation.

These authorizations policies are all implemented under the lib/lightning/policies folder. In that folder you can find 3 files:

  • The users.ex file has all the policies for the instances wide access levels
  • The project_users.ex file has all the policies for the project wide access levels
  • The permissions.ex file defines the Lightning.Policies.Permissions.can/4 interface. Which is a wrapper around the Bodyguard.permit/4 function. -We use that interface to be able to harmonize the use of policies accross the entire app.

All the policies are tested in the test/lightning/policies folder. And the test are written in a way that allows the reader to quickly who can do what in the app.

We have two variants of the Lightning.Policies.Permissions.can/4 interface:

  • Lightning.Policies.Permissions.can(policy, action, actor, resource) returns :ok if the actor can perform the action on the resource and {:error, :unauthorized} otherwise.
  • Lightning.Policies.Permissions.can?(policy, action, actor, resource) returns true if the actor can perform the action on the resource and false otherwise.

Here is an example of how we the Lightning.Policies.Permissions.can/4 interface to check if the a user can edit a job or not

can_edit_job = Lightning.Policies.ProjectUsers |> Lightning.Policies.Permissions.can?(:edit_job, socket.assigns.current_user, socket.assigns.project)
+We use that interface to be able to harmonize the use of policies accross the entire app.

All the policies are tested in the test/lightning/policies folder. And the test are written in a way that allows the reader to quickly who can do what in the app.

We have two variants of the Lightning.Policies.Permissions.can/4 interface:

  • Lightning.Policies.Permissions.can(policy, action, actor, resource) returns :ok if the actor can perform the action on the resource and {:error, :unauthorized} otherwise.
  • Lightning.Policies.Permissions.can?(policy, action, actor, resource) returns true if the actor can perform the action on the resource and false otherwise.

Here is an example of how we the Lightning.Policies.Permissions.can/4 interface to check if the a user can edit a job or not

can_edit_job = Lightning.Policies.ProjectUsers |> Lightning.Policies.Permissions.can?(:edit_job, socket.assigns.current_user, socket.assigns.project)
 
-if can_edit_job do
+if can_edit_job do
   # allow user to edit the job
-else
+else
   # quick user out
-end
+
end
@@ -199,11 +199,11 @@

can(policy, action, user, params \\ [])

Examples -
iex> can(Lightning.Policies.Users, :create_workflow, user, project)
+
iex> can(Lightning.Policies.Users, :create_workflow, user, project)
 :ok
 
-iex> can(Lightning.Policies.Users, :create_project, user, %{})
-{:error, :unauthorized}
+
iex> can(Lightning.Policies.Users, :create_project, user, %{}) +{:error, :unauthorized}
@@ -233,10 +233,10 @@

can?(policy, action, user, params \\ []) Examples

-
iex> can(Lightning.Policies.Users, :create_workflow, user, project)
+
iex> can(Lightning.Policies.Users, :create_workflow, user, project)
 true
 
-iex> can(Lightning.Policies.Users, :create_project, user, %{})
+iex> can(Lightning.Policies.Users, :create_project, user, %{})
 false
diff --git a/Lightning.Projects.html b/Lightning.Projects.html index c9221e6520..a73f0905d4 100644 --- a/Lightning.Projects.html +++ b/Lightning.Projects.html @@ -540,8 +540,8 @@

change_project(project, attrs \\ %{})

Examples -
iex> change_project(project)
-%Ecto.Changeset{data: %Project{}}
+
iex> change_project(project)
+%Ecto.Changeset{data: %Project{}}
@@ -571,11 +571,11 @@

create_project(attrs \\ %{})

Examples -
iex> create_project(%{field: value})
-{:ok, %Project{}}
+
iex> create_project(%{field: value})
+{:ok, %Project{}}
 
-iex> create_project(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_project(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -604,11 +604,11 @@

delete_project(project)

Examples -
iex> delete_project(project)
-{:ok, %Project{}}
+
iex> delete_project(project)
+{:ok, %Project{}}
 
-iex> delete_project(project)
-{:error, %Ecto.Changeset{}}
+
iex> delete_project(project) +{:error, %Ecto.Changeset{}}
@@ -642,8 +642,8 @@

export_project(atom, project_id)

Examples -
iex> export_project(:yaml, project_id)
-{:ok, string}
+
iex> export_project(:yaml, project_id)
+{:ok, string}
@@ -693,10 +693,10 @@

get_project!(id)

Examples -
iex> get_project!(123)
-%Project{}
+
iex> get_project!(123)
+%Project{}
 
-iex> get_project!(456)
+iex> get_project!(456)
 ** (Ecto.NoResultsError)
@@ -791,10 +791,10 @@

get_project_user!(id)

Examples -
iex> get_project_user!(123)
-%ProjectUser{}
+
iex> get_project_user!(123)
+%ProjectUser{}
 
-iex> get_project_user!(456)
+iex> get_project_user!(456)
 ** (Ecto.NoResultsError)
@@ -824,16 +824,16 @@

get_project_user_role(user, project)

Examples -
iex> get_project_user_role(user, project)
+
iex> get_project_user_role(user, project)
 :admin
 
-iex> get_project_user_role(user, project)
+iex> get_project_user_role(user, project)
 :viewer
 
-iex> get_project_user_role(user, project)
+iex> get_project_user_role(user, project)
 :editor
 
-iex> get_project_user_role(user, project)
+iex> get_project_user_role(user, project)
 :owner
@@ -862,10 +862,10 @@

get_project_with_users!(id)

Examples -
iex> get_project!(123)
-%Project{}
+
iex> get_project!(123)
+%Project{}
 
-iex> get_project!(456)
+iex> get_project!(456)
 ** (Ecto.NoResultsError)
@@ -976,8 +976,8 @@

list_projects()

Examples -
iex> list_projects()
-[%Project{}, ...]
+
iex> list_projects()
+[%Project{}, ...]
@@ -1435,11 +1435,11 @@

update_project(project, attrs)

Examples -
iex> update_project(project, %{field: new_value})
-{:ok, %Project{}}
+
iex> update_project(project, %{field: new_value})
+{:ok, %Project{}}
 
-iex> update_project(project, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_project(project, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -1467,11 +1467,11 @@

update_project_user(project_user, attrs) Examples

-
iex> update_project_user(project_user, %{field: new_value})
-{:ok, %ProjectUser{}}
+
iex> update_project_user(project_user, %{field: new_value})
+{:ok, %ProjectUser{}}
 
-iex> update_project_user(projectUser, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_project_user(projectUser, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -1521,8 +1521,8 @@

validate_for_deletion(project, attrs)

Examples -
iex> validate_for_deletion(project)
-%Ecto.Changeset{data: %Project{}}
+
iex> validate_for_deletion(project)
+%Ecto.Changeset{data: %Project{}}
diff --git a/Lightning.Runtime.LogAgent.html b/Lightning.Runtime.LogAgent.html index a6bc92d402..05ba063da7 100644 --- a/Lightning.Runtime.LogAgent.html +++ b/Lightning.Runtime.LogAgent.html @@ -115,9 +115,9 @@

Agent facility to consume STDOUT/STDERR byte by byte.

Since it works on a byte by byte basis, you will need to perform line-splitting -yourself.

Usage:

{:ok, log} = LogAgent.start_link()
-"foo" = LogAgent.process_chunk(log, {:stdout, "foo"})
-"foobar" = LogAgent.process_chunk(log, {:stdout, "bar"})
+yourself.

Usage:

{:ok, log} = LogAgent.start_link()
+"foo" = LogAgent.process_chunk(log, {:stdout, "foo"})
+"foobar" = LogAgent.process_chunk(log, {:stdout, "bar"})
diff --git a/Lightning.Scrubber.html b/Lightning.Scrubber.html index c314961b0f..a5ef7e93c2 100644 --- a/Lightning.Scrubber.html +++ b/Lightning.Scrubber.html @@ -114,11 +114,11 @@

-

Process used to scrub strings of sensitive information.

Can be started via start_link/1.

{:ok, scrubber} =
-  Lightning.Scrubber.start_link(
+

Process used to scrub strings of sensitive information.

Can be started via start_link/1.

{:ok, scrubber} =
+  Lightning.Scrubber.start_link(
     samples:
-      Lightning.Credentials.sensitive_values_for(credential)
-  )

Takes an optional :name key, in case you need to name the process.

+
Lightning.Credentials.sensitive_values_for(credential) + )

Takes an optional :name key, in case you need to name the process.

diff --git a/Lightning.TaskWorker.html b/Lightning.TaskWorker.html index 0094acd777..301675f0d2 100644 --- a/Lightning.TaskWorker.html +++ b/Lightning.TaskWorker.html @@ -117,7 +117,7 @@

A TaskWorker with concurrency limits.

A simple concurrency limiter that wraps Task.Supervisor, which already does have the ability to specify max_children; it throws an error when that limit is exceeded.

To use it, start it like any other process; ideally in your supervision tree.

  ...,
-  {Lightning.TaskWorker, name: :cli_task_worker, max_tasks: 4}

Options

  • :max_tasks Defaults to the number of system schedulers available to the vm.
+ {Lightning.TaskWorker, name: :cli_task_worker, max_tasks: 4}

Options

  • :max_tasks Defaults to the number of system schedulers available to the vm.
diff --git a/Lightning.Validators.html b/Lightning.Validators.html index d55cb5372c..5bef836b3c 100644 --- a/Lightning.Validators.html +++ b/Lightning.Validators.html @@ -192,10 +192,10 @@

validate_exclusive(changeset, fields, messa

Validate that only one of the fields is set at a time.

Example:

changeset
-|> validate_exclusive(
-  [:source_job_id, :source_trigger_id],
+|> validate_exclusive(
+  [:source_job_id, :source_trigger_id],
   "source_job_id and source_trigger_id are mutually exclusive"
-)
+)

diff --git a/Lightning.WorkOrderService.html b/Lightning.WorkOrderService.html index 7fee045de5..dbea625cb6 100644 --- a/Lightning.WorkOrderService.html +++ b/Lightning.WorkOrderService.html @@ -342,11 +342,11 @@

create_work_order(attrs \\ %{})

Examples -
iex> create_work_order(%{field: value})
-{:ok, %WorkOrder{}}
+
iex> create_work_order(%{field: value})
+{:ok, %WorkOrder{}}
 
-iex> create_work_order(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_work_order(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.Workflows.html b/Lightning.Workflows.html index 834751f63f..7eca5233b8 100644 --- a/Lightning.Workflows.html +++ b/Lightning.Workflows.html @@ -349,8 +349,8 @@

change_workflow(workflow, attrs \\ %{})

Examples -
iex> change_workflow(workflow)
-%Ecto.Changeset{data: %Workflow{}}
+
iex> change_workflow(workflow)
+%Ecto.Changeset{data: %Workflow{}}
@@ -402,11 +402,11 @@

create_workflow(attrs \\ %{})

Examples -
iex> create_workflow(%{field: value})
-{:ok, %Workflow{}}
+
iex> create_workflow(%{field: value})
+{:ok, %Workflow{}}
 
-iex> create_workflow(%{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> create_workflow(%{field: bad_value}) +{:error, %Ecto.Changeset{}}
@@ -434,11 +434,11 @@

delete_workflow(workflow)

Examples -
iex> delete_workflow(workflow)
-{:ok, %Workflow{}}
+
iex> delete_workflow(workflow)
+{:ok, %Workflow{}}
 
-iex> delete_workflow(workflow)
-{:error, %Ecto.Changeset{}}
+
iex> delete_workflow(workflow) +{:error, %Ecto.Changeset{}}
@@ -539,10 +539,10 @@

get_workflow!(id)

Examples -
iex> get_workflow!(123)
-%Workflow{}
+
iex> get_workflow!(123)
+%Workflow{}
 
-iex> get_workflow!(456)
+iex> get_workflow!(456)
 ** (Ecto.NoResultsError)
@@ -623,8 +623,8 @@

list_workflows()

Examples -
iex> list_workflows()
-[%Workflow{}, ...]
+
iex> list_workflows()
+[%Workflow{}, ...]
@@ -654,8 +654,8 @@

mark_for_deletion(workflow, attrs \\ %{}) Examples

-
iex> change_request_deletion(workflow)
-%Ecto.Changeset{data: %Workflow{}}
+
iex> change_request_deletion(workflow)
+%Ecto.Changeset{data: %Workflow{}}
@@ -733,11 +733,11 @@

update_workflow(workflow, attrs)

Examples -
iex> update_workflow(workflow, %{field: new_value})
-{:ok, %Workflow{}}
+
iex> update_workflow(workflow, %{field: new_value})
+{:ok, %Workflow{}}
 
-iex> update_workflow(workflow, %{field: bad_value})
-{:error, %Ecto.Changeset{}}
+
iex> update_workflow(workflow, %{field: bad_value}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.epub b/Lightning.epub index 522fc681920073a1868122961d6752c2a583a37d..5053930fc48dcdf7189edb3590de0130c75559e1 100644 GIT binary patch delta 91970 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 z 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ġjawga}L#O)mDaq*Su|EkrV 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
    DN$pLl%$?-_72$2GPOd+vCge0 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 I;1H7j2inWi@&Et; diff --git a/LightningWeb.Gettext.html b/LightningWeb.Gettext.html index 4f63ac1593..654c2cddc8 100644 --- a/LightningWeb.Gettext.html +++ b/LightningWeb.Gettext.html @@ -118,15 +118,15 @@

    your module gains a set of macros for translations, for example:

    import LightningWeb.Gettext
     
     # Simple translation
    -gettext("Here is the string to translate")
    +gettext("Here is the string to translate")
     
     # Plural translation
    -ngettext("Here is the string to translate",
    +ngettext("Here is the string to translate",
              "Here are the strings to translate",
    -         3)
    +         3)
     
     # Domain-based translation
    -dgettext("errors", "Here is the error message to translate")

    See the Gettext Docs for detailed usage.

    +dgettext("errors", "Here is the error message to translate")

See the Gettext Docs for detailed usage.

diff --git a/LightningWeb.OauthCredentialHelper.html b/LightningWeb.OauthCredentialHelper.html index ac4b3a15d1..4519e6c7c6 100644 --- a/LightningWeb.OauthCredentialHelper.html +++ b/LightningWeb.OauthCredentialHelper.html @@ -240,10 +240,10 @@

broadcast_forward(subscription_id, mod, opt

Broadcast a message specifically for forwarding a message to a component. It expects a subscription_id, the module of the component and opts being a keyword list containing an :id key of the specific component.

See: Phoenix.LiveView.send_update/3 for more info.

A corresponding LiveView (that is subscribed) is expected to have a matching -handle_info/2 that looks like this:

def handle_info({:forward, mod, opts}, socket) do
-  send_update(mod, opts)
-  {:noreply, socket}
-end
+handle_info/2 that looks like this:

def handle_info({:forward, mod, opts}, socket) do
+  send_update(mod, opts)
+  {:noreply, socket}
+end
diff --git a/LightningWeb.Pagination.html b/LightningWeb.Pagination.html index 5bb6906538..583810b35c 100644 --- a/LightningWeb.Pagination.html +++ b/LightningWeb.Pagination.html @@ -243,13 +243,13 @@

raw_pagination_links(paginator, options \\

Returns the raw data in order to generate the proper HTML for pagination links. Data is returned in a {text, page_number} format where text is intended to be the text of the link and page_number is the page it should go to. Defaults are already supplied -and they are as follows:

[distance: 5, next: :next, previous: :previous, first: true, last: true, ellipsis: :ellipsis]

distance must be a positive non-zero integer or an exception is raised. next and previous should be +and they are as follows:

[distance: 5, next: :next, previous: :previous, first: true, last: true, ellipsis: :ellipsis]

distance must be a positive non-zero integer or an exception is raised. next and previous should be strings but can be anything you want as long as it is truthy, falsey values will remove them from the output. first and last are only booleans, and they just include/remove -their respective link from output. An example of the data returned:

iex> Scrivener.HTML.raw_pagination_links(%{total_pages: 10, page_number: 5})
-[{"<<", 4}, {1, 1}, {2, 2}, {3, 3}, {4, 4}, {5, 5}, {6, 6}, {7, 7}, {8, 8}, {9, 9}, {10, 10}, {">>", 6}]
-iex> Scrivener.HTML.raw_pagination_links(%{total_pages: 20, page_number: 10}, first: ["←"], last: ["→"])
-[{"<<", 9}, {["←"], 1}, {:ellipsis, {:safe, "&hellip;"}}, {5, 5}, {6, 6},{7, 7}, {8, 8}, {9, 9}, {10, 10}, {11, 11}, {12, 12}, {13, 13}, {14, 14},{15, 15}, {:ellipsis, {:safe, "&hellip;"}}, {["→"], 20}, {">>", 11}]

Simply loop and pattern match over each item and transform it to your custom HTML.

+their respective link from output. An example of the data returned:

iex> Scrivener.HTML.raw_pagination_links(%{total_pages: 10, page_number: 5})
+[{"<<", 4}, {1, 1}, {2, 2}, {3, 3}, {4, 4}, {5, 5}, {6, 6}, {7, 7}, {8, 8}, {9, 9}, {10, 10}, {">>", 6}]
+iex> Scrivener.HTML.raw_pagination_links(%{total_pages: 20, page_number: 10}, first: ["←"], last: ["→"])
+[{"<<", 9}, {["←"], 1}, {:ellipsis, {:safe, "&hellip;"}}, {5, 5}, {6, 6},{7, 7}, {8, 8}, {9, 9}, {10, 10}, {11, 11}, {12, 12}, {13, 13}, {14, 14},{15, 15}, {:ellipsis, {:safe, "&hellip;"}}, {["→"], 20}, {">>", 11}]

Simply loop and pattern match over each item and transform it to your custom HTML.

diff --git a/Mix.Tasks.Lightning.InstallSchemas.html b/Mix.Tasks.Lightning.InstallSchemas.html index 446ad6a4bb..7811de130d 100644 --- a/Mix.Tasks.Lightning.InstallSchemas.html +++ b/Mix.Tasks.Lightning.InstallSchemas.html @@ -1797,14 +1797,14 @@

request(request)

Examples -
request = %HTTPoison.Request{
+
request = %HTTPoison.Request{
   method: :post,
   url: "https://my.website.com",
   body: "{\"foo\": 3}",
-  headers: [{"Accept", "application/json"}]
-}
+  headers: [{"Accept", "application/json"}]
+}
 
-request(request)
+
request(request)
@@ -1862,7 +1862,7 @@

request(method, url, body \\ "", Examples

-
request(:post, "https://my.website.com", "{\"foo\": 3}", [{"Accept", "application/json"}])
+
request(:post, "https://my.website.com", "{\"foo\": 3}", [{"Accept", "application/json"}])
diff --git a/changelog.html b/changelog.html index 8ccbd31970..3c7399b744 100644 --- a/changelog.html +++ b/changelog.html @@ -132,7 +132,7 @@

Changed

  • Moved to Elixir 1.15 and Erlang 26.0.2 to sort our an annoying ElixirLS issue -that was slowing down our engineers.

+that was slowing down our engineers.
  • Update Debian base to use bookworm (Debian 12) for our Docker images
  • fixed

    diff --git a/dist/search_items-C2951775.js b/dist/search_items-01F3CC93.js similarity index 93% rename from dist/search_items-C2951775.js rename to dist/search_items-01F3CC93.js index d0edc547e8..a2f83d6a23 100644 --- a/dist/search_items-C2951775.js +++ b/dist/search_items-01F3CC93.js @@ -1 +1 @@ -searchNodes=[{"type":"module","title":"Lightning","doc":"Lightning keeps the contexts that define your domain and business logic. Contexts are also responsible for managing your data, regardless if it comes from the database, an external API or others.","ref":"Lightning.html"},{"type":"module","title":"Lightning.AdaptorRegistry","doc":"Registry process to query and maintain a list of adaptors available for writing jobs. Currently it queries NPM for all modules in the @openfn organization and filters out modules that are known not to be adaptors. Usage # Starting the process AdaptorRegistry . start_link ( ) # Getting a list of all adaptors Lightning.AdaptorRegistry.AdaptorRegistry . all ( ) Caching By default the results are cached to disk, and will be reused every start. In order to disable or configure caching pass see: start_link/1 . The process uses :continue to return before the adaptors have been queried. This does mean that the first call to the process will be delayed until the handle_continue/2 has finished. Timeouts There is a 'general' timeout of 30s, this is used for GenServer calls like all/1 and also internally when the modules are being queried. NPM can be extremely fast to respond if the package is cached on their side, but can take a couple of seconds if not cached.","ref":"Lightning.AdaptorRegistry.html"},{"type":"function","title":"Lightning.AdaptorRegistry.all/1","doc":"Get the current in-process list of adaptors. This call will wait behind the :continue message when the process starts up, so it may take a while the first time it is called (and the list hasn't been fetched yet).","ref":"Lightning.AdaptorRegistry.html#all/1"},{"type":"function","title":"Lightning.AdaptorRegistry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorRegistry.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorRegistry.fetch/0","doc":"Fetch a list of packages for the @openfn organisation","ref":"Lightning.AdaptorRegistry.html#fetch/0"},{"type":"function","title":"Lightning.AdaptorRegistry.latest_for/2","doc":"Get a latest version for a given module.","ref":"Lightning.AdaptorRegistry.html#latest_for/2"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_adaptor/1","doc":"","ref":"Lightning.AdaptorRegistry.html#resolve_adaptor/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name/1","doc":"Destructures an NPM style package name into module name and version. Example iex> resolve_package_name ( "@openfn/language-salesforce@1.2.3" ) { "@openfn/language-salesforce" , "1.2.3" } iex> resolve_package_name ( "@openfn/language-salesforce" ) { "@openfn/language-salesforce" , nil }","ref":"Lightning.AdaptorRegistry.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name!/1","doc":"Same as resolve_package_name/1 except will throw an exception if a package name cannot be matched.","ref":"Lightning.AdaptorRegistry.html#resolve_package_name!/1"},{"type":"function","title":"Lightning.AdaptorRegistry.start_link/1","doc":"Starts the AdaptorRegistry Options :use_cache (defaults to false) - stores the last set of results on disk and uses the cached file for every subsequent start. It can either be a boolean, or a string - the latter being a file path to set where the cache file is located. :name (defaults to AdaptorRegistry) - the name of the process, useful for testing and/or running multiple versions of the registry","ref":"Lightning.AdaptorRegistry.html#start_link/1"},{"type":"function","title":"Lightning.AdaptorRegistry.versions_for/2","doc":"Get a list of versions for a given module.","ref":"Lightning.AdaptorRegistry.html#versions_for/2"},{"type":"module","title":"Lightning.AdaptorRegistry.Npm","doc":"NPM API functions","ref":"Lightning.AdaptorRegistry.Npm.html"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.package_detail/1","doc":"Retrieve all details for an NPM package","ref":"Lightning.AdaptorRegistry.Npm.html#package_detail/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_body/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_options/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_params/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_url/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AdaptorRegistry.Npm.html#request/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AdaptorRegistry.Npm.html#request/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AdaptorRegistry.Npm.html#request!/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AdaptorRegistry.Npm.html#start/0"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#stream_next/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.user_packages/1","doc":"Retrieve all packages for a given user or organization. Return empty list if application cannot connect to NPM. (E.g., because it's started offline.)","ref":"Lightning.AdaptorRegistry.Npm.html#user_packages/1"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.body/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:body/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.headers/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:headers/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.method/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:method/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.options/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:options/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.params/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:params/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.request/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:request/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.url/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:url/0"},{"type":"module","title":"Lightning.AdaptorService","doc":"The Adaptor Service is use to query and install adaptors in order to run jobs. On startup, it queries the filesystem for package.json files and builds up a list of available adaptors. Configuration The service requires at least :adaptors_path , which is used to both query which adaptors are installed and when to install new adaptors. Another optional setting is: :repo , which must point at a module that will be used to do the querying and installing. Installing Adaptors Using the install/2 function an adaptor can be installed, which will also add it to the list of available adaptors. The adaptor is marked as :installing , to allow for conditional behaviour elsewhere such as delaying or rejecting processing until the adaptor becomes available. Looking up adaptors The module leans on Elixir's built-in Version module to provide version lookups. When looking up an adaptor, either a string or a tuple can be used. In the case of requesting the latest version, any one of these will return the latest version the service is aware of. @openfn/language-http @openfn/language-http@latest {"@openfn/language-http", nil} {"@openfn/language-http", "latest"} {~r/language-http/, "latest"} You can also request a specific version, or use a range specification: @openfn/language-http@1.2.3 {"@openfn/language-http", "~> 1.2.0"} {"@openfn/language-http", "< 2.0.0"} NOTE More complex npm style install strings like: ">=0.1.0 <0.2.0" are not supported. Generally the tuple style is preferred when using range specifications as the npm style strings have a simplistic regex splitter. See Version for more details on matching versions.","ref":"Lightning.AdaptorService.html"},{"type":"function","title":"Lightning.AdaptorService.build_aliased_name/1","doc":"Turns a package name and version into a string for NPM. Since multiple versions of the same package can be installed, it's important to rely on npms built-in package aliasing. E.g. @openfn/language-http@1.2.8 turns into: ` @openfn / language - http - 1.2 . 8 @npm : @openfn / language - http @ 1.2 . 8 ` Which is pretty long winded but necessary for the reason above. If using this module as a base, it's likely you would need to adaptor this to suit your particular naming strategy.","ref":"Lightning.AdaptorService.html#build_aliased_name/1"},{"type":"function","title":"Lightning.AdaptorService.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorService.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorService.find_adaptor/2","doc":"","ref":"Lightning.AdaptorService.html#find_adaptor/2"},{"type":"function","title":"Lightning.AdaptorService.get_adaptors/1","doc":"","ref":"Lightning.AdaptorService.html#get_adaptors/1"},{"type":"function","title":"Lightning.AdaptorService.install/2","doc":"","ref":"Lightning.AdaptorService.html#install/2"},{"type":"function","title":"Lightning.AdaptorService.install!/2","doc":"","ref":"Lightning.AdaptorService.html#install!/2"},{"type":"function","title":"Lightning.AdaptorService.installed?/2","doc":"","ref":"Lightning.AdaptorService.html#installed?/2"},{"type":"function","title":"Lightning.AdaptorService.resolve_package_name/1","doc":"","ref":"Lightning.AdaptorService.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorService.start_link/1","doc":"","ref":"Lightning.AdaptorService.html#start_link/1"},{"type":"type","title":"Lightning.AdaptorService.package_spec/0","doc":"","ref":"Lightning.AdaptorService.html#t:package_spec/0"},{"type":"module","title":"Lightning.Attempt","doc":"Ecto model for Attempts.","ref":"Lightning.Attempt.html"},{"type":"function","title":"Lightning.Attempt.new/1","doc":"","ref":"Lightning.Attempt.html#new/1"},{"type":"type","title":"Lightning.Attempt.t/0","doc":"","ref":"Lightning.Attempt.html#t:t/0"},{"type":"module","title":"Lightning.AttemptRun","doc":"Ecto model for an Attempts Runs.","ref":"Lightning.AttemptRun.html"},{"type":"function","title":"Lightning.AttemptRun.new/1","doc":"","ref":"Lightning.AttemptRun.html#new/1"},{"type":"function","title":"Lightning.AttemptRun.new/2","doc":"","ref":"Lightning.AttemptRun.html#new/2"},{"type":"type","title":"Lightning.AttemptRun.t/0","doc":"","ref":"Lightning.AttemptRun.html#t:t/0"},{"type":"module","title":"Lightning.AttemptService","doc":"The Attempts context.","ref":"Lightning.AttemptService.html"},{"type":"function","title":"Lightning.AttemptService.append/2","doc":"Adds an Attempt to an unsaved Run When given an Attempt, it simply adds the Run to a new AttemptRun. However when given an AttemptRun, the Run (from the AttemptRun) is set as the previous Run for the new unsaved Run.","ref":"Lightning.AttemptService.html#append/2"},{"type":"function","title":"Lightning.AttemptService.build_attempt/2","doc":"","ref":"Lightning.AttemptService.html#build_attempt/2"},{"type":"function","title":"Lightning.AttemptService.calculate_runs/3","doc":"","ref":"Lightning.AttemptService.html#calculate_runs/3"},{"type":"function","title":"Lightning.AttemptService.create_attempt/3","doc":"Create an attempt Examples iex> create_attempt ( %{ field : value } ) { :ok , % Attempt { } } iex> create_attempt ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.AttemptService.html#create_attempt/3"},{"type":"function","title":"Lightning.AttemptService.get_for_rerun/2","doc":"","ref":"Lightning.AttemptService.html#get_for_rerun/2"},{"type":"function","title":"Lightning.AttemptService.get_last_attempt_for/1","doc":"Get the latest attempt associated to a given run","ref":"Lightning.AttemptService.html#get_last_attempt_for/1"},{"type":"function","title":"Lightning.AttemptService.get_workflow_for/1","doc":"","ref":"Lightning.AttemptService.html#get_workflow_for/1"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_job/2","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of workorder ids that are associated to the given Job","ref":"Lightning.AttemptService.html#list_for_rerun_from_job/2"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_start/1","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of work order ids.","ref":"Lightning.AttemptService.html#list_for_rerun_from_start/1"},{"type":"function","title":"Lightning.AttemptService.retry/3","doc":"Creates a new Attempt starting from a given run. All upstream/prior Runs that were performed on that attempt are associated with the new Attempt, where as the specified run is used to create a new one and is added to the Attempt. Any runs downstream from the Run given are ignored.","ref":"Lightning.AttemptService.html#retry/3"},{"type":"function","title":"Lightning.AttemptService.retry_many/2","doc":"Creates new Attempts for each pair of corresponding AttemptRun and InvocationReason.","ref":"Lightning.AttemptService.html#retry_many/2"},{"type":"module","title":"Lightning.Auditing","doc":"Context for working with Audit records.","ref":"Lightning.Auditing.html"},{"type":"function","title":"Lightning.Auditing.list_all/1","doc":"","ref":"Lightning.Auditing.html#list_all/1"},{"type":"module","title":"Lightning.Auditing.Model","doc":"Macro module to add common model behaviour to a given Ecto model","ref":"Lightning.Auditing.Model.html"},{"type":"function","title":"Lightning.Auditing.Model.event/5","doc":"Creates a schema changeset for the event identified by row_id and caused by actor_id . The given metadata can be either nil , Ecto.Changeset , struct or map. It returns :no_changes in case of an Ecto.Changeset metadata that changed nothing or an Ecto.Changeset with the event ready to be inserted.","ref":"Lightning.Auditing.Model.html#event/5"},{"type":"function","title":"Lightning.Auditing.Model.save/2","doc":"Saves the event to the Repo . In case of nothing changes, do nothing. It returns {:ok, :no_changes} if nothing changed, {:ok, struct} if the log has been successfully saved or {:error, changeset} in case of error.","ref":"Lightning.Auditing.Model.html#save/2"},{"type":"module","title":"Lightning.AuthProviders","doc":"Context module for dealing with external Auth Providers.","ref":"Lightning.AuthProviders.html"},{"type":"function","title":"Lightning.AuthProviders.build_handler/2","doc":"","ref":"Lightning.AuthProviders.html#build_handler/2"},{"type":"function","title":"Lightning.AuthProviders.create/1","doc":"","ref":"Lightning.AuthProviders.html#create/1"},{"type":"function","title":"Lightning.AuthProviders.create_handler/1","doc":"","ref":"Lightning.AuthProviders.html#create_handler/1"},{"type":"function","title":"Lightning.AuthProviders.delete!/1","doc":"","ref":"Lightning.AuthProviders.html#delete!/1"},{"type":"function","title":"Lightning.AuthProviders.get_authorize_url/1","doc":"Retrieve the authorization url for a given handler or handler name.","ref":"Lightning.AuthProviders.html#get_authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.get_existing/0","doc":"","ref":"Lightning.AuthProviders.html#get_existing/0"},{"type":"function","title":"Lightning.AuthProviders.get_existing/1","doc":"","ref":"Lightning.AuthProviders.html#get_existing/1"},{"type":"function","title":"Lightning.AuthProviders.get_handler/1","doc":"","ref":"Lightning.AuthProviders.html#get_handler/1"},{"type":"function","title":"Lightning.AuthProviders.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.new/0","doc":"","ref":"Lightning.AuthProviders.html#new/0"},{"type":"function","title":"Lightning.AuthProviders.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.html#remove_handler/1"},{"type":"function","title":"Lightning.AuthProviders.update/2","doc":"","ref":"Lightning.AuthProviders.html#update/2"},{"type":"module","title":"Lightning.AuthProviders.AuthConfig","doc":"AuthProvider model","ref":"Lightning.AuthProviders.AuthConfig.html"},{"type":"type","title":"Lightning.AuthProviders.AuthConfig.t/0","doc":"","ref":"Lightning.AuthProviders.AuthConfig.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.CacheWarmer","doc":"Dummy warmer which caches database rows every 30s.","ref":"Lightning.AuthProviders.CacheWarmer.html"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AuthProviders.CacheWarmer.html#child_spec/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.execute/1","doc":"Executes this cache warmer with a connection.","ref":"Lightning.AuthProviders.CacheWarmer.html#execute/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.interval/0","doc":"Returns the interval for this warmer.","ref":"Lightning.AuthProviders.CacheWarmer.html#interval/0"},{"type":"module","title":"Lightning.AuthProviders.Google","doc":"","ref":"Lightning.AuthProviders.Google.html"},{"type":"function","title":"Lightning.AuthProviders.Google.authorize_url/2","doc":"","ref":"Lightning.AuthProviders.Google.html#authorize_url/2"},{"type":"function","title":"Lightning.AuthProviders.Google.build_client/1","doc":"Builds a new client","ref":"Lightning.AuthProviders.Google.html#build_client/1"},{"type":"function","title":"Lightning.AuthProviders.Google.get_config/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_config/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown!/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown!/0"},{"type":"function","title":"Lightning.AuthProviders.Google.refresh_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#refresh_token/2"},{"type":"module","title":"Lightning.AuthProviders.Handler","doc":"Module which wraps Oauth configuration and a WellKnown document into a convenient struct that can be used to authenticate users against any OIDC compliant provider.","ref":"Lightning.AuthProviders.Handler.html"},{"type":"function","title":"Lightning.AuthProviders.Handler.authorize_url/1","doc":"","ref":"Lightning.AuthProviders.Handler.html#authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.from_model/1","doc":"Returns a Handler from a given AuthConfig","ref":"Lightning.AuthProviders.Handler.html#from_model/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_token/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.new/2","doc":"Create a new Provider struct, expects a name and opts: :client_id - The providers issued id :client_secret - Secret for the client :redirect_uri - The URI for redirecting after authentication, usually the callback url in the router. :wellknown - A AuthProviders.WellKnown struct with the providers .well-known/openid-configuration .","ref":"Lightning.AuthProviders.Handler.html#new/2"},{"type":"type","title":"Lightning.AuthProviders.Handler.opts/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:opts/0"},{"type":"type","title":"Lightning.AuthProviders.Handler.t/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.Store","doc":"Store module for caching Handlers. Since Handlers often have to fetch their .well-known files when being initialized we cache these in order to avoid repeatedly making HTTP requests to a providers API.","ref":"Lightning.AuthProviders.Store.html"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.Store.put_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#put_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.Store.html#remove_handler/1"},{"type":"type","title":"Lightning.AuthProviders.Store.finder/0","doc":"","ref":"Lightning.AuthProviders.Store.html#t:finder/0"},{"type":"module","title":"Lightning.AuthProviders.WellKnown","doc":"A datastructure to fetch and hold information about a given OIDC/OAuth provider","ref":"Lightning.AuthProviders.WellKnown.html"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch!/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch!/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.new/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#new/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_options/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_params/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AuthProviders.WellKnown.html#request/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AuthProviders.WellKnown.html#request/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AuthProviders.WellKnown.html#request!/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AuthProviders.WellKnown.html#start/0"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#stream_next/1"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.body/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:body/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.headers/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:headers/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.method/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:method/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.options/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:options/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.params/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:params/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.request/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:request/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.t/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:t/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.url/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:url/0"},{"type":"module","title":"Lightning.CLI","doc":"Module providing facilities to make calls to the OpenFn CLI. See @openfn/cli","ref":"Lightning.CLI.html"},{"type":"function","title":"Lightning.CLI.execute/1","doc":"Execute a command in a child process and parse the results.","ref":"Lightning.CLI.html#execute/1"},{"type":"function","title":"Lightning.CLI.metadata/2","doc":"Retrieve metadata for a given adaptor and configuration.","ref":"Lightning.CLI.html#metadata/2"},{"type":"module","title":"Lightning.CLI.Result","doc":"Struct that wraps the output of an OpenFn CLI call. Containing the keys: start_time end_time status logs Logs The OpenFn CLI returns JSON formatted log lines, which are decoded and added to a Result struct. There are two kinds of output: { "level" :"<<level>>" , "name" :"<<module>>" , "message" :"..." ] , "time" :< < timestamp >> } These are usually for general logging, and debugging. { "message" : [ "<<message|filepath|output>>" ] } The above is the equivalent of the output of a command","ref":"Lightning.CLI.Result.html"},{"type":"function","title":"Lightning.CLI.Result.get_messages/1","doc":"Returns message type log lines from a Result .","ref":"Lightning.CLI.Result.html#get_messages/1"},{"type":"function","title":"Lightning.CLI.Result.new/1","doc":"","ref":"Lightning.CLI.Result.html#new/1"},{"type":"function","title":"Lightning.CLI.Result.parse/2","doc":"","ref":"Lightning.CLI.Result.html#parse/2"},{"type":"type","title":"Lightning.CLI.Result.t/0","doc":"","ref":"Lightning.CLI.Result.html#t:t/0"},{"type":"module","title":"Lightning.Demo","doc":"Demo encapsulates logic for setting up a demonstration site.","ref":"Lightning.Demo.html"},{"type":"function","title":"Lightning.Demo.reset_demo/0","doc":"Deletes everything in the database including the superuser and creates a set of publicly available users for a demo site via a command that can be run on Kubernetes-deployed systems.","ref":"Lightning.Demo.html#reset_demo/0"},{"type":"module","title":"Lightning.ExportUtils","doc":"Module that expose a function generating a complete and valid yaml string from a project and its workflows.","ref":"Lightning.ExportUtils.html"},{"type":"function","title":"Lightning.ExportUtils.build_yaml_tree/2","doc":"","ref":"Lightning.ExportUtils.html#build_yaml_tree/2"},{"type":"function","title":"Lightning.ExportUtils.generate_new_yaml/1","doc":"","ref":"Lightning.ExportUtils.html#generate_new_yaml/1"},{"type":"module","title":"Lightning.FailureNotifierView","doc":"","ref":"Lightning.FailureNotifierView.html"},{"type":"function","title":"Lightning.FailureNotifierView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"Lightning.FailureNotifierView.html#__resource__/0"},{"type":"function","title":"Lightning.FailureNotifierView.failure_alert.html/1","doc":"","ref":"Lightning.FailureNotifierView.html#failure_alert.html/1"},{"type":"function","title":"Lightning.FailureNotifierView.render/2","doc":"Renders the given template locally.","ref":"Lightning.FailureNotifierView.html#render/2"},{"type":"function","title":"Lightning.FailureNotifierView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"Lightning.FailureNotifierView.html#template_not_found/2"},{"type":"module","title":"Lightning.Helpers","doc":"Common functions for the context","ref":"Lightning.Helpers.html"},{"type":"function","title":"Lightning.Helpers.coerce_json_field/2","doc":"Changes a given maps field from a json string to a map. If it cannot be converted, it leaves the original value","ref":"Lightning.Helpers.html#coerce_json_field/2"},{"type":"function","title":"Lightning.Helpers.indefinite_article/1","doc":"","ref":"Lightning.Helpers.html#indefinite_article/1"},{"type":"function","title":"Lightning.Helpers.json_safe/1","doc":"Recursively ensures a given map is safe to convert to JSON, where all keys are strings and all values are json safe (primitive values).","ref":"Lightning.Helpers.html#json_safe/1"},{"type":"function","title":"Lightning.Helpers.ms_to_human/1","doc":"Converts milliseconds (integer) to a human duration, such as "1 minute" or "45 years, 6 months, 5 days, 21 hours, 12 minutes, 34 seconds" using Timex.Format.Duration.Formatters.Humanized.format() .","ref":"Lightning.Helpers.html#ms_to_human/1"},{"type":"module","title":"Lightning.MetadataService","doc":"Retrieves metadata for a given credential and adaptor using the OpenFn CLI.","ref":"Lightning.MetadataService.html"},{"type":"function","title":"Lightning.MetadataService.fetch/2","doc":"Retrieve metadata for a given adaptor and credential. The adaptor must be an npm specification.","ref":"Lightning.MetadataService.html#fetch/2"},{"type":"module","title":"Lightning.Name","doc":"Generates a random names.","ref":"Lightning.Name.html"},{"type":"function","title":"Lightning.Name.generate/1","doc":"","ref":"Lightning.Name.html#generate/1"},{"type":"module","title":"Lightning.ObanManager","doc":"The Oban Manager","ref":"Lightning.ObanManager.html"},{"type":"function","title":"Lightning.ObanManager.handle_event/4","doc":"","ref":"Lightning.ObanManager.html#handle_event/4"},{"type":"module","title":"Lightning.Policies.Permissions","doc":"This module defines a unique interface managing authorizations in Lightning. Users in Lightning have instance-wide and project-wide roles which determine their level of access to resources in the application. Fo rmore details see the documentation . These authorizations policies are all implemented under the lib/lightning/policies folder. In that folder you can find 3 files: The users.ex file has all the policies for the instances wide access levels The project_users.ex file has all the policies for the project wide access levels The permissions.ex file defines the Lightning.Policies.Permissions.can/4 interface. Which is a wrapper around the Bodyguard.permit/4 function. We use that interface to be able to harmonize the use of policies accross the entire app. All the policies are tested in the test/lightning/policies folder. And the test are written in a way that allows the reader to quickly who can do what in the app. We have two variants of the Lightning.Policies.Permissions.can/4 interface: Lightning.Policies.Permissions.can(policy, action, actor, resource) returns :ok if the actor can perform the action on the resource and {:error, :unauthorized} otherwise. Lightning.Policies.Permissions.can?(policy, action, actor, resource) returns true if the actor can perform the action on the resource and false otherwise. Here is an example of how we the Lightning.Policies.Permissions.can/4 interface to check if the a user can edit a job or not can_edit_job = Lightning.Policies.ProjectUsers |> Lightning.Policies.Permissions . can? ( :edit_job , socket . assigns . current_user , socket . assigns . project ) if can_edit_job do # allow user to edit the job else # quick user out end","ref":"Lightning.Policies.Permissions.html"},{"type":"function","title":"Lightning.Policies.Permissions.can/4","doc":"checks if user has the permissions to apply action using some policy module Returns :ok if user can apply action and {:error, :unauthorized} otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) :ok iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) { :error , :unauthorized }","ref":"Lightning.Policies.Permissions.html#can/4"},{"type":"function","title":"Lightning.Policies.Permissions.can?/4","doc":"same as can/4 but returns true if user can apply action and false otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) true iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) false","ref":"Lightning.Policies.Permissions.html#can?/4"},{"type":"module","title":"Lightning.Policies.ProjectUsers","doc":"The Bodyguard Policy module for projects members roles.","ref":"Lightning.Policies.ProjectUsers.html"},{"type":"function","title":"Lightning.Policies.ProjectUsers.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action in that project and false if they cannot. Note that permissions are grouped by action, rather than by user role. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.ProjectUsers.html#authorize/3"},{"type":"type","title":"Lightning.Policies.ProjectUsers.actions/0","doc":"","ref":"Lightning.Policies.ProjectUsers.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Provisioning","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Provisioning.html"},{"type":"function","title":"Lightning.Policies.Provisioning.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied. Only a superuser can provision a new project. Owners and admins can update existing projects.","ref":"Lightning.Policies.Provisioning.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Provisioning.actions/0","doc":"","ref":"Lightning.Policies.Provisioning.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Users","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Users.html"},{"type":"function","title":"Lightning.Policies.Users.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.Users.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Users.actions/0","doc":"","ref":"Lightning.Policies.Users.html#t:actions/0"},{"type":"module","title":"Lightning.Release","doc":"Used for executing DB release tasks when run in production without Mix installed.","ref":"Lightning.Release.html"},{"type":"function","title":"Lightning.Release.create_db/0","doc":"","ref":"Lightning.Release.html#create_db/0"},{"type":"function","title":"Lightning.Release.load_app/0","doc":"","ref":"Lightning.Release.html#load_app/0"},{"type":"function","title":"Lightning.Release.migrate/0","doc":"","ref":"Lightning.Release.html#migrate/0"},{"type":"function","title":"Lightning.Release.rollback/2","doc":"","ref":"Lightning.Release.html#rollback/2"},{"type":"module","title":"Lightning.Repo","doc":"","ref":"Lightning.Repo.html"},{"type":"function","title":"Lightning.Repo.aggregate/3","doc":"Callback implementation for Ecto.Repo.aggregate/3 .","ref":"Lightning.Repo.html#aggregate/3"},{"type":"function","title":"Lightning.Repo.aggregate/4","doc":"Callback implementation for Ecto.Repo.aggregate/4 .","ref":"Lightning.Repo.html#aggregate/4"},{"type":"function","title":"Lightning.Repo.all/2","doc":"Callback implementation for Ecto.Repo.all/2 .","ref":"Lightning.Repo.html#all/2"},{"type":"function","title":"Lightning.Repo.checked_out?/0","doc":"Callback implementation for Ecto.Repo.checked_out?/0 .","ref":"Lightning.Repo.html#checked_out?/0"},{"type":"function","title":"Lightning.Repo.checkout/2","doc":"Callback implementation for Ecto.Repo.checkout/2 .","ref":"Lightning.Repo.html#checkout/2"},{"type":"function","title":"Lightning.Repo.child_spec/1","doc":"","ref":"Lightning.Repo.html#child_spec/1"},{"type":"function","title":"Lightning.Repo.config/0","doc":"Callback implementation for Ecto.Repo.config/0 .","ref":"Lightning.Repo.html#config/0"},{"type":"function","title":"Lightning.Repo.default_options/1","doc":"Callback implementation for Ecto.Repo.default_options/1 .","ref":"Lightning.Repo.html#default_options/1"},{"type":"function","title":"Lightning.Repo.delete/2","doc":"Callback implementation for Ecto.Repo.delete/2 .","ref":"Lightning.Repo.html#delete/2"},{"type":"function","title":"Lightning.Repo.delete!/2","doc":"Callback implementation for Ecto.Repo.delete!/2 .","ref":"Lightning.Repo.html#delete!/2"},{"type":"function","title":"Lightning.Repo.delete_all/2","doc":"Callback implementation for Ecto.Repo.delete_all/2 .","ref":"Lightning.Repo.html#delete_all/2"},{"type":"function","title":"Lightning.Repo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the pool to disconnect within the given interval. See Ecto.Adapters.SQL.disconnect_all/3 for more information.","ref":"Lightning.Repo.html#disconnect_all/2"},{"type":"function","title":"Lightning.Repo.exists?/2","doc":"Callback implementation for Ecto.Repo.exists?/2 .","ref":"Lightning.Repo.html#exists?/2"},{"type":"function","title":"Lightning.Repo.explain/3","doc":"A convenience function for SQL-based repositories that executes an EXPLAIN statement or similar depending on the adapter to obtain statistics for the given query. See Ecto.Adapters.SQL.explain/4 for more information.","ref":"Lightning.Repo.html#explain/3"},{"type":"function","title":"Lightning.Repo.get/3","doc":"Callback implementation for Ecto.Repo.get/3 .","ref":"Lightning.Repo.html#get/3"},{"type":"function","title":"Lightning.Repo.get!/3","doc":"Callback implementation for Ecto.Repo.get!/3 .","ref":"Lightning.Repo.html#get!/3"},{"type":"function","title":"Lightning.Repo.get_by/3","doc":"Callback implementation for Ecto.Repo.get_by/3 .","ref":"Lightning.Repo.html#get_by/3"},{"type":"function","title":"Lightning.Repo.get_by!/3","doc":"Callback implementation for Ecto.Repo.get_by!/3 .","ref":"Lightning.Repo.html#get_by!/3"},{"type":"function","title":"Lightning.Repo.get_dynamic_repo/0","doc":"Callback implementation for Ecto.Repo.get_dynamic_repo/0 .","ref":"Lightning.Repo.html#get_dynamic_repo/0"},{"type":"function","title":"Lightning.Repo.in_transaction?/0","doc":"Callback implementation for Ecto.Repo.in_transaction?/0 .","ref":"Lightning.Repo.html#in_transaction?/0"},{"type":"function","title":"Lightning.Repo.insert/2","doc":"Callback implementation for Ecto.Repo.insert/2 .","ref":"Lightning.Repo.html#insert/2"},{"type":"function","title":"Lightning.Repo.insert!/2","doc":"Callback implementation for Ecto.Repo.insert!/2 .","ref":"Lightning.Repo.html#insert!/2"},{"type":"function","title":"Lightning.Repo.insert_all/3","doc":"Callback implementation for Ecto.Repo.insert_all/3 .","ref":"Lightning.Repo.html#insert_all/3"},{"type":"function","title":"Lightning.Repo.insert_or_update/2","doc":"Callback implementation for Ecto.Repo.insert_or_update/2 .","ref":"Lightning.Repo.html#insert_or_update/2"},{"type":"function","title":"Lightning.Repo.insert_or_update!/2","doc":"Callback implementation for Ecto.Repo.insert_or_update!/2 .","ref":"Lightning.Repo.html#insert_or_update!/2"},{"type":"function","title":"Lightning.Repo.load/2","doc":"Callback implementation for Ecto.Repo.load/2 .","ref":"Lightning.Repo.html#load/2"},{"type":"function","title":"Lightning.Repo.one/2","doc":"Callback implementation for Ecto.Repo.one/2 .","ref":"Lightning.Repo.html#one/2"},{"type":"function","title":"Lightning.Repo.one!/2","doc":"Callback implementation for Ecto.Repo.one!/2 .","ref":"Lightning.Repo.html#one!/2"},{"type":"function","title":"Lightning.Repo.paginate/2","doc":"","ref":"Lightning.Repo.html#paginate/2"},{"type":"function","title":"Lightning.Repo.preload/3","doc":"Callback implementation for Ecto.Repo.preload/3 .","ref":"Lightning.Repo.html#preload/3"},{"type":"function","title":"Lightning.Repo.prepare_query/3","doc":"Callback implementation for Ecto.Repo.prepare_query/3 .","ref":"Lightning.Repo.html#prepare_query/3"},{"type":"function","title":"Lightning.Repo.put_dynamic_repo/1","doc":"Callback implementation for Ecto.Repo.put_dynamic_repo/1 .","ref":"Lightning.Repo.html#put_dynamic_repo/1"},{"type":"function","title":"Lightning.Repo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query/4 for more information.","ref":"Lightning.Repo.html#query/3"},{"type":"function","title":"Lightning.Repo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query!/4 for more information.","ref":"Lightning.Repo.html#query!/3"},{"type":"function","title":"Lightning.Repo.query_many/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many/4 for more information.","ref":"Lightning.Repo.html#query_many/3"},{"type":"function","title":"Lightning.Repo.query_many!/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many!/4 for more information.","ref":"Lightning.Repo.html#query_many!/3"},{"type":"function","title":"Lightning.Repo.reload/2","doc":"Callback implementation for Ecto.Repo.reload/2 .","ref":"Lightning.Repo.html#reload/2"},{"type":"function","title":"Lightning.Repo.reload!/2","doc":"Callback implementation for Ecto.Repo.reload!/2 .","ref":"Lightning.Repo.html#reload!/2"},{"type":"function","title":"Lightning.Repo.rollback/1","doc":"Callback implementation for Ecto.Repo.rollback/1 .","ref":"Lightning.Repo.html#rollback/1"},{"type":"function","title":"Lightning.Repo.scrivener_defaults/0","doc":"","ref":"Lightning.Repo.html#scrivener_defaults/0"},{"type":"function","title":"Lightning.Repo.start_link/1","doc":"Callback implementation for Ecto.Repo.start_link/1 .","ref":"Lightning.Repo.html#start_link/1"},{"type":"function","title":"Lightning.Repo.stop/1","doc":"Callback implementation for Ecto.Repo.stop/1 .","ref":"Lightning.Repo.html#stop/1"},{"type":"function","title":"Lightning.Repo.stream/2","doc":"Callback implementation for Ecto.Repo.stream/2 .","ref":"Lightning.Repo.html#stream/2"},{"type":"function","title":"Lightning.Repo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL. See Ecto.Adapters.SQL.to_sql/3 for more information.","ref":"Lightning.Repo.html#to_sql/2"},{"type":"function","title":"Lightning.Repo.transaction/2","doc":"Callback implementation for Ecto.Repo.transaction/2 .","ref":"Lightning.Repo.html#transaction/2"},{"type":"function","title":"Lightning.Repo.update/2","doc":"Callback implementation for Ecto.Repo.update/2 .","ref":"Lightning.Repo.html#update/2"},{"type":"function","title":"Lightning.Repo.update!/2","doc":"Callback implementation for Ecto.Repo.update!/2 .","ref":"Lightning.Repo.html#update!/2"},{"type":"function","title":"Lightning.Repo.update_all/3","doc":"Callback implementation for Ecto.Repo.update_all/3 .","ref":"Lightning.Repo.html#update_all/3"},{"type":"module","title":"Lightning.RunSearchForm","doc":"Run filtering search form.","ref":"Lightning.RunSearchForm.html"},{"type":"module","title":"Lightning.SafetyString","doc":"Utilities for securely encoding serializable structs, lists and strings into URL-safe strings. In order to pass the state around in a URL, in a manner that protects secrets from leaking - and allows us to avoid persistance we take a set of parameters and: Encode into a URI query string gzip it to save characters encrypt the string base64 encode it for URI encoding safety","ref":"Lightning.SafetyString.html"},{"type":"function","title":"Lightning.SafetyString.decode/1","doc":"","ref":"Lightning.SafetyString.html#decode/1"},{"type":"function","title":"Lightning.SafetyString.encode/1","doc":"","ref":"Lightning.SafetyString.html#encode/1"},{"type":"module","title":"Lightning.Scrubber","doc":"Process used to scrub strings of sensitive information. Can be started via start_link/1 . { :ok , scrubber } = Lightning.Scrubber . start_link ( samples : Lightning.Credentials . sensitive_values_for ( credential ) ) Takes an optional :name key, in case you need to name the process.","ref":"Lightning.Scrubber.html"},{"type":"function","title":"Lightning.Scrubber.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Scrubber.html#child_spec/1"},{"type":"function","title":"Lightning.Scrubber.encode_samples/1","doc":"Prepare a list of sensitive samples (strings) into a potentially bigger list composed of variations a sample may appear.","ref":"Lightning.Scrubber.html#encode_samples/1"},{"type":"function","title":"Lightning.Scrubber.samples/1","doc":"","ref":"Lightning.Scrubber.html#samples/1"},{"type":"function","title":"Lightning.Scrubber.scrub/2","doc":"","ref":"Lightning.Scrubber.html#scrub/2"},{"type":"function","title":"Lightning.Scrubber.start_link/1","doc":"","ref":"Lightning.Scrubber.html#start_link/1"},{"type":"module","title":"Lightning.SetupUtils","doc":"SetupUtils encapsulates logic for setting up initial data for various sites.","ref":"Lightning.SetupUtils.html"},{"type":"function","title":"Lightning.SetupUtils.add_and_update_runs/3","doc":"","ref":"Lightning.SetupUtils.html#add_and_update_runs/3"},{"type":"function","title":"Lightning.SetupUtils.create_dhis2_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_dhis2_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_openhie_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_openhie_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_starter_project/2","doc":"","ref":"Lightning.SetupUtils.html#create_starter_project/2"},{"type":"function","title":"Lightning.SetupUtils.setup_demo/1","doc":"Creates initial data and returns the created records.","ref":"Lightning.SetupUtils.html#setup_demo/1"},{"type":"function","title":"Lightning.SetupUtils.tear_down/1","doc":"","ref":"Lightning.SetupUtils.html#tear_down/1"},{"type":"module","title":"Lightning.TaskWorker","doc":"A TaskWorker with concurrency limits. A simple concurrency limiter that wraps Task.Supervisor , which already does have the ability to specify max_children ; it throws an error when that limit is exceeded. To use it, start it like any other process; ideally in your supervision tree. ... , { Lightning.TaskWorker , name : :cli_task_worker , max_tasks : 4 } Options :max_tasks Defaults to the number of system schedulers available to the vm.","ref":"Lightning.TaskWorker.html"},{"type":"function","title":"Lightning.TaskWorker.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.TaskWorker.html#child_spec/1"},{"type":"function","title":"Lightning.TaskWorker.get_status/1","doc":"","ref":"Lightning.TaskWorker.html#get_status/1"},{"type":"function","title":"Lightning.TaskWorker.start_link/1","doc":"","ref":"Lightning.TaskWorker.html#start_link/1"},{"type":"function","title":"Lightning.TaskWorker.start_task/2","doc":"","ref":"Lightning.TaskWorker.html#start_task/2"},{"type":"module","title":"Lightning.Validators","doc":"Extra validators for Ecto.Changeset.","ref":"Lightning.Validators.html"},{"type":"function","title":"Lightning.Validators.validate_exclusive/3","doc":"Validate that only one of the fields is set at a time. Example: changeset |> validate_exclusive ( [ :source_job_id , :source_trigger_id ] , "source_job_id and source_trigger_id are mutually exclusive" )","ref":"Lightning.Validators.html#validate_exclusive/3"},{"type":"function","title":"Lightning.Validators.validate_one_required/3","doc":"Validate that at least one of the fields is set.","ref":"Lightning.Validators.html#validate_one_required/3"},{"type":"module","title":"Lightning.Vault","doc":"Module for handling the encryption and decryption of database fields.","ref":"Lightning.Vault.html"},{"type":"function","title":"Lightning.Vault.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Vault.html#child_spec/1"},{"type":"function","title":"Lightning.Vault.start_link/1","doc":"","ref":"Lightning.Vault.html#start_link/1"},{"type":"module","title":"Lightning.WorkOrder","doc":"Ecto model for Workorders.","ref":"Lightning.WorkOrder.html"},{"type":"function","title":"Lightning.WorkOrder.new/0","doc":"","ref":"Lightning.WorkOrder.html#new/0"},{"type":"type","title":"Lightning.WorkOrder.t/0","doc":"","ref":"Lightning.WorkOrder.html#t:t/0"},{"type":"module","title":"Lightning.WorkOrderService","doc":"The WorkOrderService.","ref":"Lightning.WorkOrderService.html"},{"type":"function","title":"Lightning.WorkOrderService.attempt_updated/1","doc":"","ref":"Lightning.WorkOrderService.html#attempt_updated/1"},{"type":"function","title":"Lightning.WorkOrderService.build/2","doc":"","ref":"Lightning.WorkOrderService.html#build/2"},{"type":"function","title":"Lightning.WorkOrderService.create_manual_workorder/3","doc":"","ref":"Lightning.WorkOrderService.html#create_manual_workorder/3"},{"type":"function","title":"Lightning.WorkOrderService.create_webhook_workorder/2","doc":"","ref":"Lightning.WorkOrderService.html#create_webhook_workorder/2"},{"type":"function","title":"Lightning.WorkOrderService.create_work_order/1","doc":"Creates a work_order. Examples iex> create_work_order ( %{ field : value } ) { :ok , % WorkOrder { } } iex> create_work_order ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.WorkOrderService.html#create_work_order/1"},{"type":"function","title":"Lightning.WorkOrderService.multi_for/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for/3"},{"type":"function","title":"Lightning.WorkOrderService.multi_for_manual/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for_manual/3"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_run/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_run/2"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_runs/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_runs/2"},{"type":"function","title":"Lightning.WorkOrderService.subscribe/1","doc":"","ref":"Lightning.WorkOrderService.html#subscribe/1"},{"type":"module","title":"Lightning.Workflows","doc":"The Workflows context.","ref":"Lightning.Workflows.html"},{"type":"function","title":"Lightning.Workflows.build_trigger/1","doc":"Builds a Trigger","ref":"Lightning.Workflows.html#build_trigger/1"},{"type":"function","title":"Lightning.Workflows.change_workflow/2","doc":"Returns an %Ecto.Changeset{} for tracking workflow changes. Examples iex> change_workflow ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#change_workflow/2"},{"type":"function","title":"Lightning.Workflows.create_edge/1","doc":"Creates an edge","ref":"Lightning.Workflows.html#create_edge/1"},{"type":"function","title":"Lightning.Workflows.create_workflow/1","doc":"Creates a workflow. Examples iex> create_workflow ( %{ field : value } ) { :ok , % Workflow { } } iex> create_workflow ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#create_workflow/1"},{"type":"function","title":"Lightning.Workflows.delete_workflow/1","doc":"Deletes a workflow. Examples iex> delete_workflow ( workflow ) { :ok , % Workflow { } } iex> delete_workflow ( workflow ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#delete_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_edge_by_webhook/1","doc":"Gets a Single Edge by it's webhook trigger.","ref":"Lightning.Workflows.html#get_edge_by_webhook/1"},{"type":"function","title":"Lightning.Workflows.get_edges_for_cron_execution/1","doc":"Returns a list of edges with jobs to execute, given a current timestamp in Unix. This is used by the scheduler, which calls this function once every minute.","ref":"Lightning.Workflows.html#get_edges_for_cron_execution/1"},{"type":"function","title":"Lightning.Workflows.get_workflow/1","doc":"","ref":"Lightning.Workflows.html#get_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_workflow!/1","doc":"Gets a single workflow. Raises Ecto.NoResultsError if the Workflow does not exist. Examples iex> get_workflow! ( 123 ) % Workflow { } iex> get_workflow! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Workflows.html#get_workflow!/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for/1","doc":"Retrieves a list of Workflows with their jobs and triggers preloaded.","ref":"Lightning.Workflows.html#get_workflows_for/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for_query/1","doc":"","ref":"Lightning.Workflows.html#get_workflows_for_query/1"},{"type":"function","title":"Lightning.Workflows.list_workflows/0","doc":"Returns the list of workflows. Examples iex> list_workflows ( ) [ % Workflow { } , ... ]","ref":"Lightning.Workflows.html#list_workflows/0"},{"type":"function","title":"Lightning.Workflows.mark_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the workflow request_deletion. Examples iex> change_request_deletion ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#mark_for_deletion/2"},{"type":"function","title":"Lightning.Workflows.to_project_space/1","doc":"","ref":"Lightning.Workflows.html#to_project_space/1"},{"type":"function","title":"Lightning.Workflows.update_trigger/2","doc":"Updates a trigger","ref":"Lightning.Workflows.html#update_trigger/2"},{"type":"function","title":"Lightning.Workflows.update_workflow/2","doc":"Updates a workflow. Examples iex> update_workflow ( workflow , %{ field : new_value } ) { :ok , % Workflow { } } iex> update_workflow ( workflow , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#update_workflow/2"},{"type":"module","title":"Lightning.Workflows.Edge","doc":"Ecto model for Workflow Edges. A Workflow Edge represents a connection between two jobs (or a trigger and a job) in a workflow. The source of the edge is either a job or a trigger. The target of the edge is always a job.","ref":"Lightning.Workflows.Edge.html"},{"type":"function","title":"Lightning.Workflows.Edge.changeset/2","doc":"","ref":"Lightning.Workflows.Edge.html#changeset/2"},{"type":"function","title":"Lightning.Workflows.Edge.new/1","doc":"","ref":"Lightning.Workflows.Edge.html#new/1"},{"type":"function","title":"Lightning.Workflows.Edge.validate/1","doc":"","ref":"Lightning.Workflows.Edge.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Edge.edge_condition/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:edge_condition/0"},{"type":"type","title":"Lightning.Workflows.Edge.t/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:t/0"},{"type":"module","title":"Lightning.Workflows.Graph","doc":"Utility to construct and manipulate a graph/plan made out of Jobs","ref":"Lightning.Workflows.Graph.html"},{"type":"function","title":"Lightning.Workflows.Graph.new/1","doc":"","ref":"Lightning.Workflows.Graph.html#new/1"},{"type":"function","title":"Lightning.Workflows.Graph.remove/2","doc":"","ref":"Lightning.Workflows.Graph.html#remove/2"},{"type":"function","title":"Lightning.Workflows.Graph.vertices/1","doc":"","ref":"Lightning.Workflows.Graph.html#vertices/1"},{"type":"type","title":"Lightning.Workflows.Graph.t/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:t/0"},{"type":"type","title":"Lightning.Workflows.Graph.vertex/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:vertex/0"},{"type":"module","title":"Lightning.Workflows.Workflow","doc":"Ecto model for Workflows. A Workflow contains the fields for defining a workflow. name A plain text identifier","ref":"Lightning.Workflows.Workflow.html"},{"type":"function","title":"Lightning.Workflows.Workflow.request_deletion_changeset/2","doc":"","ref":"Lightning.Workflows.Workflow.html#request_deletion_changeset/2"},{"type":"function","title":"Lightning.Workflows.Workflow.validate/1","doc":"","ref":"Lightning.Workflows.Workflow.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Workflow.t/0","doc":"","ref":"Lightning.Workflows.Workflow.html#t:t/0"},{"type":"module","title":"Lightning.Workorders.SearchParams","doc":"This module is used to parse search parameters for workorders and provide a query to the database.","ref":"Lightning.Workorders.SearchParams.html"},{"type":"function","title":"Lightning.Workorders.SearchParams.new/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#new/1"},{"type":"function","title":"Lightning.Workorders.SearchParams.to_uri_params/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#to_uri_params/1"},{"type":"type","title":"Lightning.Workorders.SearchParams.t/0","doc":"","ref":"Lightning.Workorders.SearchParams.html#t:t/0"},{"type":"module","title":"LightningWeb","doc":"The entrypoint for defining your web interface, such as controllers, views, channels and so on. This can be used in your application as: use LightningWeb , :controller use LightningWeb , :view The definitions below will be executed for every view, controller, etc, so keep them short and clean, focused on imports, uses and aliases. Do NOT define functions inside the quoted expressions below. Instead, define any helper function in modules and import those modules here.","ref":"LightningWeb.html"},{"type":"macro","title":"LightningWeb.__using__/1","doc":"When used, dispatch to the appropriate controller/view/etc.","ref":"LightningWeb.html#__using__/1"},{"type":"function","title":"LightningWeb.channel/0","doc":"","ref":"LightningWeb.html#channel/0"},{"type":"function","title":"LightningWeb.component/0","doc":"","ref":"LightningWeb.html#component/0"},{"type":"function","title":"LightningWeb.controller/0","doc":"","ref":"LightningWeb.html#controller/0"},{"type":"function","title":"LightningWeb.html/0","doc":"","ref":"LightningWeb.html#html/0"},{"type":"function","title":"LightningWeb.live_component/0","doc":"","ref":"LightningWeb.html#live_component/0"},{"type":"function","title":"LightningWeb.live_view/1","doc":"","ref":"LightningWeb.html#live_view/1"},{"type":"function","title":"LightningWeb.router/0","doc":"","ref":"LightningWeb.html#router/0"},{"type":"function","title":"LightningWeb.static_paths/0","doc":"","ref":"LightningWeb.html#static_paths/0"},{"type":"function","title":"LightningWeb.verified_routes/0","doc":"","ref":"LightningWeb.html#verified_routes/0"},{"type":"function","title":"LightningWeb.view/0","doc":"","ref":"LightningWeb.html#view/0"},{"type":"module","title":"LightningWeb.API.Helpers","doc":"Helpers for the API views","ref":"LightningWeb.API.Helpers.html"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_link/3","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_link/3"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_links/2","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_links/2"},{"type":"function","title":"LightningWeb.API.Helpers.url_for/2","doc":"","ref":"LightningWeb.API.Helpers.html#url_for/2"},{"type":"module","title":"LightningWeb.API.JobController","doc":"","ref":"LightningWeb.API.JobController.html"},{"type":"function","title":"LightningWeb.API.JobController.index/2","doc":"","ref":"LightningWeb.API.JobController.html#index/2"},{"type":"function","title":"LightningWeb.API.JobController.show/2","doc":"","ref":"LightningWeb.API.JobController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProjectController","doc":"","ref":"LightningWeb.API.ProjectController.html"},{"type":"function","title":"LightningWeb.API.ProjectController.index/2","doc":"","ref":"LightningWeb.API.ProjectController.html#index/2"},{"type":"function","title":"LightningWeb.API.ProjectController.show/2","doc":"","ref":"LightningWeb.API.ProjectController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProvisioningController","doc":"","ref":"LightningWeb.API.ProvisioningController.html"},{"type":"function","title":"LightningWeb.API.ProvisioningController.create/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#create/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#show/2"},{"type":"module","title":"LightningWeb.API.RunController","doc":"","ref":"LightningWeb.API.RunController.html"},{"type":"function","title":"LightningWeb.API.RunController.index/2","doc":"","ref":"LightningWeb.API.RunController.html#index/2"},{"type":"function","title":"LightningWeb.API.RunController.show/2","doc":"","ref":"LightningWeb.API.RunController.html#show/2"},{"type":"module","title":"LightningWeb.AuditLive.Index","doc":"LiveView for listing Audit events","ref":"LightningWeb.AuditLive.Index.html"},{"type":"function","title":"LightningWeb.AuditLive.Index.diff/1","doc":"","ref":"LightningWeb.AuditLive.Index.html#diff/1"},{"type":"function","title":"LightningWeb.AuditLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.AuditLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.AuthProvidersLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.AuthProvidersLive.FormComponent.html"},{"type":"module","title":"LightningWeb.AuthProvidersLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.AuthProvidersLive.Index.html"},{"type":"module","title":"LightningWeb.ChangesetJSON","doc":"Renders changesets as JSON.","ref":"LightningWeb.ChangesetJSON.html"},{"type":"function","title":"LightningWeb.ChangesetJSON.error/1","doc":"","ref":"LightningWeb.ChangesetJSON.html#error/1"},{"type":"module","title":"LightningWeb.ChangesetView","doc":"","ref":"LightningWeb.ChangesetView.html"},{"type":"function","title":"LightningWeb.ChangesetView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ChangesetView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ChangesetView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ChangesetView.html#render/2"},{"type":"function","title":"LightningWeb.ChangesetView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ChangesetView.html#template_not_found/2"},{"type":"function","title":"LightningWeb.ChangesetView.translate_errors/1","doc":"Traverses and translates changeset errors. See Ecto.Changeset.traverse_errors/2 and LightningWeb.ErrorHelpers.translate_error/1 for more details.","ref":"LightningWeb.ChangesetView.html#translate_errors/1"},{"type":"module","title":"LightningWeb.Components.Modal","doc":"A modal component that can be used to display a modal on the page. This currently isn't used anywhere but should be used in the future to replace the existing modal implementations.","ref":"LightningWeb.Components.Modal.html"},{"type":"module","title":"LightningWeb.CredentialLive.Edit","doc":"LiveView for editing a single Credential, which inturn uses LightningWeb.CredentialLive.FormComponent for common functionality.","ref":"LightningWeb.CredentialLive.Edit.html"},{"type":"function","title":"LightningWeb.CredentialLive.Edit.handle_info/2","doc":"A generic handler for forwarding updates from PubSub","ref":"LightningWeb.CredentialLive.Edit.html#handle_info/2"},{"type":"module","title":"LightningWeb.CredentialLive.FormComponent","doc":"Form Component for working with a single Credential","ref":"LightningWeb.CredentialLive.FormComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.credential_transfer/1","doc":"Attributes users ( :list ) (required) form ( :map ) (required)","ref":"LightningWeb.CredentialLive.FormComponent.html#credential_transfer/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.form_component/1","doc":"Switcher components for different types of credentials. Attributes type ( :string ) (required) form ( :map ) (required) update_body ( :any ) Slots inner_block","ref":"LightningWeb.CredentialLive.FormComponent.html#form_component/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.project_credentials/1","doc":"Attributes form ( :map ) (required) projects ( :list ) (required) selected ( :map ) (required) phx_target ( :any ) - Defaults to nil .","ref":"LightningWeb.CredentialLive.FormComponent.html#project_credentials/1"},{"type":"module","title":"LightningWeb.CredentialLive.GoogleSheetsComponent","doc":"Form component to setup a Google Sheets component. This component has several moving parts: Subscribes to a PubSub topic specially link to the component id See: LightningWeb.OauthCredentialHelper . Uses the Lightning.Google module to set up an OAuth client for generating urls, exchanging the code and requesting a new access_token . The flow for creating a new token is: Generate an authorization link which contains: The authorization url from the Google client with the applications callback_url A state string that is an encrypted set of data with the components module and id in it Once the user authorizes the client the callback is requested with a code The LightningWeb.OidcController decodes the state returned to it and does a 'broadcast_forward' which is simply a message expected to be received by a LiveView and applied to Phoenix.LiveView.send_update/3 . The component receives the code and requests a token. Any changes to the token (Credential body) are still handled by the parent component and so a update_body function is passed in to send params changes back up to update the form.","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.disabled_authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#disabled_authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.error_block/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#error_block/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.fieldset/1","doc":"Attributes form ( :map ) (required) update_body ( :any ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.userinfo/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#userinfo/1"},{"type":"module","title":"LightningWeb.CredentialLive.Index","doc":"LiveView for listing and managing credentials","ref":"LightningWeb.CredentialLive.Index.html"},{"type":"function","title":"LightningWeb.CredentialLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.CredentialLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.schema_input/1","doc":"Attributes form ( :map ) (required) schema ( :map ) (required) field ( :any ) (required)","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#schema_input/1"},{"type":"module","title":"LightningWeb.CredentialLive.RawBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.RawBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.RawBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.RawBodyComponent.html#fieldset/1"},{"type":"module","title":"LightningWeb.CredentialLive.TypePicker","doc":"","ref":"LightningWeb.CredentialLive.TypePicker.html"},{"type":"module","title":"LightningWeb.DataclipLive.Edit","doc":"LiveView for editing a single dataclip.","ref":"LightningWeb.DataclipLive.Edit.html"},{"type":"function","title":"LightningWeb.DataclipLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.FormComponent","doc":"Form Component for working with a single dataclip","ref":"LightningWeb.DataclipLive.FormComponent.html"},{"type":"function","title":"LightningWeb.DataclipLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.DataclipLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.Index","doc":"LiveView for listing and working with a list of Dataclips","ref":"LightningWeb.DataclipLive.Index.html"},{"type":"function","title":"LightningWeb.DataclipLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.DownloadsController","doc":"","ref":"LightningWeb.DownloadsController.html"},{"type":"function","title":"LightningWeb.DownloadsController.download_project_yaml/2","doc":"","ref":"LightningWeb.DownloadsController.html#download_project_yaml/2"},{"type":"module","title":"LightningWeb.Endpoint","doc":"","ref":"LightningWeb.Endpoint.html"},{"type":"function","title":"LightningWeb.Endpoint.broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast/3 .","ref":"LightningWeb.Endpoint.html#broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast!/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast!/3 .","ref":"LightningWeb.Endpoint.html#broadcast!/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from!/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from!/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from!/4"},{"type":"function","title":"LightningWeb.Endpoint.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.Endpoint.html#call/2"},{"type":"function","title":"LightningWeb.Endpoint.child_spec/1","doc":"Returns the child specification to start the endpoint under a supervision tree.","ref":"LightningWeb.Endpoint.html#child_spec/1"},{"type":"function","title":"LightningWeb.Endpoint.config/2","doc":"Returns the endpoint configuration for key Returns default if the key does not exist.","ref":"LightningWeb.Endpoint.html#config/2"},{"type":"function","title":"LightningWeb.Endpoint.config_change/2","doc":"Reloads the configuration given the application environment changes.","ref":"LightningWeb.Endpoint.html#config_change/2"},{"type":"function","title":"LightningWeb.Endpoint.host/0","doc":"Returns the host for the given endpoint.","ref":"LightningWeb.Endpoint.html#host/0"},{"type":"function","title":"LightningWeb.Endpoint.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.Endpoint.html#init/1"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast/3 .","ref":"LightningWeb.Endpoint.html#local_broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#local_broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.path/1","doc":"Generates the path information when routing to this endpoint.","ref":"LightningWeb.Endpoint.html#path/1"},{"type":"function","title":"LightningWeb.Endpoint.script_name/0","doc":"Generates the script name.","ref":"LightningWeb.Endpoint.html#script_name/0"},{"type":"function","title":"LightningWeb.Endpoint.start_link/1","doc":"Starts the endpoint supervision tree. All other options are merged into the endpoint configuration.","ref":"LightningWeb.Endpoint.html#start_link/1"},{"type":"function","title":"LightningWeb.Endpoint.static_integrity/1","doc":"Generates a base64-encoded cryptographic hash (sha512) to a static file in priv/static . Meant to be used for Subresource Integrity with CDNs.","ref":"LightningWeb.Endpoint.html#static_integrity/1"},{"type":"function","title":"LightningWeb.Endpoint.static_lookup/1","doc":"Returns a two item tuple with the first item being the static_path and the second item being the static_integrity .","ref":"LightningWeb.Endpoint.html#static_lookup/1"},{"type":"function","title":"LightningWeb.Endpoint.static_path/1","doc":"Generates a route to a static file in priv/static .","ref":"LightningWeb.Endpoint.html#static_path/1"},{"type":"function","title":"LightningWeb.Endpoint.static_url/0","doc":"Generates the static URL without any path information. It uses the configuration under :static_url to generate such. It falls back to :url if :static_url is not set.","ref":"LightningWeb.Endpoint.html#static_url/0"},{"type":"function","title":"LightningWeb.Endpoint.struct_url/0","doc":"Generates the endpoint base URL but as a URI struct. It uses the configuration under :url to generate such. Useful for manipulating the URL data and passing it to URL helpers.","ref":"LightningWeb.Endpoint.html#struct_url/0"},{"type":"function","title":"LightningWeb.Endpoint.subscribe/2","doc":"Callback implementation for Phoenix.Endpoint.subscribe/2 .","ref":"LightningWeb.Endpoint.html#subscribe/2"},{"type":"function","title":"LightningWeb.Endpoint.unsubscribe/1","doc":"Callback implementation for Phoenix.Endpoint.unsubscribe/1 .","ref":"LightningWeb.Endpoint.html#unsubscribe/1"},{"type":"function","title":"LightningWeb.Endpoint.url/0","doc":"Generates the endpoint base URL without any path information. It uses the configuration under :url to generate such.","ref":"LightningWeb.Endpoint.html#url/0"},{"type":"module","title":"LightningWeb.ErrorView","doc":"","ref":"LightningWeb.ErrorView.html"},{"type":"function","title":"LightningWeb.ErrorView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ErrorView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ErrorView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ErrorView.html#render/2"},{"type":"function","title":"LightningWeb.ErrorView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ErrorView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.FallbackController","doc":"Translates controller action results into valid Plug.Conn responses. See Phoenix.Controller.action_fallback/1 for more details.","ref":"LightningWeb.FallbackController.html"},{"type":"module","title":"LightningWeb.FirstSetupLive.Superuser","doc":"Superuser setup liveview Allows the creation on the first user in the system. It has only one action: :show","ref":"LightningWeb.FirstSetupLive.Superuser.html"},{"type":"function","title":"LightningWeb.FirstSetupLive.Superuser.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.FirstSetupLive.Superuser.html#render/1"},{"type":"module","title":"LightningWeb.FormHelpers","doc":"Conveniences for building forms.","ref":"LightningWeb.FormHelpers.html"},{"type":"module","title":"LightningWeb.Gettext","doc":"A module providing Internationalization with a gettext-based API. By using Gettext , your module gains a set of macros for translations, for example: import LightningWeb.Gettext # Simple translation gettext ( "Here is the string to translate" ) # Plural translation ngettext ( "Here is the string to translate" , "Here are the strings to translate" , 3 ) # Domain-based translation dgettext ( "errors" , "Here is the error message to translate" ) See the Gettext Docs for detailed usage.","ref":"LightningWeb.Gettext.html"},{"type":"macro","title":"LightningWeb.Gettext.dgettext/3","doc":"Callback implementation for Gettext.Backend.dgettext/3 .","ref":"LightningWeb.Gettext.html#dgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.dgettext_noop/2","doc":"Callback implementation for Gettext.Backend.dgettext_noop/2 .","ref":"LightningWeb.Gettext.html#dgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.dngettext/5","doc":"Callback implementation for Gettext.Backend.dngettext/5 .","ref":"LightningWeb.Gettext.html#dngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.dngettext_noop/3","doc":"Callback implementation for Gettext.Backend.dngettext_noop/3 .","ref":"LightningWeb.Gettext.html#dngettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext/4","doc":"Callback implementation for Gettext.Backend.dpgettext/4 .","ref":"LightningWeb.Gettext.html#dpgettext/4"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#dpgettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext/6","doc":"Callback implementation for Gettext.Backend.dpngettext/6 .","ref":"LightningWeb.Gettext.html#dpngettext/6"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext_noop/4","doc":"","ref":"LightningWeb.Gettext.html#dpngettext_noop/4"},{"type":"macro","title":"LightningWeb.Gettext.gettext/2","doc":"Callback implementation for Gettext.Backend.gettext/2 .","ref":"LightningWeb.Gettext.html#gettext/2"},{"type":"macro","title":"LightningWeb.Gettext.gettext_comment/1","doc":"Callback implementation for Gettext.Backend.gettext_comment/1 .","ref":"LightningWeb.Gettext.html#gettext_comment/1"},{"type":"macro","title":"LightningWeb.Gettext.gettext_noop/1","doc":"Callback implementation for Gettext.Backend.gettext_noop/1 .","ref":"LightningWeb.Gettext.html#gettext_noop/1"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_bindings/2","doc":"Callback implementation for Gettext.Backend.handle_missing_bindings/2 .","ref":"LightningWeb.Gettext.html#handle_missing_bindings/2"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_plural_translation/7","doc":"Callback implementation for Gettext.Backend.handle_missing_plural_translation/7 .","ref":"LightningWeb.Gettext.html#handle_missing_plural_translation/7"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_translation/5","doc":"Callback implementation for Gettext.Backend.handle_missing_translation/5 .","ref":"LightningWeb.Gettext.html#handle_missing_translation/5"},{"type":"function","title":"LightningWeb.Gettext.lgettext/5","doc":"","ref":"LightningWeb.Gettext.html#lgettext/5"},{"type":"function","title":"LightningWeb.Gettext.lngettext/7","doc":"","ref":"LightningWeb.Gettext.html#lngettext/7"},{"type":"macro","title":"LightningWeb.Gettext.ngettext/4","doc":"Callback implementation for Gettext.Backend.ngettext/4 .","ref":"LightningWeb.Gettext.html#ngettext/4"},{"type":"macro","title":"LightningWeb.Gettext.ngettext_noop/2","doc":"Callback implementation for Gettext.Backend.ngettext_noop/2 .","ref":"LightningWeb.Gettext.html#ngettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pgettext/3","doc":"Callback implementation for Gettext.Backend.pgettext/3 .","ref":"LightningWeb.Gettext.html#pgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.pgettext_noop/2","doc":"","ref":"LightningWeb.Gettext.html#pgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pngettext/5","doc":"Callback implementation for Gettext.Backend.pngettext/5 .","ref":"LightningWeb.Gettext.html#pngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.pngettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#pngettext_noop/3"},{"type":"module","title":"LightningWeb.HealthCheck","doc":"","ref":"LightningWeb.HealthCheck.html"},{"type":"function","title":"LightningWeb.HealthCheck.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.HealthCheck.html#call/2"},{"type":"function","title":"LightningWeb.HealthCheck.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.HealthCheck.html#init/1"},{"type":"module","title":"LightningWeb.Hooks","doc":"LiveView Hooks","ref":"LightningWeb.Hooks.html"},{"type":"function","title":"LightningWeb.Hooks.on_mount/4","doc":"Finds and assigns a project to the socket, if a user doesn't have access they are redirected and shown a 'No Access' screen via a :nav flash message. There is a fallthru function, when there is no project_id in the params - this is for liveviews that may or may not have a project_id depending on usage - like DashboardLive .","ref":"LightningWeb.Hooks.html#on_mount/4"},{"type":"module","title":"LightningWeb.InitAssigns","doc":"Ensures common assigns are applied to all LiveViews attaching this hook.","ref":"LightningWeb.InitAssigns.html"},{"type":"function","title":"LightningWeb.InitAssigns.on_mount/4","doc":"","ref":"LightningWeb.InitAssigns.html#on_mount/4"},{"type":"module","title":"LightningWeb.JobLive.AdaptorPicker","doc":"Component allowing selecting an adaptor and it's version","ref":"LightningWeb.JobLive.AdaptorPicker.html"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.display_name_for_adaptor/1","doc":"Converts standard adaptor names into "label","value" lists and returns non-standard names as merely "value"; both can be passed directly into a select option list.","ref":"LightningWeb.JobLive.AdaptorPicker.html#display_name_for_adaptor/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.get_adaptor_version_options/1","doc":"","ref":"LightningWeb.JobLive.AdaptorPicker.html#get_adaptor_version_options/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.render/1","doc":"Attributes form ( :map ) (required) on_change ( :any ) - Defaults to nil . disabled ( :boolean ) - Defaults to false .","ref":"LightningWeb.JobLive.AdaptorPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CredentialPicker","doc":"Component allowing selecting a credential or creating a new one via a modal.","ref":"LightningWeb.JobLive.CredentialPicker.html"},{"type":"function","title":"LightningWeb.JobLive.CredentialPicker.render/1","doc":"Attributes form ( :map ) (required) disabled ( :boolean ) - Defaults to false . credentials ( :list ) (required) on_change ( :any ) - Defaults to nil .","ref":"LightningWeb.JobLive.CredentialPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CronSetupComponent","doc":"A live component for managing cron setup in a form. The CronSetupComponent provides an interactive form for configuring cron settings. It includes fields for specifying the frequency, minute, hour, weekday, monthday, and cron expression. Usage Include the CronSetupComponent in your live view or template. Pass the necessary assigns to the component, such as form , on_change , and disabled . Handle the cron_expression_change event to capture changes in the form inputs. Use the updated cron_expression in your application logic.","ref":"LightningWeb.JobLive.CronSetupComponent.html"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.build_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#build_cron_expression/2"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.frequency_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#frequency_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.get_cron_data/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#get_cron_data/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.hour_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#hour_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.minute_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#minute_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.monthday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#monthday_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.render/1","doc":"Attributes id ( :string ) (required) form ( :map ) (required) on_change ( :any ) (required) disabled ( :boolean ) (required)","ref":"LightningWeb.JobLive.CronSetupComponent.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.time_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#time_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.weekday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#weekday_field/1"},{"type":"module","title":"LightningWeb.JobLive.JobBuilder","doc":"Job Builder Panel","ref":"LightningWeb.JobLive.JobBuilder.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.delete_title/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#delete_title/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.follow_run/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#follow_run/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.render/1","doc":"Attributes return_to ( :string ) (required) params ( :map ) - Defaults to %{} . can_edit_job ( :boolean ) (required) can_run_job ( :boolean ) (required)","ref":"LightningWeb.JobLive.JobBuilder.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_adaptor/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_adaptor/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_credential/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_credential/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.update_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#update_cron_expression/2"},{"type":"module","title":"LightningWeb.JobLive.JobBuilderComponents","doc":"","ref":"LightningWeb.JobLive.JobBuilderComponents.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.job_editor_component/1","doc":"Attributes adaptor ( :string ) (required) disabled ( :boolean ) - Defaults to false . source ( :string ) (required) change_event ( :string ) - Defaults to "job_body_changed" . Global attributes are accepted.","ref":"LightningWeb.JobLive.JobBuilderComponents.html#job_editor_component/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.trigger_picker/1","doc":"Attributes form ( :map ) (required) upstream_jobs ( :list ) (required) on_cron_change ( :any ) (required) disabled ( :boolean ) - Defaults to true .","ref":"LightningWeb.JobLive.JobBuilderComponents.html#trigger_picker/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.when_invalid/1","doc":"Attributes changeset ( :map ) (required) field ( :atom ) (required) Slots inner_block (required)","ref":"LightningWeb.JobLive.JobBuilderComponents.html#when_invalid/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.render/1","doc":"Attributes job ( :map ) (required) on_run ( :any ) (required) user ( :map ) (required) selected_dataclip_id ( :string ) (required)","ref":"LightningWeb.JobLive.ManualRunComponent.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.changeset/2","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html#changeset/2"},{"type":"module","title":"LightningWeb.LiveHelpers","doc":"General purpose LiveView helper functions","ref":"LightningWeb.LiveHelpers.html"},{"type":"function","title":"LightningWeb.LiveHelpers.display_short_uuid/1","doc":"","ref":"LightningWeb.LiveHelpers.html#display_short_uuid/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_in/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_in/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_out/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_out/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_error_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_error_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_info_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_info_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_nav_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_nav_block/1"},{"type":"module","title":"LightningWeb.ModalPortal","doc":"Component for rendering content inside layout without full DOM patch.","ref":"LightningWeb.ModalPortal.html"},{"type":"function","title":"LightningWeb.ModalPortal.close_modal/0","doc":"","ref":"LightningWeb.ModalPortal.html#close_modal/0"},{"type":"function","title":"LightningWeb.ModalPortal.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.ModalPortal.html#handle_event/3"},{"type":"function","title":"LightningWeb.ModalPortal.on_hide/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_hide/2"},{"type":"function","title":"LightningWeb.ModalPortal.on_show/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_show/2"},{"type":"function","title":"LightningWeb.ModalPortal.open_modal/2","doc":"","ref":"LightningWeb.ModalPortal.html#open_modal/2"},{"type":"function","title":"LightningWeb.ModalPortal.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ModalPortal.html#render/1"},{"type":"function","title":"LightningWeb.ModalPortal.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.ModalPortal.html#update/2"},{"type":"module","title":"LightningWeb.OauthCredentialHelper","doc":"A set of helper functions to encodes state and coordinate OAuth callbacks back to a LiveView component.","ref":"LightningWeb.OauthCredentialHelper.html"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast/2","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#broadcast/2"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast_forward/3","doc":"Broadcast a message specifically for forwarding a message to a component. It expects a subscription_id , the module of the component and opts being a keyword list containing an :id key of the specific component. See: Phoenix.LiveView.send_update/3 for more info. A corresponding LiveView (that is subscribed) is expected to have a matching handle_info/2 that looks like this: def handle_info ( { :forward , mod , opts } , socket ) do send_update ( mod , opts ) { :noreply , socket } end","ref":"LightningWeb.OauthCredentialHelper.html#broadcast_forward/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.build_state/3","doc":"Encode and encrypt the callback data which will be sent so a provider as the state key in the request. The values are: subscription_id The same ID used to subscribe. The component module The LiveView component that is going to receive update The component id The ID of the component","ref":"LightningWeb.OauthCredentialHelper.html#build_state/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.decode_state/1","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#decode_state/1"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.subscribe/1","doc":"Subscribe to the oauth_credential topic. It expects the a unique ID for the topic, usually the LiveView's socket.id .","ref":"LightningWeb.OauthCredentialHelper.html#subscribe/1"},{"type":"module","title":"LightningWeb.OidcController","doc":"","ref":"LightningWeb.OidcController.html"},{"type":"function","title":"LightningWeb.OidcController.new/2","doc":"Once the user has completed the authorization flow from above, they are returned here, and the authorization code is used to log them in.","ref":"LightningWeb.OidcController.html#new/2"},{"type":"function","title":"LightningWeb.OidcController.show/2","doc":"Given a known provider, redirect them to the authorize url on the provider","ref":"LightningWeb.OidcController.html#show/2"},{"type":"module","title":"LightningWeb.PageView","doc":"","ref":"LightningWeb.PageView.html"},{"type":"function","title":"LightningWeb.PageView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.PageView.html#__resource__/0"},{"type":"function","title":"LightningWeb.PageView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.PageView.html#render/2"},{"type":"function","title":"LightningWeb.PageView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.PageView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.Pagination","doc":"Pagination Components This has been extracted and adapted from scrivener_html . See: https://github.com/mgwidmann/scrivener_html","ref":"LightningWeb.Pagination.html"},{"type":"function","title":"LightningWeb.Pagination.page_link/1","doc":"","ref":"LightningWeb.Pagination.html#page_link/1"},{"type":"function","title":"LightningWeb.Pagination.pagination_bar/1","doc":"","ref":"LightningWeb.Pagination.html#pagination_bar/1"},{"type":"function","title":"LightningWeb.Pagination.raw_pagination_links/2","doc":"Returns the raw data in order to generate the proper HTML for pagination links. Data is returned in a {text, page_number} format where text is intended to be the text of the link and page_number is the page it should go to. Defaults are already supplied and they are as follows: [ distance : 5 , next : :next , previous : :previous , first : true , last : true , ellipsis : :ellipsis ] distance must be a positive non-zero integer or an exception is raised. next and previous should be strings but can be anything you want as long as it is truthy, falsey values will remove them from the output. first and last are only booleans, and they just include/remove their respective link from output. An example of the data returned: iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 10 , page_number : 5 } ) [ { "<<" , 4 } , { 1 , 1 } , { 2 , 2 } , { 3 , 3 } , { 4 , 4 } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { ">>" , 6 } ] iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 20 , page_number : 10 } , first : [ "←" ] , last : [ "→" ] ) [ { "<<" , 9 } , { [ "←" ] , 1 } , { :ellipsis , { :safe , "&hellip;" } } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { 11 , 11 } , { 12 , 12 } , { 13 , 13 } , { 14 , 14 } , { 15 , 15 } , { :ellipsis , { :safe , "&hellip;" } } , { [ "→" ] , 20 } , { ">>" , 11 } ] Simply loop and pattern match over each item and transform it to your custom HTML.","ref":"LightningWeb.Pagination.html#raw_pagination_links/2"},{"type":"module","title":"LightningWeb.Plugs.FirstSetup","doc":"Plug to redirect HTTP requests to /first_setup if there are no superusers in the system yet.","ref":"LightningWeb.Plugs.FirstSetup.html"},{"type":"module","title":"LightningWeb.ProfileLive.Edit","doc":"LiveView for user profile page.","ref":"LightningWeb.ProfileLive.Edit.html"},{"type":"function","title":"LightningWeb.ProfileLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProfileLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.FormComponent","doc":"Form component update profile email and password","ref":"LightningWeb.ProfileLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.MfaComponent","doc":"Component to enable MFA on a User's account","ref":"LightningWeb.ProfileLive.MfaComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.MfaComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.MfaComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.ProjectLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProjectLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProjectLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.ProjectLive.Index.html"},{"type":"function","title":"LightningWeb.ProjectLive.Index.delete_action/1","doc":"","ref":"LightningWeb.ProjectLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.ProjectLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Settings","doc":"Index Liveview for Runs","ref":"LightningWeb.ProjectLive.Settings.html"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.digest/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#digest/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.failure_alert/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#failure_alert/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Settings.html#render/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.role/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#role/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.user/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#user/1"},{"type":"module","title":"LightningWeb.RouteHelpers","doc":"Convenience functions for generating paths.","ref":"LightningWeb.RouteHelpers.html"},{"type":"function","title":"LightningWeb.RouteHelpers.oidc_callback_url/0","doc":"","ref":"LightningWeb.RouteHelpers.html#oidc_callback_url/0"},{"type":"function","title":"LightningWeb.RouteHelpers.show_run_url/2","doc":"","ref":"LightningWeb.RouteHelpers.html#show_run_url/2"},{"type":"module","title":"LightningWeb.Router","doc":"","ref":"LightningWeb.Router.html"},{"type":"function","title":"LightningWeb.Router.api/2","doc":"","ref":"LightningWeb.Router.html#api/2"},{"type":"function","title":"LightningWeb.Router.browser/2","doc":"","ref":"LightningWeb.Router.html#browser/2"},{"type":"function","title":"LightningWeb.Router.call/2","doc":"Callback invoked by Plug on every request.","ref":"LightningWeb.Router.html#call/2"},{"type":"function","title":"LightningWeb.Router.init/1","doc":"Callback required by Plug that initializes the router for serving web requests.","ref":"LightningWeb.Router.html#init/1"},{"type":"function","title":"LightningWeb.Router.storybook_assets/2","doc":"","ref":"LightningWeb.Router.html#storybook_assets/2"},{"type":"function","title":"LightningWeb.Router.storybook_browser/2","doc":"","ref":"LightningWeb.Router.html#storybook_browser/2"},{"type":"module","title":"LightningWeb.RunLive.Index","doc":"Index Liveview for Runs","ref":"LightningWeb.RunLive.Index.html"},{"type":"function","title":"LightningWeb.RunLive.Index.checked/2","doc":"","ref":"LightningWeb.RunLive.Index.html#checked/2"},{"type":"function","title":"LightningWeb.RunLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.RunLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.RunLive.RerunJobComponent","doc":"Rerun job component","ref":"LightningWeb.RunLive.RerunJobComponent.html"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#handle_event/3"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#render/1"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#update/2"},{"type":"module","title":"LightningWeb.RunLive.RunViewerLive","doc":"","ref":"LightningWeb.RunLive.RunViewerLive.html"},{"type":"function","title":"LightningWeb.RunLive.RunViewerLive.handle_info/2","doc":"Reload the run when any update messages arrive.","ref":"LightningWeb.RunLive.RunViewerLive.html#handle_info/2"},{"type":"module","title":"LightningWeb.RunLive.Show","doc":"Show page for individual runs.","ref":"LightningWeb.RunLive.Show.html"},{"type":"function","title":"LightningWeb.RunLive.Show.apply_action/3","doc":"","ref":"LightningWeb.RunLive.Show.html#apply_action/3"},{"type":"module","title":"LightningWeb.RunLive.WorkOrderComponent","doc":"Workorder component","ref":"LightningWeb.RunLive.WorkOrderComponent.html"},{"type":"function","title":"LightningWeb.RunLive.WorkOrderComponent.render/1","doc":"Attributes show_details ( :boolean ) - Defaults to false . entry_selected ( :boolean ) - Defaults to false .","ref":"LightningWeb.RunLive.WorkOrderComponent.html#render/1"},{"type":"module","title":"LightningWeb.Telemetry","doc":"Assorted metrics to collect during runtime. See https://hexdocs.pm/phoenix/telemetry.html","ref":"LightningWeb.Telemetry.html"},{"type":"function","title":"LightningWeb.Telemetry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"LightningWeb.Telemetry.html#child_spec/1"},{"type":"function","title":"LightningWeb.Telemetry.metrics/0","doc":"","ref":"LightningWeb.Telemetry.html#metrics/0"},{"type":"function","title":"LightningWeb.Telemetry.start_link/1","doc":"","ref":"LightningWeb.Telemetry.html#start_link/1"},{"type":"module","title":"LightningWeb.TokensLive.Index","doc":"LiveView for listing and managing tokens","ref":"LightningWeb.TokensLive.Index.html"},{"type":"function","title":"LightningWeb.TokensLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.TokensLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserAuth","doc":"The UserAuth controller.","ref":"LightningWeb.UserAuth.html"},{"type":"function","title":"LightningWeb.UserAuth.authenticate_bearer/2","doc":"","ref":"LightningWeb.UserAuth.html#authenticate_bearer/2"},{"type":"function","title":"LightningWeb.UserAuth.fetch_current_user/2","doc":"Authenticates the user by looking into the session and remember me token.","ref":"LightningWeb.UserAuth.html#fetch_current_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_in_user/2","doc":"Logs the user in by creating a new session token.","ref":"LightningWeb.UserAuth.html#log_in_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_out_user/1","doc":"Logs the user out. It clears all session data for safety. See renew_session.","ref":"LightningWeb.UserAuth.html#log_out_user/1"},{"type":"function","title":"LightningWeb.UserAuth.mark_totp_pending/1","doc":"","ref":"LightningWeb.UserAuth.html#mark_totp_pending/1"},{"type":"function","title":"LightningWeb.UserAuth.new_session/2","doc":"Assigns the token to a new session. It renews the session ID and clears the whole session to avoid fixation attacks. See the renew_session function to customize this behaviour. It also sets a :live_socket_id key in the session, so LiveView sessions are identified and automatically disconnected on log out. The line can be safely removed if you are not using LiveView.","ref":"LightningWeb.UserAuth.html#new_session/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_if_user_is_authenticated/2","doc":"Used for routes that require the user to not be authenticated.","ref":"LightningWeb.UserAuth.html#redirect_if_user_is_authenticated/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_with_return_to/2","doc":"Returns to or redirects to the dashboard and potentially set remember_me token.","ref":"LightningWeb.UserAuth.html#redirect_with_return_to/2"},{"type":"function","title":"LightningWeb.UserAuth.require_authenticated_user/2","doc":"Used for routes that require the user to be authenticated. If you want to enforce the user email is confirmed before they use the application at all, here would be a good place.","ref":"LightningWeb.UserAuth.html#require_authenticated_user/2"},{"type":"function","title":"LightningWeb.UserAuth.totp_pending?/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_pending?/1"},{"type":"function","title":"LightningWeb.UserAuth.totp_validated/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_validated/1"},{"type":"module","title":"LightningWeb.UserConfirmationController","doc":"","ref":"LightningWeb.UserConfirmationController.html"},{"type":"function","title":"LightningWeb.UserConfirmationController.confirm_email/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#confirm_email/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.create/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#create/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.edit/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#edit/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.new/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#new/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.update/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#update/2"},{"type":"module","title":"LightningWeb.UserLive.Edit","doc":"LiveView for editing a single job, which inturn uses LightningWeb.JobLive.BigFormComponent for common functionality.","ref":"LightningWeb.UserLive.Edit.html"},{"type":"function","title":"LightningWeb.UserLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.FormComponent","doc":"Form component for creating and editing users","ref":"LightningWeb.UserLive.FormComponent.html"},{"type":"function","title":"LightningWeb.UserLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.UserLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.Index","doc":"Index page for listing users","ref":"LightningWeb.UserLive.Index.html"},{"type":"function","title":"LightningWeb.UserLive.Index.delete_action/1","doc":"","ref":"LightningWeb.UserLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.UserLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserRegistrationController","doc":"","ref":"LightningWeb.UserRegistrationController.html"},{"type":"function","title":"LightningWeb.UserRegistrationController.create/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#create/2"},{"type":"function","title":"LightningWeb.UserRegistrationController.new/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#new/2"},{"type":"module","title":"LightningWeb.UserResetPasswordController","doc":"","ref":"LightningWeb.UserResetPasswordController.html"},{"type":"function","title":"LightningWeb.UserResetPasswordController.create/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#create/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.edit/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#edit/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.new/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#new/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.update/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#update/2"},{"type":"module","title":"LightningWeb.UserSessionController","doc":"","ref":"LightningWeb.UserSessionController.html"},{"type":"function","title":"LightningWeb.UserSessionController.auth_handler_url/0","doc":"","ref":"LightningWeb.UserSessionController.html#auth_handler_url/0"},{"type":"function","title":"LightningWeb.UserSessionController.create/2","doc":"","ref":"LightningWeb.UserSessionController.html#create/2"},{"type":"function","title":"LightningWeb.UserSessionController.delete/2","doc":"","ref":"LightningWeb.UserSessionController.html#delete/2"},{"type":"function","title":"LightningWeb.UserSessionController.exchange_token/2","doc":"","ref":"LightningWeb.UserSessionController.html#exchange_token/2"},{"type":"function","title":"LightningWeb.UserSessionController.new/2","doc":"","ref":"LightningWeb.UserSessionController.html#new/2"},{"type":"module","title":"LightningWeb.UserTOTPController","doc":"","ref":"LightningWeb.UserTOTPController.html"},{"type":"function","title":"LightningWeb.UserTOTPController.create/2","doc":"","ref":"LightningWeb.UserTOTPController.html#create/2"},{"type":"function","title":"LightningWeb.UserTOTPController.new/2","doc":"","ref":"LightningWeb.UserTOTPController.html#new/2"},{"type":"module","title":"LightningWeb.WebhooksController","doc":"","ref":"LightningWeb.WebhooksController.html"},{"type":"function","title":"LightningWeb.WebhooksController.create/2","doc":"","ref":"LightningWeb.WebhooksController.html#create/2"},{"type":"module","title":"LightningWeb.WorkflowLive.EditorPane","doc":"","ref":"LightningWeb.WorkflowLive.EditorPane.html"},{"type":"function","title":"LightningWeb.WorkflowLive.EditorPane.render/1","doc":"Attributes id ( :string ) (required) disabled ( :boolean ) - Defaults to false . class ( :string ) - Defaults to "" . on_change ( :any ) (required) adaptor ( :string ) (required) source ( :string ) (required) job_id ( :string ) (required)","ref":"LightningWeb.WorkflowLive.EditorPane.html#render/1"},{"type":"module","title":"LightningWeb.WorkflowLive.JobView","doc":"","ref":"LightningWeb.WorkflowLive.JobView.html"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.container/1","doc":"Attributes id ( :string ) (required) Slots top column - Accepts attributes: class ( :string ) - Extra CSS classes for the column. bottom","ref":"LightningWeb.WorkflowLive.JobView.html#container/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.input_pane/1","doc":"Attributes job ( :map ) (required) user ( :map ) (required) project ( :map ) (required) on_run ( :any ) (required) - Callback to run a job manually. can_run_job ( :boolean ) - Defaults to true .","ref":"LightningWeb.WorkflowLive.JobView.html#input_pane/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.job_edit_view/1","doc":"Attributes job ( :map ) (required) form ( :map ) (required) - A form built from a job. current_user ( :map ) (required) project ( :map ) (required) close_url ( :any ) (required) socket ( :any ) (required) on_run ( :any ) (required) - Callback to run a job manually. follow_run_id ( :any ) - Defaults to nil . Slots footer","ref":"LightningWeb.WorkflowLive.JobView.html#job_edit_view/1"},{"type":"module","title":"LightningWeb.WorkflowNewLive.WorkflowParams","doc":"Various function for reconciling changes to a workflow params map. The front end editor uses JSON patches to represent changes to the workflow.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_form_params/2","doc":"Produce a new set of params by applying the given form params to the current params.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_form_params/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_patches/2","doc":"Produce a new set of params by applying the given patches to the current parms","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_patches/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_map/1","doc":"Convert a changeset to a serializable map of workflow params, suitable for sending to the front end editor. It uses Lightning.Helpers.json_safe/1 to ensure that the map is safe to serialize to JSON. This is necessary because the underlying model may contain atom values.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_map/1"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_patches/2","doc":"Produce a set of patches that represent the difference between the initial params and the target params. This usually is used to produce a set of patches that represent the changes introduced by a changeset.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_patches/2"},{"type":"task","title":"Mix.Tasks.Lightning.GenEncryptionKey","doc":"Helper to generate a unique encryption key for Vault","ref":"Mix.Tasks.Lightning.GenEncryptionKey.html"},{"type":"task","title":"Mix.Tasks.Lightning.InstallRuntime","doc":"Installs the following NodeJS packages: core language-common","ref":"Mix.Tasks.Lightning.InstallRuntime.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.packages/0","doc":"","ref":"Mix.Tasks.Lightning.InstallRuntime.html#packages/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallRuntime.html#run/1"},{"type":"task","title":"Mix.Tasks.Lightning.InstallSchemas","doc":"Install the credential json schemas Use --exclude language-package1, language-package2 to exclude specific packages","ref":"Mix.Tasks.Lightning.InstallSchemas.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.fetch_schemas/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#fetch_schemas/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.parse_excluded/1","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#parse_excluded/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.persist_schema/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#persist_schema/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_options/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_params/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_chunk/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request!/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#run/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#start/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#stream_next/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.write_schema/3","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#write_schema/3"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.body/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:body/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.headers/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:headers/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.method/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:method/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.options/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:options/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.params/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:params/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.request/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:request/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.url/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:url/0"},{"type":"module","title":"ObanPruner","doc":"The Oban Pruner removes completed Oban jobs. It leaves everything else for manual inspection.","ref":"ObanPruner.html"},{"type":"function","title":"ObanPruner.perform/1","doc":"Deletes completed Oban jobs, leaving discarded for manual inspection.","ref":"ObanPruner.html#perform/1"},{"type":"module","title":"Storybook.Root","doc":"","ref":"Storybook.Root.html"},{"type":"module","title":"Lightning.Accounts","doc":"The Accounts context.","ref":"Lightning.Accounts.html"},{"type":"function","title":"Lightning.Accounts.apply_user_email/3","doc":"Emulates that the email will change without actually changing it in the database. Examples iex> apply_user_email ( user , "valid password" , %{ email : ... } ) { :ok , % User { } } role : :superuser iex> apply_user_email ( user , "invalid password" , %{ email : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#apply_user_email/3"},{"type":"function","title":"Lightning.Accounts.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Accounts.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Accounts.change_scheduled_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the user scheduled_deletion. Examples iex> change_scheduled_deletion ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_scheduled_deletion/2"},{"type":"function","title":"Lightning.Accounts.change_superuser_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking superuser changes. Examples iex> change_superuser_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_superuser_registration/1"},{"type":"function","title":"Lightning.Accounts.change_user_details/2","doc":"","ref":"Lightning.Accounts.html#change_user_details/2"},{"type":"function","title":"Lightning.Accounts.change_user_email/2","doc":"Returns an %Ecto.Changeset{} for changing the user email. Examples iex> change_user_email ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_email/2"},{"type":"function","title":"Lightning.Accounts.change_user_password/2","doc":"Returns an %Ecto.Changeset{} for changing the user password. Examples iex> change_user_password ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_password/2"},{"type":"function","title":"Lightning.Accounts.change_user_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking user changes. Examples iex> change_user_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_registration/1"},{"type":"function","title":"Lightning.Accounts.confirm_user/1","doc":"Confirms a user by the given token. If the token matches, the user account is marked as confirmed and the token is deleted.","ref":"Lightning.Accounts.html#confirm_user/1"},{"type":"function","title":"Lightning.Accounts.delete_auth_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_auth_token/1"},{"type":"function","title":"Lightning.Accounts.delete_session_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_session_token/1"},{"type":"function","title":"Lightning.Accounts.delete_token/1","doc":"Deletes a token. Examples iex> delete_token ( token ) { :ok , % UserToken { } } iex> delete_token ( token ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_token/1"},{"type":"function","title":"Lightning.Accounts.delete_user/1","doc":"Deletes a user. Examples iex> delete_user ( user ) { :ok , % User { } } iex> delete_user ( user ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_user/1"},{"type":"function","title":"Lightning.Accounts.delete_user_totp/1","doc":"Deletes the given user's TOTP","ref":"Lightning.Accounts.html#delete_user_totp/1"},{"type":"function","title":"Lightning.Accounts.deliver_update_email_instructions/3","doc":"Delivers the update email instructions to the given user. Examples iex> deliver_update_email_instructions ( user , current_email , & Routes . user_update_email_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_update_email_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/2","doc":"Delivers the confirmation email instructions to the given user. Examples iex> deliver_user_confirmation_instructions ( user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } } iex> deliver_user_confirmation_instructions ( confirmed_user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :error , :already_confirmed }","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/3","doc":"","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_reset_password_instructions/2","doc":"Delivers the reset password email to the given user. Examples iex> deliver_user_reset_password_instructions ( user , & Routes . user_reset_password_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_user_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.exchange_auth_token/1","doc":"Exchanges an auth token for a session token. The auth token is removed from the database if successful.","ref":"Lightning.Accounts.html#exchange_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_api_token/1","doc":"Generates an API token for a user.","ref":"Lightning.Accounts.html#generate_api_token/1"},{"type":"function","title":"Lightning.Accounts.generate_auth_token/1","doc":"Generates an auth token.","ref":"Lightning.Accounts.html#generate_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_user_session_token/1","doc":"Generates a session token.","ref":"Lightning.Accounts.html#generate_user_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_token!/1","doc":"Gets a single token. Raises Ecto.NoResultsError if the UserToken does not exist. Examples iex> get_token! ( 123 ) % UserToken { } iex> get_token! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_token!/1"},{"type":"function","title":"Lightning.Accounts.get_user!/1","doc":"Gets a single user. Raises Ecto.NoResultsError if the User does not exist. Examples iex> get_user! ( 123 ) % User { } iex> get_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_user!/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_api_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_api_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_auth_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_auth_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email/1","doc":"Gets a user by email. Examples iex> get_user_by_email ( "foo@example.com" ) % User { } iex> get_user_by_email ( "unknown@example.com" ) nil","ref":"Lightning.Accounts.html#get_user_by_email/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email_and_password/2","doc":"Gets a user by email and password. Examples iex> get_user_by_email_and_password ( "foo@example.com" , "correct_password" ) % User { } iex> get_user_by_email_and_password ( "foo@example.com" , "invalid_password" ) nil","ref":"Lightning.Accounts.html#get_user_by_email_and_password/2"},{"type":"function","title":"Lightning.Accounts.get_user_by_reset_password_token/1","doc":"Gets the user by reset password token. Examples iex> get_user_by_reset_password_token ( "validtoken" ) % User { } iex> get_user_by_reset_password_token ( "invalidtoken" ) nil","ref":"Lightning.Accounts.html#get_user_by_reset_password_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_session_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_totp/1","doc":"Gets a single UserTOTP if any exists.","ref":"Lightning.Accounts.html#get_user_totp/1"},{"type":"function","title":"Lightning.Accounts.get_users_to_alert_for_project/1","doc":"Gets all users to alert of workflow failure for a project","ref":"Lightning.Accounts.html#get_users_to_alert_for_project/1"},{"type":"function","title":"Lightning.Accounts.has_activity_in_projects?/1","doc":"","ref":"Lightning.Accounts.html#has_activity_in_projects?/1"},{"type":"function","title":"Lightning.Accounts.has_one_superuser?/0","doc":"Used to determine if there is at least one Superuser in the system. This triggers the setup page on fresh installs.","ref":"Lightning.Accounts.html#has_one_superuser?/0"},{"type":"function","title":"Lightning.Accounts.list_api_tokens/1","doc":"Lists all user tokens","ref":"Lightning.Accounts.html#list_api_tokens/1"},{"type":"function","title":"Lightning.Accounts.list_users/0","doc":"Returns the list of users. Examples iex> list_users ( ) [ % User { } , ... ]","ref":"Lightning.Accounts.html#list_users/0"},{"type":"function","title":"Lightning.Accounts.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find users that are ready for permanent deletion and purge them.","ref":"Lightning.Accounts.html#perform/1"},{"type":"function","title":"Lightning.Accounts.purge_user/1","doc":"","ref":"Lightning.Accounts.html#purge_user/1"},{"type":"function","title":"Lightning.Accounts.register_superuser/1","doc":"Registers a superuser. Examples iex> register_superuser ( %{ field : value } ) { :ok , % User { } } iex> register_superuser ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_superuser/1"},{"type":"function","title":"Lightning.Accounts.register_user/1","doc":"Registers a user. Examples iex> register_user ( %{ field : value } ) { :ok , % User { } } iex> register_user ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_user/1"},{"type":"function","title":"Lightning.Accounts.reset_user_password/2","doc":"Resets the user password. Examples iex> reset_user_password ( user , %{ password : "new long password" , password_confirmation : "new long password" } ) { :ok , % User { } } iex> reset_user_password ( user , %{ password : "valid" , password_confirmation : "not the same" } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#reset_user_password/2"},{"type":"function","title":"Lightning.Accounts.schedule_user_deletion/2","doc":"Given a user and a confirmation email, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic user purge cronjob will never run. (Note that subsequent logins will be blocked for users pending deletion.)","ref":"Lightning.Accounts.html#schedule_user_deletion/2"},{"type":"function","title":"Lightning.Accounts.update_user_details/2","doc":"","ref":"Lightning.Accounts.html#update_user_details/2"},{"type":"function","title":"Lightning.Accounts.update_user_email/2","doc":"Updates the user email using the given token. If the token matches, the user email is updated and the token is deleted. The confirmed_at date is also updated to the current time.","ref":"Lightning.Accounts.html#update_user_email/2"},{"type":"function","title":"Lightning.Accounts.update_user_password/3","doc":"Updates the user password. Examples iex> update_user_password ( user , "valid password" , %{ password : ... } ) { :ok , % User { } } iex> update_user_password ( user , "invalid password" , %{ password : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#update_user_password/3"},{"type":"function","title":"Lightning.Accounts.upsert_user_totp/2","doc":"Updates or Inserts the user's TOTP","ref":"Lightning.Accounts.html#upsert_user_totp/2"},{"type":"function","title":"Lightning.Accounts.valid_user_totp?/2","doc":"Validates if the given TOTP code is valid.","ref":"Lightning.Accounts.html#valid_user_totp?/2"},{"type":"function","title":"Lightning.Accounts.validate_change_user_email/2","doc":"","ref":"Lightning.Accounts.html#validate_change_user_email/2"},{"type":"module","title":"Lightning.Accounts.User","doc":"The User model.","ref":"Lightning.Accounts.User.html"},{"type":"function","title":"Lightning.Accounts.User.confirm_changeset/1","doc":"Confirms the account by setting confirmed_at .","ref":"Lightning.Accounts.User.html#confirm_changeset/1"},{"type":"function","title":"Lightning.Accounts.User.details_changeset/2","doc":"A user changeset for user details: email first_name last_name role","ref":"Lightning.Accounts.User.html#details_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.email_changeset/2","doc":"A user changeset for changing the email. It requires the email to change otherwise an error is added.","ref":"Lightning.Accounts.User.html#email_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.password_changeset/3","doc":"A user changeset for changing the password. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#password_changeset/3"},{"type":"function","title":"Lightning.Accounts.User.scheduled_deletion_changeset/2","doc":"A user changeset for changing the scheduled_deletion property.","ref":"Lightning.Accounts.User.html#scheduled_deletion_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.superuser_registration_changeset/2","doc":"A superuser changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#superuser_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.user_registration_changeset/2","doc":"A user changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#user_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.valid_password?/2","doc":"Verifies the password. If there is no user or the user doesn't have a password, we call Bcrypt.no_user_verify/0 to avoid timing attacks.","ref":"Lightning.Accounts.User.html#valid_password?/2"},{"type":"function","title":"Lightning.Accounts.User.validate_current_password/2","doc":"Validates the current password otherwise adds an error to the changeset.","ref":"Lightning.Accounts.User.html#validate_current_password/2"},{"type":"type","title":"Lightning.Accounts.User.t/0","doc":"","ref":"Lightning.Accounts.User.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.User.RolesEnum","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.create_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Accounts.User.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schema/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Accounts.User.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Accounts.User.RolesEnum.t/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserNotifier","doc":"The UserNotifier module.","ref":"Lightning.Accounts.UserNotifier.html"},{"type":"function","title":"Lightning.Accounts.UserNotifier.build_digest_url/3","doc":"","ref":"Lightning.Accounts.UserNotifier.html#build_digest_url/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/2","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/3","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_addition_notification/2","doc":"Deliver email to notify user of his addition of a project.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_addition_notification/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_digest/2","doc":"Deliver a project digest of daily/weekly or monthly activity to a user.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_digest/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_reset_password_instructions/2","doc":"Deliver instructions to reset a user password.","ref":"Lightning.Accounts.UserNotifier.html#deliver_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_instructions/2","doc":"Deliver instructions to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_warning/2","doc":"Deliver warning to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_warning/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.notify_project_deletion/2","doc":"","ref":"Lightning.Accounts.UserNotifier.html#notify_project_deletion/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.send_deletion_notification_email/1","doc":"Deliver an email to notify the user about their account being deleted","ref":"Lightning.Accounts.UserNotifier.html#send_deletion_notification_email/1"},{"type":"module","title":"Lightning.Accounts.UserTOTP","doc":"User Time based OTPs schema","ref":"Lightning.Accounts.UserTOTP.html"},{"type":"function","title":"Lightning.Accounts.UserTOTP.changeset/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserTOTP.valid_totp?/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#valid_totp?/2"},{"type":"type","title":"Lightning.Accounts.UserTOTP.t/0","doc":"","ref":"Lightning.Accounts.UserTOTP.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserToken","doc":"The UserToken model. The reason why we store session tokens in the database, even though Phoenix already provides a session cookie, is because Phoenix' default session cookies are not persisted, they are simply signed and potentially encrypted. This means they are valid indefinitely, unless you change the signing/encryption salt. Therefore, storing them allows individual user sessions to be expired. The token system can also be extended to store additional data, such as the device used for logging in. You could then use this information to display all valid sessions and devices in the UI and allow users to explicitly expire any session they deem invalid.","ref":"Lightning.Accounts.UserToken.html"},{"type":"function","title":"Lightning.Accounts.UserToken.build_email_token/3","doc":"Builds a token and its hash to be delivered to the user's email. The non-hashed token is sent to the user email while the hashed part is stored in the database. The original token cannot be reconstructed, which means anyone with read-only access to the database cannot directly use the token in the application to gain access. Furthermore, if the user changes their email in the system, the tokens sent to the previous email are no longer valid. Users can easily adapt the existing code to provide other types of delivery methods, for example, by phone numbers.","ref":"Lightning.Accounts.UserToken.html#build_email_token/3"},{"type":"function","title":"Lightning.Accounts.UserToken.build_token/2","doc":"Generates a token that will be stored in a signed place, such as session or cookie. As they are signed, those tokens do not need to be hashed.","ref":"Lightning.Accounts.UserToken.html#build_token/2"},{"type":"function","title":"Lightning.Accounts.UserToken.changeset/2","doc":"","ref":"Lightning.Accounts.UserToken.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign/2","doc":"Combines generate_claims/1 and encode_and_sign/2","ref":"Lightning.Accounts.UserToken.html#generate_and_sign/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign!/2","doc":"Same as generate_and_sign/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#generate_and_sign!/2"},{"type":"function","title":"Lightning.Accounts.UserToken.last_used_changeset/1","doc":"Update when the api token was last used by setting last_used_at .","ref":"Lightning.Accounts.UserToken.html#last_used_changeset/1"},{"type":"function","title":"Lightning.Accounts.UserToken.token_and_context_query/2","doc":"Returns the token struct for the given token value and context.","ref":"Lightning.Accounts.UserToken.html#token_and_context_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.user_and_contexts_query/2","doc":"Gets all tokens for the given user for the given contexts.","ref":"Lightning.Accounts.UserToken.html#user_and_contexts_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate/3","doc":"Combines verify/2 and validate/2","ref":"Lightning.Accounts.UserToken.html#verify_and_validate/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate!/3","doc":"Same as verify_and_validate/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#verify_and_validate!/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_change_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. This is used to validate requests to change the user email. It is different from verify_email_token_query/2 precisely because verify_email_token_query/2 validates the email has not changed, which is the starting point by this function. The given token is valid if it matches its hashed counterpart in the database and if it has not expired (after @change_email_validity_in_days). The context must always start with "change:".","ref":"Lightning.Accounts.UserToken.html#verify_change_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The given token is valid if it matches its hashed counterpart in the database and the user email has not changed. This function also checks if the token is being used within a certain period, depending on the context. The default contexts supported by this function are either "confirm", for account confirmation emails, and "reset_password", for resetting the password. For verifying requests to change the email, see verify_change_email_token_query/2 .","ref":"Lightning.Accounts.UserToken.html#verify_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The token is valid if it matches the value in the database and it has not expired (after @auth_validity_in_seconds or @session_validity_in_days).","ref":"Lightning.Accounts.UserToken.html#verify_token_query/2"},{"type":"module","title":"Lightning.Credentials","doc":"The Credentials context.","ref":"Lightning.Credentials.html"},{"type":"function","title":"Lightning.Credentials.change_credential/2","doc":"Returns an %Ecto.Changeset{} for tracking credential changes. Examples iex> change_credential ( credential ) % Ecto.Changeset { data : % Credential { } }","ref":"Lightning.Credentials.html#change_credential/2"},{"type":"function","title":"Lightning.Credentials.create_credential/1","doc":"Creates a credential. Examples iex> create_credential ( %{ field : value } ) { :ok , % Credential { } } iex> create_credential ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#create_credential/1"},{"type":"function","title":"Lightning.Credentials.delete_credential/1","doc":"Deletes a credential. Examples iex> delete_credential ( credential ) { :ok , % Credential { } } iex> delete_credential ( credential ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#delete_credential/1"},{"type":"function","title":"Lightning.Credentials.get_credential!/1","doc":"Gets a single credential. Raises Ecto.NoResultsError if the Credential does not exist. Examples iex> get_credential! ( 123 ) % Credential { } iex> get_credential! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Credentials.html#get_credential!/1"},{"type":"function","title":"Lightning.Credentials.invalid_projects_for_user/2","doc":"Given a credential and a user, returns a list of invalid projects—i.e., those that the credential is shared with but that the user does not have access to. This is used to generate a validation error when a credential cannot be transferred. Examples iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ ] iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ "52ea8758-6ce5-43d7-912f-6a1e1f11dc55" ]","ref":"Lightning.Credentials.html#invalid_projects_for_user/2"},{"type":"function","title":"Lightning.Credentials.list_credentials/0","doc":"Returns the list of credentials. Examples iex> list_credentials ( ) [ % Credential { } , ... ]","ref":"Lightning.Credentials.html#list_credentials/0"},{"type":"function","title":"Lightning.Credentials.list_credentials/1","doc":"","ref":"Lightning.Credentials.html#list_credentials/1"},{"type":"function","title":"Lightning.Credentials.list_credentials_for_user/1","doc":"Returns the list of credentials for a given user. Examples iex> list_credentials_for_user ( 123 ) [ % Credential { user_id : 123 } , % Credential { user_id : 123 } , ... ]","ref":"Lightning.Credentials.html#list_credentials_for_user/1"},{"type":"function","title":"Lightning.Credentials.maybe_refresh_token/1","doc":"","ref":"Lightning.Credentials.html#maybe_refresh_token/1"},{"type":"function","title":"Lightning.Credentials.sensitive_values_for/1","doc":"","ref":"Lightning.Credentials.html#sensitive_values_for/1"},{"type":"function","title":"Lightning.Credentials.update_credential/2","doc":"Updates a credential. Examples iex> update_credential ( credential , %{ field : new_value } ) { :ok , % Credential { } } iex> update_credential ( credential , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#update_credential/2"},{"type":"module","title":"Lightning.Credentials.Audit","doc":"Model for storing changes to Credentials","ref":"Lightning.Credentials.Audit.html"},{"type":"function","title":"Lightning.Credentials.Audit.event/4","doc":"","ref":"Lightning.Credentials.Audit.html#event/4"},{"type":"function","title":"Lightning.Credentials.Audit.save/1","doc":"","ref":"Lightning.Credentials.Audit.html#save/1"},{"type":"module","title":"Lightning.Credentials.Credential","doc":"The Credential model.","ref":"Lightning.Credentials.Credential.html"},{"type":"type","title":"Lightning.Credentials.Credential.t/0","doc":"","ref":"Lightning.Credentials.Credential.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.Schema","doc":"Structure that can parse JsonSchemas (using ExJsonSchema ) and validate changesets for a given schema.","ref":"Lightning.Credentials.Schema.html"},{"type":"function","title":"Lightning.Credentials.Schema.new/2","doc":"","ref":"Lightning.Credentials.Schema.html#new/2"},{"type":"function","title":"Lightning.Credentials.Schema.properties/2","doc":"","ref":"Lightning.Credentials.Schema.html#properties/2"},{"type":"function","title":"Lightning.Credentials.Schema.required?/2","doc":"","ref":"Lightning.Credentials.Schema.html#required?/2"},{"type":"function","title":"Lightning.Credentials.Schema.validate/2","doc":"","ref":"Lightning.Credentials.Schema.html#validate/2"},{"type":"type","title":"Lightning.Credentials.Schema.t/0","doc":"","ref":"Lightning.Credentials.Schema.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.SchemaDocument","doc":"Provides facilities to dynamically create and validate a changeset for a given Schema","ref":"Lightning.Credentials.SchemaDocument.html"},{"type":"function","title":"Lightning.Credentials.SchemaDocument.changeset/3","doc":"","ref":"Lightning.Credentials.SchemaDocument.html#changeset/3"},{"type":"module","title":"Lightning.Credentials.SensitiveValues","doc":"Functions to pull out sensitive values inside a credential. These values are used to scrub logs for leaked secrets.","ref":"Lightning.Credentials.SensitiveValues.html"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.flatten_map/1","doc":"Keys that are not considered sensitive","ref":"Lightning.Credentials.SensitiveValues.html#flatten_map/1"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.secret_values/1","doc":"Given a map, find all values allowed (via @safe_keys ) and return them as a list.","ref":"Lightning.Credentials.SensitiveValues.html#secret_values/1"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:pairs/0"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.raw_pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:raw_pairs/0"},{"type":"module","title":"Lightning.Invocation","doc":"The Invocation context.","ref":"Lightning.Invocation.html"},{"type":"function","title":"Lightning.Invocation.change_dataclip/2","doc":"Returns an %Ecto.Changeset{} for tracking dataclip changes. Examples iex> change_dataclip ( dataclip ) % Ecto.Changeset { data : % Dataclip { } }","ref":"Lightning.Invocation.html#change_dataclip/2"},{"type":"function","title":"Lightning.Invocation.change_run/2","doc":"Returns an %Ecto.Changeset{} for tracking run changes. Examples iex> change_run ( run ) % Ecto.Changeset { data : % Run { } }","ref":"Lightning.Invocation.html#change_run/2"},{"type":"function","title":"Lightning.Invocation.create_dataclip/1","doc":"Creates a dataclip. Examples iex> create_dataclip ( %{ field : value } ) { :ok , % Dataclip { } } iex> create_dataclip ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_dataclip/1"},{"type":"function","title":"Lightning.Invocation.create_log_line/2","doc":"","ref":"Lightning.Invocation.html#create_log_line/2"},{"type":"function","title":"Lightning.Invocation.create_run/1","doc":"Creates a run. Examples iex> create_run ( %{ field : value } ) { :ok , % Run { } } iex> create_run ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_run/1"},{"type":"function","title":"Lightning.Invocation.delete_dataclip/1","doc":"Deletes a dataclip. Examples iex> delete_dataclip ( dataclip ) { :ok , % Dataclip { } } iex> delete_dataclip ( dataclip ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_dataclip/1"},{"type":"function","title":"Lightning.Invocation.delete_run/1","doc":"Deletes a run. Examples iex> delete_run ( run ) { :ok , % Run { } } iex> delete_run ( run ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_run/1"},{"type":"function","title":"Lightning.Invocation.filter_run_body_and_logs_where/2","doc":"","ref":"Lightning.Invocation.html#filter_run_body_and_logs_where/2"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_before_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_status_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_status_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workflow_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workflow_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_before_where/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip/1","doc":"Gets a single dataclip given one of: a Dataclip uuid a Run model Returns nil if the Dataclip does not exist. Examples iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) % Dataclip { } iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) nil iex> get_dataclip ( % Run { id : "a uuid" } ) % Dataclip { }","ref":"Lightning.Invocation.html#get_dataclip/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip!/1","doc":"Gets a single dataclip. Raises Ecto.NoResultsError if the Dataclip does not exist. Examples iex> get_dataclip! ( 123 ) % Dataclip { } iex> get_dataclip! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_dataclip!/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip_query/1","doc":"Query for retrieving the dataclip that a runs starting dataclip.","ref":"Lightning.Invocation.html#get_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_result_dataclip_query/1","doc":"Query for retrieving the dataclip that was the result of a successful run.","ref":"Lightning.Invocation.html#get_result_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_run!/1","doc":"Gets a single run. Raises Ecto.NoResultsError if the Run does not exist. Examples iex> get_run! ( 123 ) % Run { } iex> get_run! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_run!/1"},{"type":"function","title":"Lightning.Invocation.get_run_with_job!/1","doc":"Fetches a run and preloads the job via the run's event.","ref":"Lightning.Invocation.html#get_run_with_job!/1"},{"type":"function","title":"Lightning.Invocation.get_workorders_by_ids/1","doc":"","ref":"Lightning.Invocation.html#get_workorders_by_ids/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips/0","doc":"Returns the list of dataclips. Examples iex> list_dataclips ( ) [ % Dataclip { } , ... ]","ref":"Lightning.Invocation.html#list_dataclips/0"},{"type":"function","title":"Lightning.Invocation.list_dataclips/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_for_job/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_for_job/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_query/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_query/1"},{"type":"function","title":"Lightning.Invocation.list_runs/0","doc":"Returns the list of runs. Examples iex> list_runs ( ) [ % Run { } , ... ]","ref":"Lightning.Invocation.html#list_runs/0"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project/2","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project/2"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project_query/1","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project_query/1"},{"type":"function","title":"Lightning.Invocation.list_work_orders_for_project_query/2","doc":"","ref":"Lightning.Invocation.html#list_work_orders_for_project_query/2"},{"type":"function","title":"Lightning.Invocation.search_workorders/1","doc":"","ref":"Lightning.Invocation.html#search_workorders/1"},{"type":"function","title":"Lightning.Invocation.search_workorders/3","doc":"","ref":"Lightning.Invocation.html#search_workorders/3"},{"type":"function","title":"Lightning.Invocation.update_dataclip/2","doc":"Updates a dataclip. Examples iex> update_dataclip ( dataclip , %{ field : new_value } ) { :ok , % Dataclip { } } iex> update_dataclip ( dataclip , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_dataclip/2"},{"type":"function","title":"Lightning.Invocation.update_run/2","doc":"Updates a run. Examples iex> update_run ( run , %{ field : new_value } ) { :ok , % Run { } } iex> update_run ( run , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_run/2"},{"type":"function","title":"Lightning.Invocation.with_attempts/1","doc":"","ref":"Lightning.Invocation.html#with_attempts/1"},{"type":"module","title":"Lightning.Invocation.Dataclip","doc":"Ecto model for Dataclips. Dataclips represent some data that arrived in the system, and records both the data and the source of the data. Types :http_request The data arrived via a webhook. :global Was created manually, and is intended to be used multiple times. When repetitive static data is needed to be maintained, instead of hard-coding into a Job - a more convenient solution is to create a :global Dataclip and access it inside the Job. :run_result The final state of a successful run. :saved_input An arbitrary input, created by a user. (Only configuration will be overwritten.)","ref":"Lightning.Invocation.Dataclip.html"},{"type":"function","title":"Lightning.Invocation.Dataclip.get_types/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#get_types/0"},{"type":"function","title":"Lightning.Invocation.Dataclip.new/1","doc":"","ref":"Lightning.Invocation.Dataclip.html#new/1"},{"type":"function","title":"Lightning.Invocation.Dataclip.validate_by_type/1","doc":"Append validations based on the type of the Dataclip. :run_result must have an associated Run model.","ref":"Lightning.Invocation.Dataclip.html#validate_by_type/1"},{"type":"type","title":"Lightning.Invocation.Dataclip.source_type/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:source_type/0"},{"type":"type","title":"Lightning.Invocation.Dataclip.t/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.LogLine","doc":"Ecto model for run logs.","ref":"Lightning.Invocation.LogLine.html"},{"type":"function","title":"Lightning.Invocation.LogLine.validate/1","doc":"","ref":"Lightning.Invocation.LogLine.html#validate/1"},{"type":"type","title":"Lightning.Invocation.LogLine.t/0","doc":"","ref":"Lightning.Invocation.LogLine.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.Query","doc":"Query functions for working with Runs and Dataclips","ref":"Lightning.Invocation.Query.html"},{"type":"function","title":"Lightning.Invocation.Query.last_run_for_job/1","doc":"The last run for a job","ref":"Lightning.Invocation.Query.html#last_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.last_successful_run_for_job/1","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#last_successful_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_for/1","doc":"Runs for a specific user","ref":"Lightning.Invocation.Query.html#runs_for/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_with_code/2","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#runs_with_code/2"},{"type":"module","title":"Lightning.Invocation.Run","doc":"Ecto model for Runs. A run represents the work initiated for a Job with an input dataclip. Once completed (successfully) it will have an output_dataclip associated with it as well.","ref":"Lightning.Invocation.Run.html"},{"type":"function","title":"Lightning.Invocation.Run.new/1","doc":"","ref":"Lightning.Invocation.Run.html#new/1"},{"type":"function","title":"Lightning.Invocation.Run.new_from/1","doc":"Creates a new Run changeset, but copies over certain fields. This is used to create new runs for retrys.","ref":"Lightning.Invocation.Run.html#new_from/1"},{"type":"type","title":"Lightning.Invocation.Run.t/0","doc":"","ref":"Lightning.Invocation.Run.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReason","doc":"Ecto model for InvocationReasons.","ref":"Lightning.InvocationReason.html"},{"type":"function","title":"Lightning.InvocationReason.new/1","doc":"","ref":"Lightning.InvocationReason.html#new/1"},{"type":"function","title":"Lightning.InvocationReason.validate_by_trigger_type/1","doc":"","ref":"Lightning.InvocationReason.html#validate_by_trigger_type/1"},{"type":"type","title":"Lightning.InvocationReason.source_type/0","doc":"","ref":"Lightning.InvocationReason.html#t:source_type/0"},{"type":"type","title":"Lightning.InvocationReason.t/0","doc":"","ref":"Lightning.InvocationReason.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReasons","doc":"The InvocationReasons context.","ref":"Lightning.InvocationReasons.html"},{"type":"function","title":"Lightning.InvocationReasons.build/2","doc":"","ref":"Lightning.InvocationReasons.html#build/2"},{"type":"function","title":"Lightning.InvocationReasons.create_reason/1","doc":"Creates a reason. Examples iex> create_reason ( %{ field : value } ) { :ok , % InvocationReason { } } iex> create_reason ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.InvocationReasons.html#create_reason/1"},{"type":"type","title":"Lightning.InvocationReasons.reason_type/0","doc":"","ref":"Lightning.InvocationReasons.html#t:reason_type/0"},{"type":"module","title":"Lightning.Pipeline","doc":"Service class to coordinate the running of jobs, and their downstream jobs.","ref":"Lightning.Pipeline.html"},{"type":"function","title":"Lightning.Pipeline.assemble_logs_for_run/1","doc":"Return all logs for a run as a string of text, separated by new line breaks","ref":"Lightning.Pipeline.html#assemble_logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.logs_for_run/1","doc":"Return all logs for a run as a list","ref":"Lightning.Pipeline.html#logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.process/1","doc":"","ref":"Lightning.Pipeline.html#process/1"},{"type":"module","title":"Lightning.Pipeline.Runner","doc":"Job running entrypoint","ref":"Lightning.Pipeline.Runner.html"},{"type":"function","title":"Lightning.Pipeline.Runner.create_dataclip_from_result/2","doc":"Creates a dataclip linked to the run that just finished. If either the file doesn't exist or there is a JSON decoding error, it logs and returns an error tuple.","ref":"Lightning.Pipeline.Runner.html#create_dataclip_from_result/2"},{"type":"function","title":"Lightning.Pipeline.Runner.find_or_install_adaptor/1","doc":"Make sure an adaptor matching the name is available. If it is available, return it's Engine.Adaptor struct - if not then install it.","ref":"Lightning.Pipeline.Runner.html#find_or_install_adaptor/1"},{"type":"function","title":"Lightning.Pipeline.Runner.scrub_result/1","doc":"Scrubs values from all keys in configuration, will be replaced by extensions to scrubber.ex, which is currently only used for logs.","ref":"Lightning.Pipeline.Runner.html#scrub_result/1"},{"type":"function","title":"Lightning.Pipeline.Runner.start/2","doc":"Execute a Run. Given a valid run: Persist the Dataclip and the Job's body to disk Create a blank output file on disk Build up a %Lightning.Runtime.Runspec{} with the paths, and adaptor module name And start it via Handler.start/2 . The callbacks implemented on Handler ( c:Handler.on_start/1 and c:Handler.on_finish/2 ) update the run when a Run is started and when it's finished, attaching the exit_code and log when they are available.","ref":"Lightning.Pipeline.Runner.html#start/2"},{"type":"module","title":"Lightning.Pipeline.Runner.Handler","doc":"Custom handler callbacks for Lightnings use of Engine to execute runs.","ref":"Lightning.Pipeline.Runner.Handler.html"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.on_start/1","doc":"The on_start handler updates the run, setting the started_at time and stamping the run with the ID of the credential that was used, if any, to facilitate easier auditing.","ref":"Lightning.Pipeline.Runner.Handler.html#on_start/1"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.stop/1","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#stop/1"},{"type":"type","title":"Lightning.Pipeline.Runner.Handler.handler_opts/0","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#t:handler_opts/0"},{"type":"module","title":"Lightning.Pipeline.StateAssembler","doc":"Facilities for building the state for a Run How state is assembled For the most common jobs, an inbound webhook will store an :http_request type dataclip. The reason that is created is associated with the dataclip. At runtime, the initial state for a Run will be in the shape of: { "data" : < the dataclip > , "configuration" : < the job ' s credential > } Saved inputs Saved custom inputs will only have state.configuration changed, everything else will remain as displayed. Flow Jobs When a Job is triggered by a previous Jobs success or failure these are the rules for constructing that Jobs state: For jobs that trigger on it's upstream jobs failure, the event will have the previous runs input dataclip as its input dataclip. The state will also have the log of the previous run attached on the error key. For Jobs triggered by a previous success, the run will have the previous runs output dataclip as its input dataclip. :run_result dataclips are expected to already have a data key, and are merged into the root.","ref":"Lightning.Pipeline.StateAssembler.html"},{"type":"function","title":"Lightning.Pipeline.StateAssembler.assemble/1","doc":"Assemble state for use in a Run.","ref":"Lightning.Pipeline.StateAssembler.html#assemble/1"},{"type":"module","title":"Lightning.Jobs","doc":"The Jobs context.","ref":"Lightning.Jobs.html"},{"type":"function","title":"Lightning.Jobs.change_job/2","doc":"Returns an %Ecto.Changeset{} for tracking job changes. Examples iex> change_job ( job ) % Ecto.Changeset { data : % Job { } }","ref":"Lightning.Jobs.html#change_job/2"},{"type":"function","title":"Lightning.Jobs.create_job/1","doc":"Creates a job. Examples iex> create_job ( %{ field : value } ) { :ok , % Job { } } iex> create_job ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#create_job/1"},{"type":"function","title":"Lightning.Jobs.delete_job/1","doc":"Deletes a job. Examples iex> delete_job ( job ) { :ok , % Job { } } iex> delete_job ( job ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#delete_job/1"},{"type":"function","title":"Lightning.Jobs.get_downstream_jobs_for/2","doc":"Returns the list of downstream jobs for a given job, optionally matching a specific trigger type. When downstream_jobs_for is called without a trigger that means its between jobs when it called with a trigger that means we are starting from outside the pipeline","ref":"Lightning.Jobs.html#get_downstream_jobs_for/2"},{"type":"function","title":"Lightning.Jobs.get_job/1","doc":"","ref":"Lightning.Jobs.html#get_job/1"},{"type":"function","title":"Lightning.Jobs.get_job!/1","doc":"Gets a single job. Raises Ecto.NoResultsError if the Job does not exist. Examples iex> get_job! ( 123 ) % Job { } iex> get_job! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Jobs.html#get_job!/1"},{"type":"function","title":"Lightning.Jobs.get_upstream_jobs_for/1","doc":"Returns the list of jobs excluding the one given.","ref":"Lightning.Jobs.html#get_upstream_jobs_for/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project_query/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project_query/1"},{"type":"function","title":"Lightning.Jobs.list_active_cron_jobs/0","doc":"","ref":"Lightning.Jobs.html#list_active_cron_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs/0","doc":"Returns the list of jobs.","ref":"Lightning.Jobs.html#list_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs_for_workflow/1","doc":"","ref":"Lightning.Jobs.html#list_jobs_for_workflow/1"},{"type":"function","title":"Lightning.Jobs.update_job/2","doc":"Updates a job. Examples iex> update_job ( job , %{ field : new_value } ) { :ok , % Job { } } iex> update_job ( job , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#update_job/2"},{"type":"module","title":"Lightning.Jobs.Job","doc":"Ecto model for Jobs. A Job contains the fields for defining a job. body The expression/javascript code name A plain text identifier adaptor An NPM style string that contains both the module name and it's version. E.g. @openfn/language-http@v1.2.3 or @openfn/language-foo@latest . While the version suffix isn't enforced here as it's not strictly necessary in this context, the front end will ensure a version is stated ( @latest being the default).","ref":"Lightning.Jobs.Job.html"},{"type":"function","title":"Lightning.Jobs.Job.new/1","doc":"","ref":"Lightning.Jobs.Job.html#new/1"},{"type":"function","title":"Lightning.Jobs.Job.put_project_credential/2","doc":"","ref":"Lightning.Jobs.Job.html#put_project_credential/2"},{"type":"function","title":"Lightning.Jobs.Job.put_workflow/2","doc":"Attaches a workflow to a job, this is useful when you have an unpersisted Workflow changeset - and want it to be created at the same time as a Job. Example: workflow = Ecto.Changeset . cast ( % Lightning.Workflows.Workflow { } , %{ "project_id" => attrs [ :project_id ] , "id" => Ecto.UUID . generate ( ) } , [ :project_id , :id ] ) job = % Job { } |> Ecto.Changeset . change ( ) |> Job . put_workflow ( workflow ) |> Job . changeset ( attrs )","ref":"Lightning.Jobs.Job.html#put_workflow/2"},{"type":"function","title":"Lightning.Jobs.Job.validate/1","doc":"","ref":"Lightning.Jobs.Job.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Job.t/0","doc":"","ref":"Lightning.Jobs.Job.html#t:t/0"},{"type":"module","title":"Lightning.Jobs.Query","doc":"Query module for finding Jobs.","ref":"Lightning.Jobs.Query.html"},{"type":"function","title":"Lightning.Jobs.Query.enabled_cron_jobs_by_edge/0","doc":"Returns active jobs with their cron triggers for use in the cron scheduling service.","ref":"Lightning.Jobs.Query.html#enabled_cron_jobs_by_edge/0"},{"type":"function","title":"Lightning.Jobs.Query.jobs_for/1","doc":"Returns all jobs accessible to a user, via their projects or all jobs in a given project.","ref":"Lightning.Jobs.Query.html#jobs_for/1"},{"type":"module","title":"Lightning.Jobs.Scheduler","doc":"The Scheduler is responsible for finding jobs that are ready to run based on their cron schedule, and then running them.","ref":"Lightning.Jobs.Scheduler.html"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/0","doc":"Find and start any cronjobs that are scheduled to run for a given time (defaults to the current time).","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/0"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/1","doc":"","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/1"},{"type":"module","title":"Lightning.Jobs.Trigger","doc":"Ecto model for Triggers. Triggers represent the criteria in which a Job might be invoked. Types Webhook (default) A webhook trigger allows a Job to invoked (via Lightning.Invocation ) when it's endpoint is called.","ref":"Lightning.Jobs.Trigger.html"},{"type":"function","title":"Lightning.Jobs.Trigger.new/1","doc":"","ref":"Lightning.Jobs.Trigger.html#new/1"},{"type":"function","title":"Lightning.Jobs.Trigger.validate/1","doc":"","ref":"Lightning.Jobs.Trigger.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Trigger.t/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:t/0"},{"type":"type","title":"Lightning.Jobs.Trigger.trigger_type/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:trigger_type/0"},{"type":"module","title":"Lightning.Projects","doc":"The Projects context.","ref":"Lightning.Projects.html"},{"type":"function","title":"Lightning.Projects.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Projects.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Projects.change_project/2","doc":"Returns an %Ecto.Changeset{} for tracking project changes. Examples iex> change_project ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#change_project/2"},{"type":"function","title":"Lightning.Projects.create_project/1","doc":"Creates a project. Examples iex> create_project ( %{ field : value } ) { :ok , % Project { } } iex> create_project ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#create_project/1"},{"type":"function","title":"Lightning.Projects.delete_project/1","doc":"Deletes a project and its related data, including workflows, work orders, runs, jobs, attempts, triggers, project users, project credentials, and dataclips Examples iex> delete_project ( project ) { :ok , % Project { } } iex> delete_project ( project ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#delete_project/1"},{"type":"function","title":"Lightning.Projects.export_project/2","doc":"Exports a project as yaml. Examples iex> export_project ( :yaml , project_id ) { :ok , string }","ref":"Lightning.Projects.html#export_project/2"},{"type":"function","title":"Lightning.Projects.get_project/1","doc":"","ref":"Lightning.Projects.html#get_project/1"},{"type":"function","title":"Lightning.Projects.get_project!/1","doc":"Gets a single project. Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project!/1"},{"type":"function","title":"Lightning.Projects.get_project_credential/2","doc":"","ref":"Lightning.Projects.html#get_project_credential/2"},{"type":"function","title":"Lightning.Projects.get_project_user/1","doc":"","ref":"Lightning.Projects.html#get_project_user/1"},{"type":"function","title":"Lightning.Projects.get_project_user/2","doc":"","ref":"Lightning.Projects.html#get_project_user/2"},{"type":"function","title":"Lightning.Projects.get_project_user!/1","doc":"Gets a single project_user. Raises Ecto.NoResultsError if the ProjectUser does not exist. Examples iex> get_project_user! ( 123 ) % ProjectUser { } iex> get_project_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_user!/1"},{"type":"function","title":"Lightning.Projects.get_project_user_role/2","doc":"Returns the role of a user in a project. Possible roles are :admin, :viewer, :editor, and :owner Examples iex> get_project_user_role ( user , project ) :admin iex> get_project_user_role ( user , project ) :viewer iex> get_project_user_role ( user , project ) :editor iex> get_project_user_role ( user , project ) :owner","ref":"Lightning.Projects.html#get_project_user_role/2"},{"type":"function","title":"Lightning.Projects.get_project_with_users!/1","doc":"Gets a single project with it's members via project_users . Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_with_users!/1"},{"type":"function","title":"Lightning.Projects.get_projects_for_user/1","doc":"","ref":"Lightning.Projects.html#get_projects_for_user/1"},{"type":"function","title":"Lightning.Projects.is_member_of?/2","doc":"","ref":"Lightning.Projects.html#is_member_of?/2"},{"type":"function","title":"Lightning.Projects.list_project_credentials/1","doc":"","ref":"Lightning.Projects.html#list_project_credentials/1"},{"type":"function","title":"Lightning.Projects.list_projects/0","doc":"Returns the list of projects. Examples iex> list_projects ( ) [ % Project { } , ... ]","ref":"Lightning.Projects.html#list_projects/0"},{"type":"function","title":"Lightning.Projects.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find projects that are ready for permanent deletion and purge them.","ref":"Lightning.Projects.html#perform/1"},{"type":"function","title":"Lightning.Projects.project_attempt_run_query/1","doc":"","ref":"Lightning.Projects.html#project_attempt_run_query/1"},{"type":"function","title":"Lightning.Projects.project_attempts_query/1","doc":"","ref":"Lightning.Projects.html#project_attempts_query/1"},{"type":"function","title":"Lightning.Projects.project_credentials_query/1","doc":"","ref":"Lightning.Projects.html#project_credentials_query/1"},{"type":"function","title":"Lightning.Projects.project_dataclip_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_dataclip_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_dataclips_query/1","doc":"","ref":"Lightning.Projects.html#project_dataclips_query/1"},{"type":"function","title":"Lightning.Projects.project_jobs_query/1","doc":"","ref":"Lightning.Projects.html#project_jobs_query/1"},{"type":"function","title":"Lightning.Projects.project_run_invocation_reasons/1","doc":"","ref":"Lightning.Projects.html#project_run_invocation_reasons/1"},{"type":"function","title":"Lightning.Projects.project_runs_query/1","doc":"","ref":"Lightning.Projects.html#project_runs_query/1"},{"type":"function","title":"Lightning.Projects.project_trigger_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_trigger_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_triggers_query/1","doc":"","ref":"Lightning.Projects.html#project_triggers_query/1"},{"type":"function","title":"Lightning.Projects.project_user_role_query/2","doc":"","ref":"Lightning.Projects.html#project_user_role_query/2"},{"type":"function","title":"Lightning.Projects.project_users_query/1","doc":"","ref":"Lightning.Projects.html#project_users_query/1"},{"type":"function","title":"Lightning.Projects.project_workflows_query/1","doc":"","ref":"Lightning.Projects.html#project_workflows_query/1"},{"type":"function","title":"Lightning.Projects.project_workorders_query/1","doc":"","ref":"Lightning.Projects.html#project_workorders_query/1"},{"type":"function","title":"Lightning.Projects.projects_for_user_query/1","doc":"","ref":"Lightning.Projects.html#projects_for_user_query/1"},{"type":"function","title":"Lightning.Projects.schedule_project_deletion/1","doc":"Given a project, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic project purge cronjob will never run. (Note that subsequent logins will be blocked for projects pending deletion.)","ref":"Lightning.Projects.html#schedule_project_deletion/1"},{"type":"function","title":"Lightning.Projects.select_first_project_for_user/1","doc":"","ref":"Lightning.Projects.html#select_first_project_for_user/1"},{"type":"function","title":"Lightning.Projects.update_project/2","doc":"Updates a project. Examples iex> update_project ( project , %{ field : new_value } ) { :ok , % Project { } } iex> update_project ( project , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project/2"},{"type":"function","title":"Lightning.Projects.update_project_user/2","doc":"Updates a project user. Examples iex> update_project_user ( project_user , %{ field : new_value } ) { :ok , % ProjectUser { } } iex> update_project_user ( projectUser , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project_user/2"},{"type":"function","title":"Lightning.Projects.url_safe_project_name/1","doc":"","ref":"Lightning.Projects.html#url_safe_project_name/1"},{"type":"function","title":"Lightning.Projects.validate_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the project scheduled_deletion. Examples iex> validate_for_deletion ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#validate_for_deletion/2"},{"type":"module","title":"Lightning.Projects.Project","doc":"Project model","ref":"Lightning.Projects.Project.html"},{"type":"function","title":"Lightning.Projects.Project.deletion_changeset/2","doc":"Changeset to validate a project deletion request, the user must enter the projects name to confirm.","ref":"Lightning.Projects.Project.html#deletion_changeset/2"},{"type":"function","title":"Lightning.Projects.Project.validate/1","doc":"","ref":"Lightning.Projects.Project.html#validate/1"},{"type":"type","title":"Lightning.Projects.Project.t/0","doc":"","ref":"Lightning.Projects.Project.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectCredential","doc":"Join table to assign credentials to a project","ref":"Lightning.Projects.ProjectCredential.html"},{"type":"type","title":"Lightning.Projects.ProjectCredential.t/0","doc":"","ref":"Lightning.Projects.ProjectCredential.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser","doc":"Join table to assign users to a project","ref":"Lightning.Projects.ProjectUser.html"},{"type":"type","title":"Lightning.Projects.ProjectUser.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.DigestEnum","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.DigestEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.RolesEnum","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.RolesEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.Provisioner","doc":"Provides functions for importing projects. This module is used by the provisioning HTTP API. When providing a project to import, all records must have an id field. It's up to the caller to ensure that the id is unique and generated ahead of time in the case of new records.","ref":"Lightning.Projects.Provisioner.html"},{"type":"function","title":"Lightning.Projects.Provisioner.import_document/3","doc":"Import a project.","ref":"Lightning.Projects.Provisioner.html#import_document/3"},{"type":"function","title":"Lightning.Projects.Provisioner.load_project/1","doc":"Load a project by ID, including all workflows and their associated jobs, triggers and edges. Returns nil if the project does not exist.","ref":"Lightning.Projects.Provisioner.html#load_project/1"},{"type":"function","title":"Lightning.Projects.Provisioner.parse_document/2","doc":"","ref":"Lightning.Projects.Provisioner.html#parse_document/2"},{"type":"function","title":"Lightning.Projects.Provisioner.validate_extraneous_params/1","doc":"Validate that there are no extraneous parameters in the changeset. For all params in the changeset, ensure that the param is in the list of known fields in the schema.","ref":"Lightning.Projects.Provisioner.html#validate_extraneous_params/1"},{"type":"module","title":"Lightning.Runtime.ChildProcess","doc":"Provides an interface between a RunSpec and the shell. Internally it calls node , and more specifically the OpenFn core CLI.","ref":"Lightning.Runtime.ChildProcess.html"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_command/1","doc":"Builds up a string for shell execution based on the RunSpec","ref":"Lightning.Runtime.ChildProcess.html#build_command/1"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_env/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#build_env/2"},{"type":"function","title":"Lightning.Runtime.ChildProcess.run/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#run/2"},{"type":"behaviour","title":"Lightning.Runtime.Handler","doc":"A strategy for executing things via ChildProcess. This module handles the dirty bits, setting up processes and coordinating results (and logs) as they arrive. Since it is a macro, see Lightning.Pipeline.Runner.Handler for a usage example.","ref":"Lightning.Runtime.Handler.html"},{"type":"function","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.log_callback/3","doc":"","ref":"Lightning.Runtime.Handler.html#c:log_callback/3"},{"type":"function","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_finish/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_finish/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_log_emit/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_log_emit/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_start/1","doc":"","ref":"Lightning.Runtime.Handler.html#on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.on_start/1","doc":"Called with context, if any - when the Run has been started.","ref":"Lightning.Runtime.Handler.html#c:on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.start/2","doc":"The entrypoint for executing a run.","ref":"Lightning.Runtime.Handler.html#c:start/2"},{"type":"type","title":"Lightning.Runtime.Handler.t/0","doc":"","ref":"Lightning.Runtime.Handler.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.LogAgent","doc":"Agent facility to consume STDOUT/STDERR byte by byte. Since it works on a byte by byte basis, you will need to perform line-splitting yourself. Usage: { :ok , log } = LogAgent . start_link ( ) "foo" = LogAgent . process_chunk ( log , { :stdout , "foo" } ) "foobar" = LogAgent . process_chunk ( log , { :stdout , "bar" } )","ref":"Lightning.Runtime.LogAgent.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Runtime.LogAgent.html#child_spec/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.start_link/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#start_link/1"},{"type":"type","title":"Lightning.Runtime.LogAgent.logline/0","doc":"","ref":"Lightning.Runtime.LogAgent.html#t:logline/0"},{"type":"module","title":"Lightning.Runtime.LogAgent.StringBuffer","doc":"Internal datastructure to hold and process new bytes for a list of characters. By checking the if the buffer is a complete grapheme, emitting the buffer once valid and returning nil otherwise. In the case of emojis and other language character sets, a character (in UTF-8) can be between 1-4 bytes - when streaming logs for example it's quite easy to receive less than the whole character which can result in crashes or corrupt text.","ref":"Lightning.Runtime.LogAgent.StringBuffer.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.new/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#new/0"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.reduce_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#reduce_chunk/2"},{"type":"type","title":"Lightning.Runtime.LogAgent.StringBuffer.t/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.Result","doc":"Data structure used to represent the result of a Run executed by Lightning.Runtime.ChildProcess .","ref":"Lightning.Runtime.Result.html"},{"type":"function","title":"Lightning.Runtime.Result.new/1","doc":"","ref":"Lightning.Runtime.Result.html#new/1"},{"type":"type","title":"Lightning.Runtime.Result.t/0","doc":"","ref":"Lightning.Runtime.Result.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.RunSpec","doc":"A struct containing all the parameters required to execute a Job.","ref":"Lightning.Runtime.RunSpec.html"},{"type":"function","title":"Lightning.Runtime.RunSpec.new/1","doc":"","ref":"Lightning.Runtime.RunSpec.html#new/1"},{"type":"type","title":"Lightning.Runtime.RunSpec.t/0","doc":"","ref":"Lightning.Runtime.RunSpec.html#t:t/0"},{"type":"exception","title":"Lightning.MetadataService.Error","doc":"","ref":"Lightning.MetadataService.Error.html"},{"type":"function","title":"Lightning.MetadataService.Error.message/1","doc":"Callback implementation for Exception.message/1 .","ref":"Lightning.MetadataService.Error.html#message/1"},{"type":"function","title":"Lightning.MetadataService.Error.new/1","doc":"","ref":"Lightning.MetadataService.Error.html#new/1"},{"type":"type","title":"Lightning.MetadataService.Error.t/0","doc":"","ref":"Lightning.MetadataService.Error.html#t:t/0"},{"type":"extras","title":"Lightning","doc":"OpenFn/Lightning is a fully open source, workflow automation platform that's used to automate critical business processes and integrate information systems. From last-mile services to national-level reporting, it boosts efficiency & effectiveness while enabling secure, stable, scalable interoperability at all levels. Read more about Lightning on OpenFn/Docs . Lightning is the latest "v2" of OpenFn, the Digital Public Good for workflow automation . Use Lightning to visually build, execute and manage workflows. The latest version of the OpenFn technology - first launched in 2014, now tried and tested by NGOs in over 40 countries Fully open source (no premium features or community edition, you get the same product whether using SaaS or self-hosted) Recognised as a Digital Public Good by the DPGA and a Global Good for Health by Digital Square","ref":"readme.html"},{"type":"extras","title":"Lightning - Contents","doc":"Demo Sign up for the BETA Features Getting started Run Lightning via Docker Deploy Lightning on Docker or Kubernetes Run Lightning on your local machine (contributors) Troubleshooting Contribute Project Configuration Quickstart Generate the documentation Security and standards Server specs for self-hosting Questions or feedback?","ref":"readme.html#contents"},{"type":"extras","title":"Lightning - Demo","doc":"Watch a quick demo here: https://www.youtube.com/watch?v=BNaxlHAWb5I Explore our demo app* with username: demo@openfn.org , password: welcome123 , or read through the features section to view screenshots of the app. *Note that the demo app refreshes daily, so do not configure workflows you want to save.","ref":"readme.html#demo"},{"type":"extras","title":"Lightning - Register for a Beta account","doc":"Register for a Beta account at app.openfn.org and go through the quick-start guide to get familiar with the app.","ref":"readme.html#register-for-a-beta-account"},{"type":"extras","title":"Lightning - Features","doc":"Build Plan and build workflows using Lightning's visual interface to quickly define when, where and what you want your automation to do. Use our CLI to quickly build, edit and deploy projects from the comfort of your own code editor. Monitor Monitor all workflow activity in one place. Filter and search runs to identify issues that need addressing and follow how a specific request has been processed Configure alerts to be notified on run failures Receive a project digest for a daily/weekly/monthly summary of your project activity Manage Manage users and access by project Roles and permissions Authorization is a central part of Lightning. As such, users are given different roles which determine what level of access they have for resources in the application. For more details about roles and permissions in Lightning, please refer to our documentation . Roadmap View our public roadmap here .","ref":"readme.html#features"},{"type":"extras","title":"Lightning - Getting Started","doc":"If you only want to RUN Lightning on your own server, we recommend using Docker. If you want to DEPLOY Lightning, we recommend Docker builds and Kubernetes. If you want to CONTRIBUTE to the project, we recommend running Lightning on your local machine .","ref":"readme.html#getting-started"},{"type":"extras","title":"Lightning - Run via Docker","doc":"Install the latest version of Docker Clone this repo using git Copy the .env.example file to .env Run docker compose run --rm web mix ecto.migrate By default the application will be running at localhost:4000 . You can then rebuild and run with docker compose build and docker compose up . See "Problems with Docker" for additional troubleshooting help. Note that you can also create your own docker-compose.yml file, configuring a postgres database and using a pre-built image from Dockerhub.","ref":"readme.html#run-via-docker"},{"type":"extras","title":"Lightning - Deploy on external infrastructure","doc":"See Deployment for more detailed information.","ref":"readme.html#deploy-on-external-infrastructure"},{"type":"extras","title":"Lightning - Run Lightning locally","doc":"Clone the repo and optionally set ENVs git clone git@github.com:OpenFn/Lightning.git # or from YOUR fork! cd Lightning cp .env.example .env # and adjust as necessary! Take note of database names and ports in particular—they've got to match across your Postgres setup and your ENVs. You can run lightning without any ENVs assuming a vanilla postgres setup (see below), but you may want to make adjustments. Database Setup If you're already using Postgres locally, create a new database called lightning_dev , for example. If you'd rather use Docker to set up a Postgres DB, create a new volume and image: docker volume create lightning-postgres-data docker create \\ --name lightning-postgres \\ --mount source=lightning-postgres-data,target=/var/lib/postgresql/data \\ --publish 5432:5432 \\ -e POSTGRES_PASSWORD=postgres \\ postgres:14.1-alpine docker start lightning-postgres Elixir & Ecto Setup We use asdf to configure our local environments. Included in the repo is a .tool-versions file that is read by asdf in order to dynamically make the specified versions of Elixir and Erlang available. You'll need asdf plugins for Erlang , NodeJs Elixir and k6 . asdf install # Install language versions mix local.hex mix deps.get mix local.rebar --force mix ecto.create # Create a development database in Postgres mix ecto.migrate [[ $(uname -m) == 'arm64' ]] && mix compile.rambo # Force compile rambo if on M1 mix lightning.install_runtime mix lightning.install_schemas npm install --prefix assets Run the app Lightning is a web app. To run it in interactive Elixir mode, start the development server by running with your environment variables by running: iex -S mix phx.server or if you have set up custom environment variables, run: env $(cat .env | grep -v "#" | xargs ) iex -S mix phx.server Once the server has started, head to localhost:4000 in your browser. Run the tests Before the first time running the tests, you need a test database setup. MIX_ENV=test mix ecto.create And then after that run the tests using: MIX_ENV=test mix test We also have test.watch installed which can be used to rerun the tests on file changes.","ref":"readme.html#run-lightning-locally"},{"type":"extras","title":"Lightning - Benchmarking","doc":"We are using k6 to benchmark Lightning. Under benchmarking folder you can find a script for benchmarking Webhook Workflows. See Benchmarking for more detailed information.","ref":"readme.html#benchmarking"},{"type":"extras","title":"Lightning - Troubleshooting","doc":"Trouble with environment variables For troubleshooting custom environment variable configuration it's important to know how an Elixir app loads and modifies configuration. The order is as follows: Stuff in config.exs is loaded. That is then modified (think: overwritten ) by stuff your ENV-specific config: dev.exs , prod.exs or test.exs . That is then modified by runtime.exs which is where you are allowed to use System.env() Finally init/2 (if present in a child application) gets called (which takes the config which has been set in steps 1-3) when that child application is started during the parent app startup defined in application.ex . Problems with Postgres If you're having connecting issues with Postgres, check the database section of your .env to ensure the DB url is correctly set for your environment — note that composing a DB url out of other, earlier declared variables, does not work while using xargs . Problems with Debian If you're getting this error on debian == > earmark_parser Compiling 1 file ( . yrl ) / usr / lib / erlang / lib / parsetools - 2.3 . 1 / include / yeccpre . hrl : no such file or directory could not compile dependency :earmark_parser , "mix compile" failed . You can recompile this dependency with "mix deps.compile earmark_parser" , update it with "mix deps.update earmark_parser" or clean it with "mix deps.clean earmark_parser" You need to install erlang development environment sudo apt install erlang-dev refer to this issue Problems with Docker Versions The build may not work on old versions of Docker and Docker compose. It has been tested against: Docker version 20.10 . 17 , build 100 c701 Docker Compose version v2 . 6.0 Problems with rambo When running mix compile.rambo on Apple Silicon (an Apple M1/M2, macarm , aarch64-apple-darwin ) and encountering the following error: ** (RuntimeError) Rambo does not ship with binaries for your environment. aarch64 - apple - darwin22 . 3.0 detected Install the Rust compiler so a binary can be prepared for you . lib / mix / tasks / compile . rambo . ex : 89 : Mix.Tasks.Compile.Rambo . compile! / 0 lib / mix / tasks / compile . rambo . ex : 51 : Mix.Tasks.Compile.Rambo . run / 1 ( mix 1.14 . 2 ) lib / mix / task . ex : 421 : anonymous fn / 3 in Mix.Task . run_task / 4 ( mix 1.14 . 2 ) lib / mix / cli . ex : 84 : Mix.CLI . run_task / 2 You can resolve this error by installing the Rust compiler using Homebrew. Run the following command in your terminal: brew install rust If you have already compiled Rambo explicitly via mix compile.rambo , and you are still seeing the following error: sh : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : No such file or directory sh : line 0 : exec : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : cannot execute : No such file or directory You can try renaming deps/rambo/priv/rambo-mac to deps/rambo/priv/rambo . If neither of the approaches above work, please raise an issue. Starting from scratch If you're actively working with docker, you start experiencing issues, and you would like to start from scratch you can clean up everything and start over like this: # To remove any ignored files and reset your .env to it's example git clean -fdx && cp .env.example .env # You can skip the line below if you want to keep your database docker compose down --rmi all --volumes docker compose build --no-cache web && \\ docker compose create --force-recreate docker compose run --rm web mix ecto.migrate docker compose up","ref":"readme.html#troubleshooting"},{"type":"extras","title":"Lightning - Project Configuration Quickstart","doc":"For help getting started with your OpenFn/Lightning Project and Workflows configuration, check out the Lightning Quickstart Guidance on OpenFn/Docs.","ref":"readme.html#project-configuration-quickstart"},{"type":"extras","title":"Lightning - Contribute to this project","doc":"First, thanks for being here! You're contributing to a digital public good that will always be free and open source and aimed at serving innovative NGOs, governments, and social impact organizations the world over! You rock. ❤️ FYI, Lightning is built in Elixir , harnessing the Phoenix Framework . Currently, the only unbundled dependency is a PostgreSQL database. If you'd like to contribute to this projects, follow the steps below: Assign yourself to an issue Read through the existing issues , assign yourself to the issue you have chosen. Leave a comment on the issue to let us know you'll be working on it, and if you have any questions of clarifications that would help you get started ask them there - we will get back to you as soon as possible. If there isn't already an issue for the feature you would like to contribute, please start a discussion in our community forum . Open a pull request Clone the Lightning repository, then fork it . Run through setting up your environment and make your changes. Make sure you have written your tests and updated /CHANGELOG.md (in the 'Unreleased' section, add a short description of the changes you are making, along with a link to your issue). Open a draft pull request by clicking "Contribute > Open Pull Request" from your forked repository. Fill out the pull request template (this will be added automatically for you), then make sure to self-review your code and go through the 'Review checklist'. Don't worry about the QA checkbox, our product manager Amber will tick that once she has reviewed your PR. You can leave any notes for the reviewer in a comment. Once you're ready to submit a pull request, you can mark your draft PR as 'Ready for review' and assign @stuartc or @taylordowns2000.","ref":"readme.html#contribute-to-this-project"},{"type":"extras","title":"Lightning - Generating Documentation","doc":"You can generate the HTML and EPUB documentation locally using: mix docs and opening doc/index.html in your browser.","ref":"readme.html#generating-documentation"},{"type":"extras","title":"Lightning - Security and Standards","doc":"We use a host of common Elixir static analysis tools to help us avoid common pitfalls and make sure we keep everything clean and consistent. In addition to our test suite, you can run the following commands: mix format --check-formatted Code formatting checker, run again without the --check-formatted flag to have your code automatically changed. mix dialyzer Static analysis for type mismatches and other common warnings. See dialyxir . mix credo Static analysis for consistency, and coding standards. See Credo . mix sobelow Check for commonly known security exploits. See Sobelow . MIX_ENV=test mix coveralls Test coverage reporter. This command also runs the test suite, and can be used in place of mix test when checking everything before pushing your code. See excoveralls . For convenience there is a verify mix task that runs all of the above and defaults the MIX_ENV to test . For more guidance on security best practices for workflow automation implementations, check out OpenFn Docs: docs.openfn.org/documentation/getting-started/security","ref":"readme.html#security-and-standards"},{"type":"extras","title":"Lightning - Server Specs for Self-Hosting","doc":"For recommend server specifications for self-hosting of Lightning, check out this Community topic: community.openfn.org/t/specs-for-self-hosting-lightning/292","ref":"readme.html#server-specs-for-self-hosting"},{"type":"extras","title":"Lightning - Support","doc":"If you have any questions, feedback, or issues, please: Post on the OpenFn Community at community.openfn.org Open an issue directly on this Github Repo: github.com/OpenFn/Lightning/issues","ref":"readme.html#support"},{"type":"extras","title":"Deployment","doc":"","ref":"deployment.html"},{"type":"extras","title":"Deployment - Encryption","doc":"Lightning enforces encryption at rest for Credentials, for which an encryption key must be provided when running in production. The key is expected to be a randomized set of bytes, 32 long; and Base64 encoded when setting the environment variable. There is a mix task that can generate keys in the correct shape for use as an environment variable: mix lightning.gen_encryption_key 0bJ9w+hn4ebQrsCaWXuA9JY49fP9kbHmywGd5K7k+/s= Copy your key (NOT THIS ONE) and set it as PRIMARY_ENCRYPTION_KEY in your environment.","ref":"deployment.html#encryption"},{"type":"extras","title":"Deployment - Environment Variables","doc":"Note that for secure deployments, it's recommended to use a combination of secrets and configMaps to generate secure environment variables. ADAPTORS_PATH - where you store your locally installed adaptors DISABLE_DB_SSL - in production the use of an SSL conntection to Postgres is required by default, setting this to "true" allows unencrypted connections to the database. This is strongly discouraged in real production environment. K8S_HEADLESS_SERVICE - this environment variable is automatically set if you're running on GKE and it is used to establish an Erlang node cluster. Note that if you're not using Kubernetes, the "gossip" strategy is used for establish clusters. LISTEN_ADDRESS " - the address the web server should bind to, defaults to 127.0.0.1 to block access from other machines. LOG_LEVEL - how noisy you want the logs to be (e.g. debug , info ) MAX_RUN_DURATION - the maximum time (in milliseconds) that jobs are allowed to run (keep this below your termination_grace_period if using kubernetes) MIX_ENV - your mix env, likely prod for deployment NODE_ENV - node env, likely production for deployment ORIGINS - the allowed origins for web traffic to the backend PORT - the port your Phoenix app runs on PRIMARY_ENCRYPTION_KEY - a base64 encoded 32 character long string. See Encryption . SCHEMAS_PATH - path to the credential schemas that provide forms for different adaptors SECRET_KEY_BASE - a secret key used as a base to generate secrets for encrypting and signing data. SENTRY_DSN - if using Sentry for error monitoring, your DSN URL_HOST - the host, used for writing urls (e.g., demo.openfn.org ) URL_PORT - the port, usually 443 for production URL_SCHEME - the scheme for writing urls, (e.g., https ) Google Using your Google Cloud account, provision a new OAuth 2.0 Client with the 'Web application' type. Set the callback url to: https://<ENDPOINT DOMAIN>/authenticate/callback . Replacing ENDPOINT DOMAIN with the host name of your instance. Once the client has been created, get/download the OAuth client JSON and set the following environment variables: GOOGLE_CLIENT_ID - Which is client_id from the client details. GOOGLE_CLIENT_SECRET - client_secret from the client details.","ref":"deployment.html#environment-variables"},{"type":"extras","title":"Benchmarking","doc":"Execute the following steps to run a benchmark on Lightning: Make sure you have k6 installed locally. If you're using asdf you can run asdf install in the project root. Spin up your Lightning local instance Run the demo setup script: mix run --no-start priv/repo/demo.exs The webhookURL is already set to default to the webhook created in the demo data In another terminal (do not stop the Lightning server) run the benchmarking/script.js file using the following command k6 run benchmarking/script.js If the script exits succesfully, this means the app met the defined performance thresholds. To collect the benchmarking data in a CSV file, run the previous command with the --out filename option. k6 run --out csv=test_results.csv benchmarking/script.js See results output for other available output formats.","ref":"benchmarking.html"},{"type":"extras","title":"Provisioning","doc":"Lightning offers the ability to configure projects via the HTTP API. By providing a JSON document with the desired configuration, the project can be configured to your liking.","ref":"provisioning.html"},{"type":"extras","title":"Provisioning - Using the API","doc":"The API is available at /api/provision , and expects an application/json Content-Type. Authentication The API requires a valid auth token to be provided in the Authorization header. Example Request curl -X POST \\ -d @project.json \\ -H "Authorization: Bearer $TOKEN" \\ -H "Content-Type: application/json" \\ $ENDPOINT/api/provision","ref":"provisioning.html#using-the-api"},{"type":"extras","title":"Provisioning - Document Structure","doc":"The provisioning document is a JSON document with the project at the root. All entities must have an id field, which is a UUIDv4 string. In the case of new entities, this must be generated by the client. The API is idempotent, and the distinction between creating and updating is determined by the presence of the id field. { "id": "<<project-id>>", "name": "<<project-name>>", "workflows": [ { "id": "<<workflow-id>>", "name": "<<workflow-name>>", "jobs": [ { "id": "<<job-id>>", "name": "<<job-name>>", "body": "<<job-body>>", "adaptor": "<<adaptor-name>>", "enabled": true } // ... more jobs ], "triggers": [ { "id": "<<trigger-id>>", "name": "<<trigger-name>>", "type": "webhook" } // ... more triggers ], "edges": [ { "id": "<<edge-id>>", "source_trigger_id": "<<trigger-id>>", "target_job_id": "<<job-id>>" } // ... more edges ] } // ... more workflows ] }","ref":"provisioning.html#document-structure"},{"type":"extras","title":"Provisioning - API Behaviour","doc":"The API expects all existing entities to be provided in the provisioning document. If the document provided is out of date (e.g. a new job was added on the server), a new reference document should be fetched and the changes applied to it. Deleting Entities Entities can be deleted by setting the disabled key to true . Example: { "id": "<<project-id>>", "workflows": [ { "id": "<<workflow-id>>", "jobs": [ { "id": "<<job-id>>", "delete": true // <== delete this job } ] } ] }","ref":"provisioning.html#api-behaviour"},{"type":"extras","title":"Provisioning - Relationship with Projects as Code","doc":"The Projects as Code spec is a superset of the provisioning API. Projects as Code allows for the user to specify a key for each entity, which makes it easier to manage the project in the future. For example: name: my-project workflows: workflow-one: jobs: job-one: body: | console.log("Hello World"); adaptor: '@openfn/language-common' enabled: true triggers: trigger-one: type: webhook edges: - source_trigger: trigger-one target_job: job-one The above YAML document illustrates the use of keys being used to identify entities. Allowing the user to provision the same project to multiple environments. The API is unaware of 'keys', and expects IDs to be provided by the client. In order to convert the above YAML document to a provisioning document, the CLI uses a local state file (if available) to map the keys to UUIDs. Using the example above a state file might look like this: { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": {}, "triggers": {} } } } The state file shows that the project and workflow already exist, but the job, trigger and edge do not. In order to create these new entities, IDs will be applied them. On a successful application of the provisioning document, the state file will be updated to reflect the new IDs and entities. { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": { "job-one": { "id": "18ed71de-caf8-4822-aefc-5b19351f4016" } }, "triggers": { "trigger-one": { "id": "e0b9f357-9cf9-4206-9924-4d5674aad830" } }, "edges": [ { "id": "c239d994-6662-4637-90f8-0293c924b461", "source_trigger_id": "e0b9f357-9cf9-4206-9924-4d5674aad830", "target_job_id": "18ed71de-caf8-4822-aefc-5b19351f4016" } ] } } }","ref":"provisioning.html#relationship-with-projects-as-code"},{"type":"extras","title":"Changelog","doc":"All notable changes to this project will be documented in this file. The format is based on Keep a Changelog , and this project adheres to Semantic Versioning .","ref":"changelog.html"},{"type":"extras","title":"Changelog - [Unreleased]","doc":"Added Changed Moved to Elixir 1.15 and Erlang 26.0.2 to sort our an annoying ElixirLS issue that was slowing down our engineers. Fixed Fix issue creating a new credential from the Job editor where the new credential was not being set on the job. #951","ref":"changelog.html#unreleased"},{"type":"extras","title":"Changelog - [v0.7.0-pre5] - 2023-07-28","doc":"Added Changed Unless otherwise specified, only show workorders with activity in last 14 days #968 Fixed","ref":"changelog.html#v0-7-0-pre5-2023-07-28"},{"type":"extras","title":"Changelog - [v0.7.0-pre4] - 2023-07-27","doc":"Added Changed Don't add cast fragments if the search_term is nil #968 Fixed","ref":"changelog.html#v0-7-0-pre4-2023-07-27"},{"type":"extras","title":"Changelog - [v0.7.0-pre3] - 2023-07-26","doc":"Added Changed Fixed Fixed an issue with newly created edges that prevented downstream jobs 977","ref":"changelog.html#v0-7-0-pre3-2023-07-26"},{"type":"extras","title":"Changelog - [v0.7.0-pre2] - 2023-07-26","doc":"Note that this is a pre-release with a couple of known bugs that are tracked in the Nodes and Edges epic . Added Added ability for a user to enable MFA on their account; using 2FA apps like Authy, Google Authenticator etc #890 Write/run sql script to convert triggers #875 Export projects as .yaml via UI #249 Changed In v0.7.0 we change the underlying workflow building and execution infrastructure to align with a standard "nodes and edges" design for directed acyclic graphs (DAGs). Make sure to run the migrations! 793 Fixed Propagate url pushState/changes to Workflow Diagram selection #944 Fix issue when deleting nodes from the workflow editor #830 Fix issue when clicking a trigger on a new/unsaved workflow #954","ref":"changelog.html#v0-7-0-pre2-2023-07-26"},{"type":"extras","title":"Changelog - [0.6.7] - 2023-07-13","doc":"Added Add feature to bulk rerun work orders from a specific step in their workflow; e.g., "rerun these 50 work orders, starting each at step 4." #906 Changed Fixed Oban exception: "value too long" when log lines are longer than 255 chars #929","ref":"changelog.html#0-6-7-2023-07-13"},{"type":"extras","title":"Changelog - [0.6.6] - 2023-06-30","doc":"Added Add public API token to the demo site setup script Check and renew OAuth credentials when running a job #646 Fixed Remove google sheets from adaptors list until supporting oauth flow #792 Remove duplicate google sheets adaptor display on credential type picklist #663 Fix demo setup script for calling from outside the app on Kubernetes deployments #917","ref":"changelog.html#0-6-6-2023-06-30"},{"type":"extras","title":"Changelog - [0.6.5] - 2023-06-22","doc":"Added Ability to rerun workorders from start by selecting one of more of them from the History page and clicking the "Rerun" button. #659 Fixed Example runs for demo incorrect #856","ref":"changelog.html#0-6-5-2023-06-22"},{"type":"extras","title":"Changelog - [0.6.3] - 2023-06-15","doc":"Fixed Prevent saving null log lines to the database, fix issue with run display #866","ref":"changelog.html#0-6-3-2023-06-15"},{"type":"extras","title":"Changelog - [0.6.2] - 2023-06-09","doc":"Fixed Fixed viewer permissions for delete workflow Fixed bug with workflow cards #859","ref":"changelog.html#0-6-2-2023-06-09"},{"type":"extras","title":"Changelog - [0.6.1] - 2023-06-08","doc":"Fixed Fixed bug with run logs #864 Correctly stagger demo runs to maintain order #856 Remove Timex use from SetupUtils in favor of DateTime to fix issue when calling it in escript.","ref":"changelog.html#0-6-1-2023-06-08"},{"type":"extras","title":"Changelog - [0.6.0]- 2023-04-12","doc":"Added Create sample runs when generating sample workflow #821 Added a provisioning api for creating and updating projects and their workflows See: PROVISIONING.md #641 Add ability for a superuser to schedule deletion, cancel deletion, and delete projects #757 Add ability for a project owner to schedule deletion, cancel deletion, and delete projects #746 Changed Ability to store run log lines as rows in a separate table #514 Fixed Incorrect project digest queries #768 ] Fix issue when purging deleted users #747 Generate a random name for Workflows when creating one via the UI. #828 Handle error when deleting a job with runs. #814","ref":"changelog.html#0-6-0-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.2]","doc":"Added Add workflow_edges table in preparation for new workflow editor implementation #794 Stamped credential_id on run directly for easier auditing of the history interface. Admins can now see which credential was used to run a run. #800 Better errors when using magic functions: "no magic yet" and "check credential" #812 Changed The delete-project function now delete all associated activities #759 Fixed","ref":"changelog.html#0-5-2"},{"type":"extras","title":"Changelog - [0.5.1] - 2023-04-12","doc":"Added Added ability to create and revoke personal API tokens #147 Add last-used at to API tokens #722 Improved "save" for job builder; users can now press Ctrl + S or ⌘ + S to save new or updated jobs job panel will not close. (Click elsewhere in the canvas or click the "Close" button to close.) #568 Add filtered search params to the history page URL #660 Changed The secret scrubber now ignores booleans 690 Fixed The secret scrubber now properly handles integer secrets from credentials 690 Updated describe-package dependency, fixing sparkles in adaptor-docs 657 Clicks on the workflow canvas were not lining up with the nodes users clicked on; they are now 733 Job panel behaves better when collapsed 774","ref":"changelog.html#0-5-1-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.0] - 2023-04-03","doc":"Added Magic functions that fetch real metadata from connected systems via credentials and suggest completions in the job builder (e.g., pressing control-space when setting the orgUnit attribute for a DHIS2 create operation will pull the actual list of orgUnits with human readable labels and fill in their orgUnit codes upon enter.) 670 A "metadata explorer" to browse actual system metadata for connected instances. 658 Resizable job builder panel for the main canvas/workflow view. 681 Changed Display timezone for cron schedule—it is always UTC. #716 Instance administrators can now configure the interval between when a project owner or user requests deletion and when these records are purged from the database. It defaults to 7, but by providing a PURGE_DELETED_AFTER_DAYS environment variable the grace period can be altered. Note that setting this variable to 0 will make automatic purging never occur but will still make "deleted" projects and users unavailable. This has been requested by certain organizations that must retain audit logs in a Lightning instance. 758 Fixed Locked CLI version to @openfn/cli@0.0.35 . #761","ref":"changelog.html#0-5-0-2023-04-03"},{"type":"extras","title":"Changelog - [0.4.8] - 2023-03-29","doc":"Added Added a test harness for monitoring critical parts of the app using Telemetry #654 Changed Set log level to info for runs. Most of the debug logging is useful for the CLI, but not for Lightning. In the future the log level will be configurable at instance > project > job level by the superuser and any project admin . Renamed license file so that automagic github icon is less confusing Fixed Broken links in failure alert email #732 Registration Submission on app.openfn.org shows internal server error in browser #686 Run the correct runtime install mix task in Dockerfile-dev #541 Users not disabled when scheduled for deletion #719","ref":"changelog.html#0-4-8-2023-03-29"},{"type":"extras","title":"Changelog - [0.4.6] - 2023-03-23","doc":"Added Implement roles and permissions across entire app #645 Fix webhook URL ( https://<<HOST_URL>>/i/cae544ab-03dc-4ccc-a09c-fb4edb255d7a ) for the OpenHIE demo workflow 448 Phoenix Storybook for improved component development Load test for webhook endpoint performance #645 Notify user via email when they're added to a project #306 Added notify user via email when their account is created #307 Changed Improved errors when decoding encryption keys for use with Cloak. #684 Allow users to run ANY job with a custom input. #629 Fixed Ensure JSON schema form inputs are in the same order as they are written in the schema #685","ref":"changelog.html#0-4-6-2023-03-23"},{"type":"extras","title":"Changelog - [0.4.4] - 2023-03-10","doc":"Added Users can receive a digest email reporting on a specified project. #638 #585 Changed Fixed","ref":"changelog.html#0-4-4-2023-03-10"},{"type":"extras","title":"Changelog - [0.4.3] - 2023-03-06","doc":"Added Tooltips on Job Builder panel #650 Changed Upgraded to Phoenix 1.7 (3945856) Fixed Issue with FailureAlerter configuration missing in prod mode.","ref":"changelog.html#0-4-3-2023-03-06"},{"type":"extras","title":"Changelog - [0.4.2] - 2023-02-24","doc":"Added A user can change their own email #247 Added a SCHEMAS_PATH environment variable to override the default folder location for credential schemas #604 Added the ability to configure Google Sheets credentials #536 Function to import a project #574 Changed Users cannot register if they have not selected the terms and conditions #531 Fixed Jobs panel slow for first open after restart #567","ref":"changelog.html#0-4-2-2023-02-24"},{"type":"extras","title":"Changelog - [0.4.0] - 2023-02-08","doc":"Added Added a Delete job button in Inspector Filter workflow runs by text/value in run logs or input body Drop "configuration" key from Run output dataclips after completion Ability to 'rerun' a run from the Run list Attempts and Runs update themselves in the Runs list Configure a project and workflow for a new registering user Run a job with a custom input Added plausible analytics Allow user to click on Webhook Trigger Node to copy webhook URL on workflow diagram Allow any user to delete a credential that they own Create any credential through a form except for OAuth Refit all diagram nodes on browser and container resize Enable distributed Erlang, allowing any number of redundant Lightning nodes to communicate with each other. Users can set up realtime alerts for a project Changed Better code-assist and intelliense in the Job Editor Updated @openfn/workflow-diagram to 0.4.0 Make plus button part of job nodes in Workflow Diagram Updated @openfn/adaptor-docs to 0.0.5 Updated @openfn/describe-package to 0.0.10 Create an follow a manual Run from the Job Inspector View all workflows in a project on the workflows index page Move @openfn/workflow-diagram into the application, the NPM module is now deprecated. Remove workflow name from first node Move the used parts of @openfn/engine into the application. [BREAKING CHANGE] Ported mix openfn.install.runtime into application, use mix lightning.install_runtime . [BREAKING CHANGE] Introduced @openfn/cli as the new runtime for Jobs Rename a workflow through the page heading Hide the dataclips tab for beta Make adaptor default to common@latest Remove jobs list page Better error handling in the docs panel Disable credential ownership transfer in dev and prod environments Add project settings page Change Workorder filters to apply to the aggregate state of the workorder and not the run directly Enable jobs by default Set log level to info Add Beta checkbox to register page User roles and permissions Fixed Don't consider disabled jobs when calculating subsequent runs Fixed overflow on Job Editor Tooltips Fixed auto-scroll when adding a new snippet in the Job Editor Fixed common operation typings in Job Editor","ref":"changelog.html#0-4-0-2023-02-08"},{"type":"extras","title":"Changelog - [0.3.1] - 2022-11-22","doc":"Fixed Fixed bug that attempted to execute HTML scripts in dataclips Fixed bug that prevented workorders from displaying in the order of their last run, descending. Remove alerts after set timeout or close","ref":"changelog.html#0-3-1-2022-11-22"},{"type":"extras","title":"Changelog - [0.3.0] - 2022-11-21","doc":"Added Add seed data for demo site Create adaptor credentials through a form Configure cron expressions through a form View runs grouped by workorders and attempts Run an existing Job with any dataclip uuid from the Job form Changed Redirect users to projects list page when they click on Admin Settings menu Move job, project, input and output Dataclips to Run table Reverse the relationship between Jobs and Triggers. Triggers now can exist on their own; setting the stage for branching and merging workflows Updated Elixir and frontend dependencies [BREAKING CHANGE] Pipeline now uses WorkOrders, previous data is not compatible. Runs, Dataclips and Attempts now all correctly use usec resolution timestamps. Upgraded LiveView to 0.18.0 Upgraded Elixir to 1.14.1 and OTP 25 Workflow Job editor now behaves like a panel Split JobLive.InspectorFormComponent into different plug-able subcomponents Ensure new jobs with cron triggers receive a default frequency Webhooks are now referenced by the trigger id instead of job id. Filter runs by status Filter runs by workflow Filter runs by date View a job run from the runs history View latest matching inputs to run a job with","ref":"changelog.html#0-3-0-2022-11-21"},{"type":"extras","title":"Changelog - [0.2.0] - 2022-09-12","doc":"Changed [BREAKING CHANGE] Add Workflow model, Jobs now belong to a Workflow This is a breaking change to the schema. Use Node.js 18, soon to be in LTS. Visualize success/fail triggers in workflow diagram. Move WorkflowDiagram related actions from DashboardLive into WorkflowLive Move WorkflowDiagram component into liveview, so that we can subscribe to channels (i.e. updating of the diagram when someone changes something). Integrate @openfn/workflow-diagram@0.0.8 and use the new Store interface for updating it. Remove component_mounted event from WorkflowDiagram hook, using a MutationObserver and a Base64 encoded JSON payload. Fixed an issue where the compiler component would try and load a 'nothing adaptor', added a condition to check an adaptor is actually selected. Removed previous Workflow CTE queries, replaced by the introduction of the Workflow model, see ( https://github.com/OpenFn/Lightning/blob/53da6883483e7d8d078783f348da327d1dd72d20/lib/lightning/workflows.ex#L111-L119 ).","ref":"changelog.html#0-2-0-2022-09-12"},{"type":"extras","title":"Changelog - [0.1.13] - 2022-08-29","doc":"Added Allow administrators to configure OIDC providers for authentication (note that this is just for authenticating, not yet for creating new accounts via OIDC) Add Monaco editor to the step/job panel Allow users to delete their own accounts. Schedule their user and credentials data for deletion when they do. Allow superusers to delete a user account. Schedule the user's credentials and user data for deletion when they do. If a user is scheduled for deletion, disable their account and prevent them from logging in. The 'User profile' and 'Credentials' page now have a sidebar menu Changed Project users now have one of the following roles: viewer, editor, admin, owner Users only have the following roles: user, superuser","ref":"changelog.html#0-1-13-2022-08-29"},{"type":"extras","title":"Changelog - [0.1.12] - 2022-08-15","doc":"Added Transfer credential ownership to another user. Create credentials via a form interface* Show "projects with access" in credentials list view. Show job in runs list and run view. Added roles and permissions to workflows and history page #645 *The form is defined by a JSON schema provided by an adaptor, in most cases: e.g., language-dhis2 provides a single schema which defines the required attributes for state.configuration , while language-common provides multiple credential schemas like "oauth" or "basic auth" which define attributes for state.configuration and which might be used by lots of different jobs.) Fixed User menu (top right) appears on top of all other components. User profile screen integrated with the rest of the liveview app.","ref":"changelog.html#0-1-12-2022-08-15"},{"type":"extras","title":"Changelog - [0.1.11] - 2022-08-05","doc":"Fixed Fixed logging in Runner when :debug log level used; note that this caused crashes in Oban","ref":"changelog.html#0-1-11-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.10] - 2022-08-05","doc":"Added Credential auditing Build/version information display for easier debugging Fixed Fixed a bug that enqueued cron-triggered jobs even when they were disabled","ref":"changelog.html#0-1-10-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.9] - 2022-07-27","doc":"Added Navigate to user profile or credentials page and log out through the user icon dropdown Create and edit dataclips Add a production tag to credentials View a dropdown of operations and their description for the language-common v2.0.0-rc2 adaptor (this pattern to be rolled out across adaptors) Changed Navigate between projects through a project picker on the navbar Fixed Run Lightning with docker Security Sensitive credential values are scrubbed from run logs All credentials are encrypted at REST","ref":"changelog.html#0-1-9-2022-07-27"},{"type":"extras","title":"Changelog - [0.1.7] - 2022-06-24","doc":"Added Run a job with a cron trigger Queue jobs via Oban/Postgres Edit jobs via the workflow canvas","ref":"changelog.html#0-1-7-2022-06-24"},{"type":"extras","title":"Changelog - [0.1.6] - 2022-06-07","doc":"Added Register, log in and log out of an account Allow superusers and admin users to create projects Allow admin users to create or disable a user’s account Allow superusers for local deployments to create users and give them access to project spaces Create and edit a job with a webhook, flow/fail or cron trigger Create and edit credentials for a job Copy a job's webhook URL View all workflows in a project visually Deploy lightning locally with Docker Enable a job to automatically process incoming requests Run a job with a webhook or flow/fail trigger View job runs along with their logs, exit code, start and end time View data clips that have initiated job runs (http requests for webhooks, run results) Changed - Removed -","ref":"changelog.html#0-1-6-2022-06-07"}] \ No newline at end of file +searchNodes=[{"type":"module","title":"Lightning","doc":"Lightning keeps the contexts that define your domain and business logic. Contexts are also responsible for managing your data, regardless if it comes from the database, an external API or others.","ref":"Lightning.html"},{"type":"module","title":"Lightning.AdaptorRegistry","doc":"Registry process to query and maintain a list of adaptors available for writing jobs. Currently it queries NPM for all modules in the @openfn organization and filters out modules that are known not to be adaptors. Usage # Starting the process AdaptorRegistry . start_link ( ) # Getting a list of all adaptors Lightning.AdaptorRegistry.AdaptorRegistry . all ( ) Caching By default the results are cached to disk, and will be reused every start. In order to disable or configure caching pass see: start_link/1 . The process uses :continue to return before the adaptors have been queried. This does mean that the first call to the process will be delayed until the handle_continue/2 has finished. Timeouts There is a 'general' timeout of 30s, this is used for GenServer calls like all/1 and also internally when the modules are being queried. NPM can be extremely fast to respond if the package is cached on their side, but can take a couple of seconds if not cached.","ref":"Lightning.AdaptorRegistry.html"},{"type":"function","title":"Lightning.AdaptorRegistry.all/1","doc":"Get the current in-process list of adaptors. This call will wait behind the :continue message when the process starts up, so it may take a while the first time it is called (and the list hasn't been fetched yet).","ref":"Lightning.AdaptorRegistry.html#all/1"},{"type":"function","title":"Lightning.AdaptorRegistry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorRegistry.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorRegistry.fetch/0","doc":"Fetch a list of packages for the @openfn organisation","ref":"Lightning.AdaptorRegistry.html#fetch/0"},{"type":"function","title":"Lightning.AdaptorRegistry.latest_for/2","doc":"Get a latest version for a given module.","ref":"Lightning.AdaptorRegistry.html#latest_for/2"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_adaptor/1","doc":"","ref":"Lightning.AdaptorRegistry.html#resolve_adaptor/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name/1","doc":"Destructures an NPM style package name into module name and version. Example iex> resolve_package_name ( "@openfn/language-salesforce@1.2.3" ) { "@openfn/language-salesforce" , "1.2.3" } iex> resolve_package_name ( "@openfn/language-salesforce" ) { "@openfn/language-salesforce" , nil }","ref":"Lightning.AdaptorRegistry.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name!/1","doc":"Same as resolve_package_name/1 except will throw an exception if a package name cannot be matched.","ref":"Lightning.AdaptorRegistry.html#resolve_package_name!/1"},{"type":"function","title":"Lightning.AdaptorRegistry.start_link/1","doc":"Starts the AdaptorRegistry Options :use_cache (defaults to false) - stores the last set of results on disk and uses the cached file for every subsequent start. It can either be a boolean, or a string - the latter being a file path to set where the cache file is located. :name (defaults to AdaptorRegistry) - the name of the process, useful for testing and/or running multiple versions of the registry","ref":"Lightning.AdaptorRegistry.html#start_link/1"},{"type":"function","title":"Lightning.AdaptorRegistry.versions_for/2","doc":"Get a list of versions for a given module.","ref":"Lightning.AdaptorRegistry.html#versions_for/2"},{"type":"module","title":"Lightning.AdaptorRegistry.Npm","doc":"NPM API functions","ref":"Lightning.AdaptorRegistry.Npm.html"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.package_detail/1","doc":"Retrieve all details for an NPM package","ref":"Lightning.AdaptorRegistry.Npm.html#package_detail/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_body/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_options/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_params/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_url/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AdaptorRegistry.Npm.html#request/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AdaptorRegistry.Npm.html#request/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AdaptorRegistry.Npm.html#request!/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AdaptorRegistry.Npm.html#start/0"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#stream_next/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.user_packages/1","doc":"Retrieve all packages for a given user or organization. Return empty list if application cannot connect to NPM. (E.g., because it's started offline.)","ref":"Lightning.AdaptorRegistry.Npm.html#user_packages/1"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.body/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:body/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.headers/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:headers/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.method/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:method/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.options/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:options/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.params/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:params/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.request/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:request/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.url/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:url/0"},{"type":"module","title":"Lightning.AdaptorService","doc":"The Adaptor Service is use to query and install adaptors in order to run jobs. On startup, it queries the filesystem for package.json files and builds up a list of available adaptors. Configuration The service requires at least :adaptors_path , which is used to both query which adaptors are installed and when to install new adaptors. Another optional setting is: :repo , which must point at a module that will be used to do the querying and installing. Installing Adaptors Using the install/2 function an adaptor can be installed, which will also add it to the list of available adaptors. The adaptor is marked as :installing , to allow for conditional behaviour elsewhere such as delaying or rejecting processing until the adaptor becomes available. Looking up adaptors The module leans on Elixir's built-in Version module to provide version lookups. When looking up an adaptor, either a string or a tuple can be used. In the case of requesting the latest version, any one of these will return the latest version the service is aware of. @openfn/language-http @openfn/language-http@latest {"@openfn/language-http", nil} {"@openfn/language-http", "latest"} {~r/language-http/, "latest"} You can also request a specific version, or use a range specification: @openfn/language-http@1.2.3 {"@openfn/language-http", "~> 1.2.0"} {"@openfn/language-http", "< 2.0.0"} NOTE More complex npm style install strings like: ">=0.1.0 <0.2.0" are not supported. Generally the tuple style is preferred when using range specifications as the npm style strings have a simplistic regex splitter. See Version for more details on matching versions.","ref":"Lightning.AdaptorService.html"},{"type":"function","title":"Lightning.AdaptorService.build_aliased_name/1","doc":"Turns a package name and version into a string for NPM. Since multiple versions of the same package can be installed, it's important to rely on npms built-in package aliasing. E.g. @openfn/language-http@1.2.8 turns into: ` @openfn / language - http - 1.2 . 8 @npm : @openfn / language - http @ 1.2 . 8 ` Which is pretty long winded but necessary for the reason above. If using this module as a base, it's likely you would need to adaptor this to suit your particular naming strategy.","ref":"Lightning.AdaptorService.html#build_aliased_name/1"},{"type":"function","title":"Lightning.AdaptorService.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorService.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorService.find_adaptor/2","doc":"","ref":"Lightning.AdaptorService.html#find_adaptor/2"},{"type":"function","title":"Lightning.AdaptorService.get_adaptors/1","doc":"","ref":"Lightning.AdaptorService.html#get_adaptors/1"},{"type":"function","title":"Lightning.AdaptorService.install/2","doc":"","ref":"Lightning.AdaptorService.html#install/2"},{"type":"function","title":"Lightning.AdaptorService.install!/2","doc":"","ref":"Lightning.AdaptorService.html#install!/2"},{"type":"function","title":"Lightning.AdaptorService.installed?/2","doc":"","ref":"Lightning.AdaptorService.html#installed?/2"},{"type":"function","title":"Lightning.AdaptorService.resolve_package_name/1","doc":"","ref":"Lightning.AdaptorService.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorService.start_link/1","doc":"","ref":"Lightning.AdaptorService.html#start_link/1"},{"type":"type","title":"Lightning.AdaptorService.package_spec/0","doc":"","ref":"Lightning.AdaptorService.html#t:package_spec/0"},{"type":"module","title":"Lightning.Attempt","doc":"Ecto model for Attempts.","ref":"Lightning.Attempt.html"},{"type":"function","title":"Lightning.Attempt.new/1","doc":"","ref":"Lightning.Attempt.html#new/1"},{"type":"type","title":"Lightning.Attempt.t/0","doc":"","ref":"Lightning.Attempt.html#t:t/0"},{"type":"module","title":"Lightning.AttemptRun","doc":"Ecto model for an Attempts Runs.","ref":"Lightning.AttemptRun.html"},{"type":"function","title":"Lightning.AttemptRun.new/1","doc":"","ref":"Lightning.AttemptRun.html#new/1"},{"type":"function","title":"Lightning.AttemptRun.new/2","doc":"","ref":"Lightning.AttemptRun.html#new/2"},{"type":"type","title":"Lightning.AttemptRun.t/0","doc":"","ref":"Lightning.AttemptRun.html#t:t/0"},{"type":"module","title":"Lightning.AttemptService","doc":"The Attempts context.","ref":"Lightning.AttemptService.html"},{"type":"function","title":"Lightning.AttemptService.append/2","doc":"Adds an Attempt to an unsaved Run When given an Attempt, it simply adds the Run to a new AttemptRun. However when given an AttemptRun, the Run (from the AttemptRun) is set as the previous Run for the new unsaved Run.","ref":"Lightning.AttemptService.html#append/2"},{"type":"function","title":"Lightning.AttemptService.build_attempt/2","doc":"","ref":"Lightning.AttemptService.html#build_attempt/2"},{"type":"function","title":"Lightning.AttemptService.calculate_runs/3","doc":"","ref":"Lightning.AttemptService.html#calculate_runs/3"},{"type":"function","title":"Lightning.AttemptService.create_attempt/3","doc":"Create an attempt Examples iex> create_attempt ( %{ field : value } ) { :ok , % Attempt { } } iex> create_attempt ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.AttemptService.html#create_attempt/3"},{"type":"function","title":"Lightning.AttemptService.get_for_rerun/2","doc":"","ref":"Lightning.AttemptService.html#get_for_rerun/2"},{"type":"function","title":"Lightning.AttemptService.get_last_attempt_for/1","doc":"Get the latest attempt associated to a given run","ref":"Lightning.AttemptService.html#get_last_attempt_for/1"},{"type":"function","title":"Lightning.AttemptService.get_workflow_for/1","doc":"","ref":"Lightning.AttemptService.html#get_workflow_for/1"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_job/2","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of workorder ids that are associated to the given Job","ref":"Lightning.AttemptService.html#list_for_rerun_from_job/2"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_start/1","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of work order ids.","ref":"Lightning.AttemptService.html#list_for_rerun_from_start/1"},{"type":"function","title":"Lightning.AttemptService.retry/3","doc":"Creates a new Attempt starting from a given run. All upstream/prior Runs that were performed on that attempt are associated with the new Attempt, where as the specified run is used to create a new one and is added to the Attempt. Any runs downstream from the Run given are ignored.","ref":"Lightning.AttemptService.html#retry/3"},{"type":"function","title":"Lightning.AttemptService.retry_many/2","doc":"Creates new Attempts for each pair of corresponding AttemptRun and InvocationReason.","ref":"Lightning.AttemptService.html#retry_many/2"},{"type":"module","title":"Lightning.Auditing","doc":"Context for working with Audit records.","ref":"Lightning.Auditing.html"},{"type":"function","title":"Lightning.Auditing.list_all/1","doc":"","ref":"Lightning.Auditing.html#list_all/1"},{"type":"module","title":"Lightning.Auditing.Model","doc":"Macro module to add common model behaviour to a given Ecto model","ref":"Lightning.Auditing.Model.html"},{"type":"function","title":"Lightning.Auditing.Model.event/5","doc":"Creates a schema changeset for the event identified by row_id and caused by actor_id . The given metadata can be either nil , Ecto.Changeset , struct or map. It returns :no_changes in case of an Ecto.Changeset metadata that changed nothing or an Ecto.Changeset with the event ready to be inserted.","ref":"Lightning.Auditing.Model.html#event/5"},{"type":"function","title":"Lightning.Auditing.Model.save/2","doc":"Saves the event to the Repo . In case of nothing changes, do nothing. It returns {:ok, :no_changes} if nothing changed, {:ok, struct} if the log has been successfully saved or {:error, changeset} in case of error.","ref":"Lightning.Auditing.Model.html#save/2"},{"type":"module","title":"Lightning.AuthProviders","doc":"Context module for dealing with external Auth Providers.","ref":"Lightning.AuthProviders.html"},{"type":"function","title":"Lightning.AuthProviders.build_handler/2","doc":"","ref":"Lightning.AuthProviders.html#build_handler/2"},{"type":"function","title":"Lightning.AuthProviders.create/1","doc":"","ref":"Lightning.AuthProviders.html#create/1"},{"type":"function","title":"Lightning.AuthProviders.create_handler/1","doc":"","ref":"Lightning.AuthProviders.html#create_handler/1"},{"type":"function","title":"Lightning.AuthProviders.delete!/1","doc":"","ref":"Lightning.AuthProviders.html#delete!/1"},{"type":"function","title":"Lightning.AuthProviders.get_authorize_url/1","doc":"Retrieve the authorization url for a given handler or handler name.","ref":"Lightning.AuthProviders.html#get_authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.get_existing/0","doc":"","ref":"Lightning.AuthProviders.html#get_existing/0"},{"type":"function","title":"Lightning.AuthProviders.get_existing/1","doc":"","ref":"Lightning.AuthProviders.html#get_existing/1"},{"type":"function","title":"Lightning.AuthProviders.get_handler/1","doc":"","ref":"Lightning.AuthProviders.html#get_handler/1"},{"type":"function","title":"Lightning.AuthProviders.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.new/0","doc":"","ref":"Lightning.AuthProviders.html#new/0"},{"type":"function","title":"Lightning.AuthProviders.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.html#remove_handler/1"},{"type":"function","title":"Lightning.AuthProviders.update/2","doc":"","ref":"Lightning.AuthProviders.html#update/2"},{"type":"module","title":"Lightning.AuthProviders.AuthConfig","doc":"AuthProvider model","ref":"Lightning.AuthProviders.AuthConfig.html"},{"type":"type","title":"Lightning.AuthProviders.AuthConfig.t/0","doc":"","ref":"Lightning.AuthProviders.AuthConfig.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.CacheWarmer","doc":"Dummy warmer which caches database rows every 30s.","ref":"Lightning.AuthProviders.CacheWarmer.html"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AuthProviders.CacheWarmer.html#child_spec/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.execute/1","doc":"Executes this cache warmer with a connection.","ref":"Lightning.AuthProviders.CacheWarmer.html#execute/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.interval/0","doc":"Returns the interval for this warmer.","ref":"Lightning.AuthProviders.CacheWarmer.html#interval/0"},{"type":"module","title":"Lightning.AuthProviders.Google","doc":"","ref":"Lightning.AuthProviders.Google.html"},{"type":"function","title":"Lightning.AuthProviders.Google.authorize_url/2","doc":"","ref":"Lightning.AuthProviders.Google.html#authorize_url/2"},{"type":"function","title":"Lightning.AuthProviders.Google.build_client/1","doc":"Builds a new client","ref":"Lightning.AuthProviders.Google.html#build_client/1"},{"type":"function","title":"Lightning.AuthProviders.Google.get_config/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_config/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown!/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown!/0"},{"type":"function","title":"Lightning.AuthProviders.Google.refresh_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#refresh_token/2"},{"type":"module","title":"Lightning.AuthProviders.Handler","doc":"Module which wraps Oauth configuration and a WellKnown document into a convenient struct that can be used to authenticate users against any OIDC compliant provider.","ref":"Lightning.AuthProviders.Handler.html"},{"type":"function","title":"Lightning.AuthProviders.Handler.authorize_url/1","doc":"","ref":"Lightning.AuthProviders.Handler.html#authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.from_model/1","doc":"Returns a Handler from a given AuthConfig","ref":"Lightning.AuthProviders.Handler.html#from_model/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_token/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.new/2","doc":"Create a new Provider struct, expects a name and opts: :client_id - The providers issued id :client_secret - Secret for the client :redirect_uri - The URI for redirecting after authentication, usually the callback url in the router. :wellknown - A AuthProviders.WellKnown struct with the providers .well-known/openid-configuration .","ref":"Lightning.AuthProviders.Handler.html#new/2"},{"type":"type","title":"Lightning.AuthProviders.Handler.opts/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:opts/0"},{"type":"type","title":"Lightning.AuthProviders.Handler.t/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.Store","doc":"Store module for caching Handlers. Since Handlers often have to fetch their .well-known files when being initialized we cache these in order to avoid repeatedly making HTTP requests to a providers API.","ref":"Lightning.AuthProviders.Store.html"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.Store.put_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#put_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.Store.html#remove_handler/1"},{"type":"type","title":"Lightning.AuthProviders.Store.finder/0","doc":"","ref":"Lightning.AuthProviders.Store.html#t:finder/0"},{"type":"module","title":"Lightning.AuthProviders.WellKnown","doc":"A datastructure to fetch and hold information about a given OIDC/OAuth provider","ref":"Lightning.AuthProviders.WellKnown.html"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch!/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch!/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.new/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#new/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_options/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_params/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AuthProviders.WellKnown.html#request/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AuthProviders.WellKnown.html#request/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AuthProviders.WellKnown.html#request!/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AuthProviders.WellKnown.html#start/0"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#stream_next/1"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.body/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:body/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.headers/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:headers/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.method/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:method/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.options/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:options/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.params/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:params/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.request/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:request/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.t/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:t/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.url/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:url/0"},{"type":"module","title":"Lightning.CLI","doc":"Module providing facilities to make calls to the OpenFn CLI. See @openfn/cli","ref":"Lightning.CLI.html"},{"type":"function","title":"Lightning.CLI.execute/1","doc":"Execute a command in a child process and parse the results.","ref":"Lightning.CLI.html#execute/1"},{"type":"function","title":"Lightning.CLI.metadata/2","doc":"Retrieve metadata for a given adaptor and configuration.","ref":"Lightning.CLI.html#metadata/2"},{"type":"module","title":"Lightning.CLI.Result","doc":"Struct that wraps the output of an OpenFn CLI call. Containing the keys: start_time end_time status logs Logs The OpenFn CLI returns JSON formatted log lines, which are decoded and added to a Result struct. There are two kinds of output: { "level" :"<<level>>" , "name" :"<<module>>" , "message" :"..." ] , "time" :< < timestamp >> } These are usually for general logging, and debugging. { "message" : [ "<<message|filepath|output>>" ] } The above is the equivalent of the output of a command","ref":"Lightning.CLI.Result.html"},{"type":"function","title":"Lightning.CLI.Result.get_messages/1","doc":"Returns message type log lines from a Result .","ref":"Lightning.CLI.Result.html#get_messages/1"},{"type":"function","title":"Lightning.CLI.Result.new/1","doc":"","ref":"Lightning.CLI.Result.html#new/1"},{"type":"function","title":"Lightning.CLI.Result.parse/2","doc":"","ref":"Lightning.CLI.Result.html#parse/2"},{"type":"type","title":"Lightning.CLI.Result.t/0","doc":"","ref":"Lightning.CLI.Result.html#t:t/0"},{"type":"module","title":"Lightning.Demo","doc":"Demo encapsulates logic for setting up a demonstration site.","ref":"Lightning.Demo.html"},{"type":"function","title":"Lightning.Demo.reset_demo/0","doc":"Deletes everything in the database including the superuser and creates a set of publicly available users for a demo site via a command that can be run on Kubernetes-deployed systems.","ref":"Lightning.Demo.html#reset_demo/0"},{"type":"module","title":"Lightning.ExportUtils","doc":"Module that expose a function generating a complete and valid yaml string from a project and its workflows.","ref":"Lightning.ExportUtils.html"},{"type":"function","title":"Lightning.ExportUtils.build_yaml_tree/2","doc":"","ref":"Lightning.ExportUtils.html#build_yaml_tree/2"},{"type":"function","title":"Lightning.ExportUtils.generate_new_yaml/1","doc":"","ref":"Lightning.ExportUtils.html#generate_new_yaml/1"},{"type":"module","title":"Lightning.FailureNotifierView","doc":"","ref":"Lightning.FailureNotifierView.html"},{"type":"function","title":"Lightning.FailureNotifierView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"Lightning.FailureNotifierView.html#__resource__/0"},{"type":"function","title":"Lightning.FailureNotifierView.failure_alert.html/1","doc":"","ref":"Lightning.FailureNotifierView.html#failure_alert.html/1"},{"type":"function","title":"Lightning.FailureNotifierView.render/2","doc":"Renders the given template locally.","ref":"Lightning.FailureNotifierView.html#render/2"},{"type":"function","title":"Lightning.FailureNotifierView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"Lightning.FailureNotifierView.html#template_not_found/2"},{"type":"module","title":"Lightning.Helpers","doc":"Common functions for the context","ref":"Lightning.Helpers.html"},{"type":"function","title":"Lightning.Helpers.coerce_json_field/2","doc":"Changes a given maps field from a json string to a map. If it cannot be converted, it leaves the original value","ref":"Lightning.Helpers.html#coerce_json_field/2"},{"type":"function","title":"Lightning.Helpers.indefinite_article/1","doc":"","ref":"Lightning.Helpers.html#indefinite_article/1"},{"type":"function","title":"Lightning.Helpers.json_safe/1","doc":"Recursively ensures a given map is safe to convert to JSON, where all keys are strings and all values are json safe (primitive values).","ref":"Lightning.Helpers.html#json_safe/1"},{"type":"function","title":"Lightning.Helpers.ms_to_human/1","doc":"Converts milliseconds (integer) to a human duration, such as "1 minute" or "45 years, 6 months, 5 days, 21 hours, 12 minutes, 34 seconds" using Timex.Format.Duration.Formatters.Humanized.format() .","ref":"Lightning.Helpers.html#ms_to_human/1"},{"type":"module","title":"Lightning.MetadataService","doc":"Retrieves metadata for a given credential and adaptor using the OpenFn CLI.","ref":"Lightning.MetadataService.html"},{"type":"function","title":"Lightning.MetadataService.fetch/2","doc":"Retrieve metadata for a given adaptor and credential. The adaptor must be an npm specification.","ref":"Lightning.MetadataService.html#fetch/2"},{"type":"module","title":"Lightning.Name","doc":"Generates a random names.","ref":"Lightning.Name.html"},{"type":"function","title":"Lightning.Name.generate/1","doc":"","ref":"Lightning.Name.html#generate/1"},{"type":"module","title":"Lightning.ObanManager","doc":"The Oban Manager","ref":"Lightning.ObanManager.html"},{"type":"function","title":"Lightning.ObanManager.handle_event/4","doc":"","ref":"Lightning.ObanManager.html#handle_event/4"},{"type":"module","title":"Lightning.Policies.Permissions","doc":"This module defines a unique interface managing authorizations in Lightning. Users in Lightning have instance-wide and project-wide roles which determine their level of access to resources in the application. Fo rmore details see the documentation . These authorizations policies are all implemented under the lib/lightning/policies folder. In that folder you can find 3 files: The users.ex file has all the policies for the instances wide access levels The project_users.ex file has all the policies for the project wide access levels The permissions.ex file defines the Lightning.Policies.Permissions.can/4 interface. Which is a wrapper around the Bodyguard.permit/4 function. We use that interface to be able to harmonize the use of policies accross the entire app. All the policies are tested in the test/lightning/policies folder. And the test are written in a way that allows the reader to quickly who can do what in the app. We have two variants of the Lightning.Policies.Permissions.can/4 interface: Lightning.Policies.Permissions.can(policy, action, actor, resource) returns :ok if the actor can perform the action on the resource and {:error, :unauthorized} otherwise. Lightning.Policies.Permissions.can?(policy, action, actor, resource) returns true if the actor can perform the action on the resource and false otherwise. Here is an example of how we the Lightning.Policies.Permissions.can/4 interface to check if the a user can edit a job or not can_edit_job = Lightning.Policies.ProjectUsers |> Lightning.Policies.Permissions . can? ( :edit_job , socket . assigns . current_user , socket . assigns . project ) if can_edit_job do # allow user to edit the job else # quick user out end","ref":"Lightning.Policies.Permissions.html"},{"type":"function","title":"Lightning.Policies.Permissions.can/4","doc":"checks if user has the permissions to apply action using some policy module Returns :ok if user can apply action and {:error, :unauthorized} otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) :ok iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) { :error , :unauthorized }","ref":"Lightning.Policies.Permissions.html#can/4"},{"type":"function","title":"Lightning.Policies.Permissions.can?/4","doc":"same as can/4 but returns true if user can apply action and false otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) true iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) false","ref":"Lightning.Policies.Permissions.html#can?/4"},{"type":"module","title":"Lightning.Policies.ProjectUsers","doc":"The Bodyguard Policy module for projects members roles.","ref":"Lightning.Policies.ProjectUsers.html"},{"type":"function","title":"Lightning.Policies.ProjectUsers.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action in that project and false if they cannot. Note that permissions are grouped by action, rather than by user role. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.ProjectUsers.html#authorize/3"},{"type":"type","title":"Lightning.Policies.ProjectUsers.actions/0","doc":"","ref":"Lightning.Policies.ProjectUsers.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Provisioning","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Provisioning.html"},{"type":"function","title":"Lightning.Policies.Provisioning.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied. Only a superuser can provision a new project. Owners and admins can update existing projects.","ref":"Lightning.Policies.Provisioning.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Provisioning.actions/0","doc":"","ref":"Lightning.Policies.Provisioning.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Users","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Users.html"},{"type":"function","title":"Lightning.Policies.Users.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.Users.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Users.actions/0","doc":"","ref":"Lightning.Policies.Users.html#t:actions/0"},{"type":"module","title":"Lightning.Release","doc":"Used for executing DB release tasks when run in production without Mix installed.","ref":"Lightning.Release.html"},{"type":"function","title":"Lightning.Release.create_db/0","doc":"","ref":"Lightning.Release.html#create_db/0"},{"type":"function","title":"Lightning.Release.load_app/0","doc":"","ref":"Lightning.Release.html#load_app/0"},{"type":"function","title":"Lightning.Release.migrate/0","doc":"","ref":"Lightning.Release.html#migrate/0"},{"type":"function","title":"Lightning.Release.rollback/2","doc":"","ref":"Lightning.Release.html#rollback/2"},{"type":"module","title":"Lightning.Repo","doc":"","ref":"Lightning.Repo.html"},{"type":"function","title":"Lightning.Repo.aggregate/3","doc":"Callback implementation for Ecto.Repo.aggregate/3 .","ref":"Lightning.Repo.html#aggregate/3"},{"type":"function","title":"Lightning.Repo.aggregate/4","doc":"Callback implementation for Ecto.Repo.aggregate/4 .","ref":"Lightning.Repo.html#aggregate/4"},{"type":"function","title":"Lightning.Repo.all/2","doc":"Callback implementation for Ecto.Repo.all/2 .","ref":"Lightning.Repo.html#all/2"},{"type":"function","title":"Lightning.Repo.checked_out?/0","doc":"Callback implementation for Ecto.Repo.checked_out?/0 .","ref":"Lightning.Repo.html#checked_out?/0"},{"type":"function","title":"Lightning.Repo.checkout/2","doc":"Callback implementation for Ecto.Repo.checkout/2 .","ref":"Lightning.Repo.html#checkout/2"},{"type":"function","title":"Lightning.Repo.child_spec/1","doc":"","ref":"Lightning.Repo.html#child_spec/1"},{"type":"function","title":"Lightning.Repo.config/0","doc":"Callback implementation for Ecto.Repo.config/0 .","ref":"Lightning.Repo.html#config/0"},{"type":"function","title":"Lightning.Repo.default_options/1","doc":"Callback implementation for Ecto.Repo.default_options/1 .","ref":"Lightning.Repo.html#default_options/1"},{"type":"function","title":"Lightning.Repo.delete/2","doc":"Callback implementation for Ecto.Repo.delete/2 .","ref":"Lightning.Repo.html#delete/2"},{"type":"function","title":"Lightning.Repo.delete!/2","doc":"Callback implementation for Ecto.Repo.delete!/2 .","ref":"Lightning.Repo.html#delete!/2"},{"type":"function","title":"Lightning.Repo.delete_all/2","doc":"Callback implementation for Ecto.Repo.delete_all/2 .","ref":"Lightning.Repo.html#delete_all/2"},{"type":"function","title":"Lightning.Repo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the pool to disconnect within the given interval. See Ecto.Adapters.SQL.disconnect_all/3 for more information.","ref":"Lightning.Repo.html#disconnect_all/2"},{"type":"function","title":"Lightning.Repo.exists?/2","doc":"Callback implementation for Ecto.Repo.exists?/2 .","ref":"Lightning.Repo.html#exists?/2"},{"type":"function","title":"Lightning.Repo.explain/3","doc":"A convenience function for SQL-based repositories that executes an EXPLAIN statement or similar depending on the adapter to obtain statistics for the given query. See Ecto.Adapters.SQL.explain/4 for more information.","ref":"Lightning.Repo.html#explain/3"},{"type":"function","title":"Lightning.Repo.get/3","doc":"Callback implementation for Ecto.Repo.get/3 .","ref":"Lightning.Repo.html#get/3"},{"type":"function","title":"Lightning.Repo.get!/3","doc":"Callback implementation for Ecto.Repo.get!/3 .","ref":"Lightning.Repo.html#get!/3"},{"type":"function","title":"Lightning.Repo.get_by/3","doc":"Callback implementation for Ecto.Repo.get_by/3 .","ref":"Lightning.Repo.html#get_by/3"},{"type":"function","title":"Lightning.Repo.get_by!/3","doc":"Callback implementation for Ecto.Repo.get_by!/3 .","ref":"Lightning.Repo.html#get_by!/3"},{"type":"function","title":"Lightning.Repo.get_dynamic_repo/0","doc":"Callback implementation for Ecto.Repo.get_dynamic_repo/0 .","ref":"Lightning.Repo.html#get_dynamic_repo/0"},{"type":"function","title":"Lightning.Repo.in_transaction?/0","doc":"Callback implementation for Ecto.Repo.in_transaction?/0 .","ref":"Lightning.Repo.html#in_transaction?/0"},{"type":"function","title":"Lightning.Repo.insert/2","doc":"Callback implementation for Ecto.Repo.insert/2 .","ref":"Lightning.Repo.html#insert/2"},{"type":"function","title":"Lightning.Repo.insert!/2","doc":"Callback implementation for Ecto.Repo.insert!/2 .","ref":"Lightning.Repo.html#insert!/2"},{"type":"function","title":"Lightning.Repo.insert_all/3","doc":"Callback implementation for Ecto.Repo.insert_all/3 .","ref":"Lightning.Repo.html#insert_all/3"},{"type":"function","title":"Lightning.Repo.insert_or_update/2","doc":"Callback implementation for Ecto.Repo.insert_or_update/2 .","ref":"Lightning.Repo.html#insert_or_update/2"},{"type":"function","title":"Lightning.Repo.insert_or_update!/2","doc":"Callback implementation for Ecto.Repo.insert_or_update!/2 .","ref":"Lightning.Repo.html#insert_or_update!/2"},{"type":"function","title":"Lightning.Repo.load/2","doc":"Callback implementation for Ecto.Repo.load/2 .","ref":"Lightning.Repo.html#load/2"},{"type":"function","title":"Lightning.Repo.one/2","doc":"Callback implementation for Ecto.Repo.one/2 .","ref":"Lightning.Repo.html#one/2"},{"type":"function","title":"Lightning.Repo.one!/2","doc":"Callback implementation for Ecto.Repo.one!/2 .","ref":"Lightning.Repo.html#one!/2"},{"type":"function","title":"Lightning.Repo.paginate/2","doc":"","ref":"Lightning.Repo.html#paginate/2"},{"type":"function","title":"Lightning.Repo.preload/3","doc":"Callback implementation for Ecto.Repo.preload/3 .","ref":"Lightning.Repo.html#preload/3"},{"type":"function","title":"Lightning.Repo.prepare_query/3","doc":"Callback implementation for Ecto.Repo.prepare_query/3 .","ref":"Lightning.Repo.html#prepare_query/3"},{"type":"function","title":"Lightning.Repo.put_dynamic_repo/1","doc":"Callback implementation for Ecto.Repo.put_dynamic_repo/1 .","ref":"Lightning.Repo.html#put_dynamic_repo/1"},{"type":"function","title":"Lightning.Repo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query/4 for more information.","ref":"Lightning.Repo.html#query/3"},{"type":"function","title":"Lightning.Repo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query!/4 for more information.","ref":"Lightning.Repo.html#query!/3"},{"type":"function","title":"Lightning.Repo.query_many/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many/4 for more information.","ref":"Lightning.Repo.html#query_many/3"},{"type":"function","title":"Lightning.Repo.query_many!/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many!/4 for more information.","ref":"Lightning.Repo.html#query_many!/3"},{"type":"function","title":"Lightning.Repo.reload/2","doc":"Callback implementation for Ecto.Repo.reload/2 .","ref":"Lightning.Repo.html#reload/2"},{"type":"function","title":"Lightning.Repo.reload!/2","doc":"Callback implementation for Ecto.Repo.reload!/2 .","ref":"Lightning.Repo.html#reload!/2"},{"type":"function","title":"Lightning.Repo.rollback/1","doc":"Callback implementation for Ecto.Repo.rollback/1 .","ref":"Lightning.Repo.html#rollback/1"},{"type":"function","title":"Lightning.Repo.scrivener_defaults/0","doc":"","ref":"Lightning.Repo.html#scrivener_defaults/0"},{"type":"function","title":"Lightning.Repo.start_link/1","doc":"Callback implementation for Ecto.Repo.start_link/1 .","ref":"Lightning.Repo.html#start_link/1"},{"type":"function","title":"Lightning.Repo.stop/1","doc":"Callback implementation for Ecto.Repo.stop/1 .","ref":"Lightning.Repo.html#stop/1"},{"type":"function","title":"Lightning.Repo.stream/2","doc":"Callback implementation for Ecto.Repo.stream/2 .","ref":"Lightning.Repo.html#stream/2"},{"type":"function","title":"Lightning.Repo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL. See Ecto.Adapters.SQL.to_sql/3 for more information.","ref":"Lightning.Repo.html#to_sql/2"},{"type":"function","title":"Lightning.Repo.transaction/2","doc":"Callback implementation for Ecto.Repo.transaction/2 .","ref":"Lightning.Repo.html#transaction/2"},{"type":"function","title":"Lightning.Repo.update/2","doc":"Callback implementation for Ecto.Repo.update/2 .","ref":"Lightning.Repo.html#update/2"},{"type":"function","title":"Lightning.Repo.update!/2","doc":"Callback implementation for Ecto.Repo.update!/2 .","ref":"Lightning.Repo.html#update!/2"},{"type":"function","title":"Lightning.Repo.update_all/3","doc":"Callback implementation for Ecto.Repo.update_all/3 .","ref":"Lightning.Repo.html#update_all/3"},{"type":"module","title":"Lightning.RunSearchForm","doc":"Run filtering search form.","ref":"Lightning.RunSearchForm.html"},{"type":"module","title":"Lightning.SafetyString","doc":"Utilities for securely encoding serializable structs, lists and strings into URL-safe strings. In order to pass the state around in a URL, in a manner that protects secrets from leaking - and allows us to avoid persistance we take a set of parameters and: Encode into a URI query string gzip it to save characters encrypt the string base64 encode it for URI encoding safety","ref":"Lightning.SafetyString.html"},{"type":"function","title":"Lightning.SafetyString.decode/1","doc":"","ref":"Lightning.SafetyString.html#decode/1"},{"type":"function","title":"Lightning.SafetyString.encode/1","doc":"","ref":"Lightning.SafetyString.html#encode/1"},{"type":"module","title":"Lightning.Scrubber","doc":"Process used to scrub strings of sensitive information. Can be started via start_link/1 . { :ok , scrubber } = Lightning.Scrubber . start_link ( samples : Lightning.Credentials . sensitive_values_for ( credential ) ) Takes an optional :name key, in case you need to name the process.","ref":"Lightning.Scrubber.html"},{"type":"function","title":"Lightning.Scrubber.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Scrubber.html#child_spec/1"},{"type":"function","title":"Lightning.Scrubber.encode_samples/1","doc":"Prepare a list of sensitive samples (strings) into a potentially bigger list composed of variations a sample may appear.","ref":"Lightning.Scrubber.html#encode_samples/1"},{"type":"function","title":"Lightning.Scrubber.samples/1","doc":"","ref":"Lightning.Scrubber.html#samples/1"},{"type":"function","title":"Lightning.Scrubber.scrub/2","doc":"","ref":"Lightning.Scrubber.html#scrub/2"},{"type":"function","title":"Lightning.Scrubber.start_link/1","doc":"","ref":"Lightning.Scrubber.html#start_link/1"},{"type":"module","title":"Lightning.SetupUtils","doc":"SetupUtils encapsulates logic for setting up initial data for various sites.","ref":"Lightning.SetupUtils.html"},{"type":"function","title":"Lightning.SetupUtils.add_and_update_runs/3","doc":"","ref":"Lightning.SetupUtils.html#add_and_update_runs/3"},{"type":"function","title":"Lightning.SetupUtils.create_dhis2_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_dhis2_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_openhie_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_openhie_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_starter_project/2","doc":"","ref":"Lightning.SetupUtils.html#create_starter_project/2"},{"type":"function","title":"Lightning.SetupUtils.setup_demo/1","doc":"Creates initial data and returns the created records.","ref":"Lightning.SetupUtils.html#setup_demo/1"},{"type":"function","title":"Lightning.SetupUtils.tear_down/1","doc":"","ref":"Lightning.SetupUtils.html#tear_down/1"},{"type":"module","title":"Lightning.TaskWorker","doc":"A TaskWorker with concurrency limits. A simple concurrency limiter that wraps Task.Supervisor , which already does have the ability to specify max_children ; it throws an error when that limit is exceeded. To use it, start it like any other process; ideally in your supervision tree. ... , { Lightning.TaskWorker , name : :cli_task_worker , max_tasks : 4 } Options :max_tasks Defaults to the number of system schedulers available to the vm.","ref":"Lightning.TaskWorker.html"},{"type":"function","title":"Lightning.TaskWorker.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.TaskWorker.html#child_spec/1"},{"type":"function","title":"Lightning.TaskWorker.get_status/1","doc":"","ref":"Lightning.TaskWorker.html#get_status/1"},{"type":"function","title":"Lightning.TaskWorker.start_link/1","doc":"","ref":"Lightning.TaskWorker.html#start_link/1"},{"type":"function","title":"Lightning.TaskWorker.start_task/2","doc":"","ref":"Lightning.TaskWorker.html#start_task/2"},{"type":"module","title":"Lightning.Validators","doc":"Extra validators for Ecto.Changeset.","ref":"Lightning.Validators.html"},{"type":"function","title":"Lightning.Validators.validate_exclusive/3","doc":"Validate that only one of the fields is set at a time. Example: changeset |> validate_exclusive ( [ :source_job_id , :source_trigger_id ] , "source_job_id and source_trigger_id are mutually exclusive" )","ref":"Lightning.Validators.html#validate_exclusive/3"},{"type":"function","title":"Lightning.Validators.validate_one_required/3","doc":"Validate that at least one of the fields is set.","ref":"Lightning.Validators.html#validate_one_required/3"},{"type":"module","title":"Lightning.Vault","doc":"Module for handling the encryption and decryption of database fields.","ref":"Lightning.Vault.html"},{"type":"function","title":"Lightning.Vault.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Vault.html#child_spec/1"},{"type":"function","title":"Lightning.Vault.start_link/1","doc":"","ref":"Lightning.Vault.html#start_link/1"},{"type":"module","title":"Lightning.WorkOrder","doc":"Ecto model for Workorders.","ref":"Lightning.WorkOrder.html"},{"type":"function","title":"Lightning.WorkOrder.new/0","doc":"","ref":"Lightning.WorkOrder.html#new/0"},{"type":"type","title":"Lightning.WorkOrder.t/0","doc":"","ref":"Lightning.WorkOrder.html#t:t/0"},{"type":"module","title":"Lightning.WorkOrderService","doc":"The WorkOrderService.","ref":"Lightning.WorkOrderService.html"},{"type":"function","title":"Lightning.WorkOrderService.attempt_updated/1","doc":"","ref":"Lightning.WorkOrderService.html#attempt_updated/1"},{"type":"function","title":"Lightning.WorkOrderService.build/2","doc":"","ref":"Lightning.WorkOrderService.html#build/2"},{"type":"function","title":"Lightning.WorkOrderService.create_manual_workorder/3","doc":"","ref":"Lightning.WorkOrderService.html#create_manual_workorder/3"},{"type":"function","title":"Lightning.WorkOrderService.create_webhook_workorder/2","doc":"","ref":"Lightning.WorkOrderService.html#create_webhook_workorder/2"},{"type":"function","title":"Lightning.WorkOrderService.create_work_order/1","doc":"Creates a work_order. Examples iex> create_work_order ( %{ field : value } ) { :ok , % WorkOrder { } } iex> create_work_order ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.WorkOrderService.html#create_work_order/1"},{"type":"function","title":"Lightning.WorkOrderService.multi_for/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for/3"},{"type":"function","title":"Lightning.WorkOrderService.multi_for_manual/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for_manual/3"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_run/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_run/2"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_runs/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_runs/2"},{"type":"function","title":"Lightning.WorkOrderService.subscribe/1","doc":"","ref":"Lightning.WorkOrderService.html#subscribe/1"},{"type":"module","title":"Lightning.Workflows","doc":"The Workflows context.","ref":"Lightning.Workflows.html"},{"type":"function","title":"Lightning.Workflows.build_trigger/1","doc":"Builds a Trigger","ref":"Lightning.Workflows.html#build_trigger/1"},{"type":"function","title":"Lightning.Workflows.change_workflow/2","doc":"Returns an %Ecto.Changeset{} for tracking workflow changes. Examples iex> change_workflow ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#change_workflow/2"},{"type":"function","title":"Lightning.Workflows.create_edge/1","doc":"Creates an edge","ref":"Lightning.Workflows.html#create_edge/1"},{"type":"function","title":"Lightning.Workflows.create_workflow/1","doc":"Creates a workflow. Examples iex> create_workflow ( %{ field : value } ) { :ok , % Workflow { } } iex> create_workflow ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#create_workflow/1"},{"type":"function","title":"Lightning.Workflows.delete_workflow/1","doc":"Deletes a workflow. Examples iex> delete_workflow ( workflow ) { :ok , % Workflow { } } iex> delete_workflow ( workflow ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#delete_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_edge_by_webhook/1","doc":"Gets a Single Edge by it's webhook trigger.","ref":"Lightning.Workflows.html#get_edge_by_webhook/1"},{"type":"function","title":"Lightning.Workflows.get_edges_for_cron_execution/1","doc":"Returns a list of edges with jobs to execute, given a current timestamp in Unix. This is used by the scheduler, which calls this function once every minute.","ref":"Lightning.Workflows.html#get_edges_for_cron_execution/1"},{"type":"function","title":"Lightning.Workflows.get_workflow/1","doc":"","ref":"Lightning.Workflows.html#get_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_workflow!/1","doc":"Gets a single workflow. Raises Ecto.NoResultsError if the Workflow does not exist. Examples iex> get_workflow! ( 123 ) % Workflow { } iex> get_workflow! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Workflows.html#get_workflow!/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for/1","doc":"Retrieves a list of Workflows with their jobs and triggers preloaded.","ref":"Lightning.Workflows.html#get_workflows_for/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for_query/1","doc":"","ref":"Lightning.Workflows.html#get_workflows_for_query/1"},{"type":"function","title":"Lightning.Workflows.list_workflows/0","doc":"Returns the list of workflows. Examples iex> list_workflows ( ) [ % Workflow { } , ... ]","ref":"Lightning.Workflows.html#list_workflows/0"},{"type":"function","title":"Lightning.Workflows.mark_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the workflow request_deletion. Examples iex> change_request_deletion ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#mark_for_deletion/2"},{"type":"function","title":"Lightning.Workflows.to_project_space/1","doc":"","ref":"Lightning.Workflows.html#to_project_space/1"},{"type":"function","title":"Lightning.Workflows.update_trigger/2","doc":"Updates a trigger","ref":"Lightning.Workflows.html#update_trigger/2"},{"type":"function","title":"Lightning.Workflows.update_workflow/2","doc":"Updates a workflow. Examples iex> update_workflow ( workflow , %{ field : new_value } ) { :ok , % Workflow { } } iex> update_workflow ( workflow , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#update_workflow/2"},{"type":"module","title":"Lightning.Workflows.Edge","doc":"Ecto model for Workflow Edges. A Workflow Edge represents a connection between two jobs (or a trigger and a job) in a workflow. The source of the edge is either a job or a trigger. The target of the edge is always a job.","ref":"Lightning.Workflows.Edge.html"},{"type":"function","title":"Lightning.Workflows.Edge.changeset/2","doc":"","ref":"Lightning.Workflows.Edge.html#changeset/2"},{"type":"function","title":"Lightning.Workflows.Edge.new/1","doc":"","ref":"Lightning.Workflows.Edge.html#new/1"},{"type":"function","title":"Lightning.Workflows.Edge.validate/1","doc":"","ref":"Lightning.Workflows.Edge.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Edge.edge_condition/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:edge_condition/0"},{"type":"type","title":"Lightning.Workflows.Edge.t/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:t/0"},{"type":"module","title":"Lightning.Workflows.Graph","doc":"Utility to construct and manipulate a graph/plan made out of Jobs","ref":"Lightning.Workflows.Graph.html"},{"type":"function","title":"Lightning.Workflows.Graph.new/1","doc":"","ref":"Lightning.Workflows.Graph.html#new/1"},{"type":"function","title":"Lightning.Workflows.Graph.remove/2","doc":"","ref":"Lightning.Workflows.Graph.html#remove/2"},{"type":"function","title":"Lightning.Workflows.Graph.vertices/1","doc":"","ref":"Lightning.Workflows.Graph.html#vertices/1"},{"type":"type","title":"Lightning.Workflows.Graph.t/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:t/0"},{"type":"type","title":"Lightning.Workflows.Graph.vertex/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:vertex/0"},{"type":"module","title":"Lightning.Workflows.Workflow","doc":"Ecto model for Workflows. A Workflow contains the fields for defining a workflow. name A plain text identifier","ref":"Lightning.Workflows.Workflow.html"},{"type":"function","title":"Lightning.Workflows.Workflow.request_deletion_changeset/2","doc":"","ref":"Lightning.Workflows.Workflow.html#request_deletion_changeset/2"},{"type":"function","title":"Lightning.Workflows.Workflow.validate/1","doc":"","ref":"Lightning.Workflows.Workflow.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Workflow.t/0","doc":"","ref":"Lightning.Workflows.Workflow.html#t:t/0"},{"type":"module","title":"Lightning.Workorders.SearchParams","doc":"This module is used to parse search parameters for workorders and provide a query to the database.","ref":"Lightning.Workorders.SearchParams.html"},{"type":"function","title":"Lightning.Workorders.SearchParams.new/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#new/1"},{"type":"function","title":"Lightning.Workorders.SearchParams.to_uri_params/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#to_uri_params/1"},{"type":"type","title":"Lightning.Workorders.SearchParams.t/0","doc":"","ref":"Lightning.Workorders.SearchParams.html#t:t/0"},{"type":"module","title":"LightningWeb","doc":"The entrypoint for defining your web interface, such as controllers, views, channels and so on. This can be used in your application as: use LightningWeb , :controller use LightningWeb , :view The definitions below will be executed for every view, controller, etc, so keep them short and clean, focused on imports, uses and aliases. Do NOT define functions inside the quoted expressions below. Instead, define any helper function in modules and import those modules here.","ref":"LightningWeb.html"},{"type":"macro","title":"LightningWeb.__using__/1","doc":"When used, dispatch to the appropriate controller/view/etc.","ref":"LightningWeb.html#__using__/1"},{"type":"function","title":"LightningWeb.channel/0","doc":"","ref":"LightningWeb.html#channel/0"},{"type":"function","title":"LightningWeb.component/0","doc":"","ref":"LightningWeb.html#component/0"},{"type":"function","title":"LightningWeb.controller/0","doc":"","ref":"LightningWeb.html#controller/0"},{"type":"function","title":"LightningWeb.html/0","doc":"","ref":"LightningWeb.html#html/0"},{"type":"function","title":"LightningWeb.live_component/0","doc":"","ref":"LightningWeb.html#live_component/0"},{"type":"function","title":"LightningWeb.live_view/1","doc":"","ref":"LightningWeb.html#live_view/1"},{"type":"function","title":"LightningWeb.router/0","doc":"","ref":"LightningWeb.html#router/0"},{"type":"function","title":"LightningWeb.static_paths/0","doc":"","ref":"LightningWeb.html#static_paths/0"},{"type":"function","title":"LightningWeb.verified_routes/0","doc":"","ref":"LightningWeb.html#verified_routes/0"},{"type":"function","title":"LightningWeb.view/0","doc":"","ref":"LightningWeb.html#view/0"},{"type":"module","title":"LightningWeb.API.Helpers","doc":"Helpers for the API views","ref":"LightningWeb.API.Helpers.html"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_link/3","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_link/3"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_links/2","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_links/2"},{"type":"function","title":"LightningWeb.API.Helpers.url_for/2","doc":"","ref":"LightningWeb.API.Helpers.html#url_for/2"},{"type":"module","title":"LightningWeb.API.JobController","doc":"","ref":"LightningWeb.API.JobController.html"},{"type":"function","title":"LightningWeb.API.JobController.index/2","doc":"","ref":"LightningWeb.API.JobController.html#index/2"},{"type":"function","title":"LightningWeb.API.JobController.show/2","doc":"","ref":"LightningWeb.API.JobController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProjectController","doc":"","ref":"LightningWeb.API.ProjectController.html"},{"type":"function","title":"LightningWeb.API.ProjectController.index/2","doc":"","ref":"LightningWeb.API.ProjectController.html#index/2"},{"type":"function","title":"LightningWeb.API.ProjectController.show/2","doc":"","ref":"LightningWeb.API.ProjectController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProvisioningController","doc":"","ref":"LightningWeb.API.ProvisioningController.html"},{"type":"function","title":"LightningWeb.API.ProvisioningController.create/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#create/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#show/2"},{"type":"module","title":"LightningWeb.API.RunController","doc":"","ref":"LightningWeb.API.RunController.html"},{"type":"function","title":"LightningWeb.API.RunController.index/2","doc":"","ref":"LightningWeb.API.RunController.html#index/2"},{"type":"function","title":"LightningWeb.API.RunController.show/2","doc":"","ref":"LightningWeb.API.RunController.html#show/2"},{"type":"module","title":"LightningWeb.AuditLive.Index","doc":"LiveView for listing Audit events","ref":"LightningWeb.AuditLive.Index.html"},{"type":"function","title":"LightningWeb.AuditLive.Index.diff/1","doc":"","ref":"LightningWeb.AuditLive.Index.html#diff/1"},{"type":"function","title":"LightningWeb.AuditLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.AuditLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.AuthProvidersLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.AuthProvidersLive.FormComponent.html"},{"type":"module","title":"LightningWeb.AuthProvidersLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.AuthProvidersLive.Index.html"},{"type":"module","title":"LightningWeb.ChangesetJSON","doc":"Renders changesets as JSON.","ref":"LightningWeb.ChangesetJSON.html"},{"type":"function","title":"LightningWeb.ChangesetJSON.error/1","doc":"","ref":"LightningWeb.ChangesetJSON.html#error/1"},{"type":"module","title":"LightningWeb.ChangesetView","doc":"","ref":"LightningWeb.ChangesetView.html"},{"type":"function","title":"LightningWeb.ChangesetView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ChangesetView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ChangesetView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ChangesetView.html#render/2"},{"type":"function","title":"LightningWeb.ChangesetView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ChangesetView.html#template_not_found/2"},{"type":"function","title":"LightningWeb.ChangesetView.translate_errors/1","doc":"Traverses and translates changeset errors. See Ecto.Changeset.traverse_errors/2 and LightningWeb.ErrorHelpers.translate_error/1 for more details.","ref":"LightningWeb.ChangesetView.html#translate_errors/1"},{"type":"module","title":"LightningWeb.Components.Modal","doc":"A modal component that can be used to display a modal on the page. This currently isn't used anywhere but should be used in the future to replace the existing modal implementations.","ref":"LightningWeb.Components.Modal.html"},{"type":"module","title":"LightningWeb.CredentialLive.Edit","doc":"LiveView for editing a single Credential, which inturn uses LightningWeb.CredentialLive.FormComponent for common functionality.","ref":"LightningWeb.CredentialLive.Edit.html"},{"type":"function","title":"LightningWeb.CredentialLive.Edit.handle_info/2","doc":"A generic handler for forwarding updates from PubSub","ref":"LightningWeb.CredentialLive.Edit.html#handle_info/2"},{"type":"module","title":"LightningWeb.CredentialLive.FormComponent","doc":"Form Component for working with a single Credential","ref":"LightningWeb.CredentialLive.FormComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.credential_transfer/1","doc":"Attributes users ( :list ) (required) form ( :map ) (required)","ref":"LightningWeb.CredentialLive.FormComponent.html#credential_transfer/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.form_component/1","doc":"Switcher components for different types of credentials. Attributes type ( :string ) (required) form ( :map ) (required) update_body ( :any ) Slots inner_block","ref":"LightningWeb.CredentialLive.FormComponent.html#form_component/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.project_credentials/1","doc":"Attributes form ( :map ) (required) projects ( :list ) (required) selected ( :map ) (required) phx_target ( :any ) - Defaults to nil .","ref":"LightningWeb.CredentialLive.FormComponent.html#project_credentials/1"},{"type":"module","title":"LightningWeb.CredentialLive.GoogleSheetsComponent","doc":"Form component to setup a Google Sheets component. This component has several moving parts: Subscribes to a PubSub topic specially link to the component id See: LightningWeb.OauthCredentialHelper . Uses the Lightning.Google module to set up an OAuth client for generating urls, exchanging the code and requesting a new access_token . The flow for creating a new token is: Generate an authorization link which contains: The authorization url from the Google client with the applications callback_url A state string that is an encrypted set of data with the components module and id in it Once the user authorizes the client the callback is requested with a code The LightningWeb.OidcController decodes the state returned to it and does a 'broadcast_forward' which is simply a message expected to be received by a LiveView and applied to Phoenix.LiveView.send_update/3 . The component receives the code and requests a token. Any changes to the token (Credential body) are still handled by the parent component and so a update_body function is passed in to send params changes back up to update the form.","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.disabled_authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#disabled_authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.error_block/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#error_block/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.fieldset/1","doc":"Attributes form ( :map ) (required) update_body ( :any ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.userinfo/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#userinfo/1"},{"type":"module","title":"LightningWeb.CredentialLive.Index","doc":"LiveView for listing and managing credentials","ref":"LightningWeb.CredentialLive.Index.html"},{"type":"function","title":"LightningWeb.CredentialLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.CredentialLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.schema_input/1","doc":"Attributes form ( :map ) (required) schema ( :map ) (required) field ( :any ) (required)","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#schema_input/1"},{"type":"module","title":"LightningWeb.CredentialLive.RawBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.RawBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.RawBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.RawBodyComponent.html#fieldset/1"},{"type":"module","title":"LightningWeb.CredentialLive.TypePicker","doc":"","ref":"LightningWeb.CredentialLive.TypePicker.html"},{"type":"module","title":"LightningWeb.DataclipLive.Edit","doc":"LiveView for editing a single dataclip.","ref":"LightningWeb.DataclipLive.Edit.html"},{"type":"function","title":"LightningWeb.DataclipLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.FormComponent","doc":"Form Component for working with a single dataclip","ref":"LightningWeb.DataclipLive.FormComponent.html"},{"type":"function","title":"LightningWeb.DataclipLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.DataclipLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.Index","doc":"LiveView for listing and working with a list of Dataclips","ref":"LightningWeb.DataclipLive.Index.html"},{"type":"function","title":"LightningWeb.DataclipLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.DownloadsController","doc":"","ref":"LightningWeb.DownloadsController.html"},{"type":"function","title":"LightningWeb.DownloadsController.download_project_yaml/2","doc":"","ref":"LightningWeb.DownloadsController.html#download_project_yaml/2"},{"type":"module","title":"LightningWeb.Endpoint","doc":"","ref":"LightningWeb.Endpoint.html"},{"type":"function","title":"LightningWeb.Endpoint.broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast/3 .","ref":"LightningWeb.Endpoint.html#broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast!/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast!/3 .","ref":"LightningWeb.Endpoint.html#broadcast!/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from!/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from!/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from!/4"},{"type":"function","title":"LightningWeb.Endpoint.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.Endpoint.html#call/2"},{"type":"function","title":"LightningWeb.Endpoint.child_spec/1","doc":"Returns the child specification to start the endpoint under a supervision tree.","ref":"LightningWeb.Endpoint.html#child_spec/1"},{"type":"function","title":"LightningWeb.Endpoint.config/2","doc":"Returns the endpoint configuration for key Returns default if the key does not exist.","ref":"LightningWeb.Endpoint.html#config/2"},{"type":"function","title":"LightningWeb.Endpoint.config_change/2","doc":"Reloads the configuration given the application environment changes.","ref":"LightningWeb.Endpoint.html#config_change/2"},{"type":"function","title":"LightningWeb.Endpoint.host/0","doc":"Returns the host for the given endpoint.","ref":"LightningWeb.Endpoint.html#host/0"},{"type":"function","title":"LightningWeb.Endpoint.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.Endpoint.html#init/1"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast/3 .","ref":"LightningWeb.Endpoint.html#local_broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#local_broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.path/1","doc":"Generates the path information when routing to this endpoint.","ref":"LightningWeb.Endpoint.html#path/1"},{"type":"function","title":"LightningWeb.Endpoint.script_name/0","doc":"Generates the script name.","ref":"LightningWeb.Endpoint.html#script_name/0"},{"type":"function","title":"LightningWeb.Endpoint.start_link/1","doc":"Starts the endpoint supervision tree. All other options are merged into the endpoint configuration.","ref":"LightningWeb.Endpoint.html#start_link/1"},{"type":"function","title":"LightningWeb.Endpoint.static_integrity/1","doc":"Generates a base64-encoded cryptographic hash (sha512) to a static file in priv/static . Meant to be used for Subresource Integrity with CDNs.","ref":"LightningWeb.Endpoint.html#static_integrity/1"},{"type":"function","title":"LightningWeb.Endpoint.static_lookup/1","doc":"Returns a two item tuple with the first item being the static_path and the second item being the static_integrity .","ref":"LightningWeb.Endpoint.html#static_lookup/1"},{"type":"function","title":"LightningWeb.Endpoint.static_path/1","doc":"Generates a route to a static file in priv/static .","ref":"LightningWeb.Endpoint.html#static_path/1"},{"type":"function","title":"LightningWeb.Endpoint.static_url/0","doc":"Generates the static URL without any path information. It uses the configuration under :static_url to generate such. It falls back to :url if :static_url is not set.","ref":"LightningWeb.Endpoint.html#static_url/0"},{"type":"function","title":"LightningWeb.Endpoint.struct_url/0","doc":"Generates the endpoint base URL but as a URI struct. It uses the configuration under :url to generate such. Useful for manipulating the URL data and passing it to URL helpers.","ref":"LightningWeb.Endpoint.html#struct_url/0"},{"type":"function","title":"LightningWeb.Endpoint.subscribe/2","doc":"Callback implementation for Phoenix.Endpoint.subscribe/2 .","ref":"LightningWeb.Endpoint.html#subscribe/2"},{"type":"function","title":"LightningWeb.Endpoint.unsubscribe/1","doc":"Callback implementation for Phoenix.Endpoint.unsubscribe/1 .","ref":"LightningWeb.Endpoint.html#unsubscribe/1"},{"type":"function","title":"LightningWeb.Endpoint.url/0","doc":"Generates the endpoint base URL without any path information. It uses the configuration under :url to generate such.","ref":"LightningWeb.Endpoint.html#url/0"},{"type":"module","title":"LightningWeb.ErrorView","doc":"","ref":"LightningWeb.ErrorView.html"},{"type":"function","title":"LightningWeb.ErrorView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ErrorView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ErrorView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ErrorView.html#render/2"},{"type":"function","title":"LightningWeb.ErrorView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ErrorView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.FallbackController","doc":"Translates controller action results into valid Plug.Conn responses. See Phoenix.Controller.action_fallback/1 for more details.","ref":"LightningWeb.FallbackController.html"},{"type":"module","title":"LightningWeb.FirstSetupLive.Superuser","doc":"Superuser setup liveview Allows the creation on the first user in the system. It has only one action: :show","ref":"LightningWeb.FirstSetupLive.Superuser.html"},{"type":"function","title":"LightningWeb.FirstSetupLive.Superuser.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.FirstSetupLive.Superuser.html#render/1"},{"type":"module","title":"LightningWeb.FormHelpers","doc":"Conveniences for building forms.","ref":"LightningWeb.FormHelpers.html"},{"type":"module","title":"LightningWeb.Gettext","doc":"A module providing Internationalization with a gettext-based API. By using Gettext , your module gains a set of macros for translations, for example: import LightningWeb.Gettext # Simple translation gettext ( "Here is the string to translate" ) # Plural translation ngettext ( "Here is the string to translate" , "Here are the strings to translate" , 3 ) # Domain-based translation dgettext ( "errors" , "Here is the error message to translate" ) See the Gettext Docs for detailed usage.","ref":"LightningWeb.Gettext.html"},{"type":"macro","title":"LightningWeb.Gettext.dgettext/3","doc":"Callback implementation for Gettext.Backend.dgettext/3 .","ref":"LightningWeb.Gettext.html#dgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.dgettext_noop/2","doc":"Callback implementation for Gettext.Backend.dgettext_noop/2 .","ref":"LightningWeb.Gettext.html#dgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.dngettext/5","doc":"Callback implementation for Gettext.Backend.dngettext/5 .","ref":"LightningWeb.Gettext.html#dngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.dngettext_noop/3","doc":"Callback implementation for Gettext.Backend.dngettext_noop/3 .","ref":"LightningWeb.Gettext.html#dngettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext/4","doc":"Callback implementation for Gettext.Backend.dpgettext/4 .","ref":"LightningWeb.Gettext.html#dpgettext/4"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#dpgettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext/6","doc":"Callback implementation for Gettext.Backend.dpngettext/6 .","ref":"LightningWeb.Gettext.html#dpngettext/6"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext_noop/4","doc":"","ref":"LightningWeb.Gettext.html#dpngettext_noop/4"},{"type":"macro","title":"LightningWeb.Gettext.gettext/2","doc":"Callback implementation for Gettext.Backend.gettext/2 .","ref":"LightningWeb.Gettext.html#gettext/2"},{"type":"macro","title":"LightningWeb.Gettext.gettext_comment/1","doc":"Callback implementation for Gettext.Backend.gettext_comment/1 .","ref":"LightningWeb.Gettext.html#gettext_comment/1"},{"type":"macro","title":"LightningWeb.Gettext.gettext_noop/1","doc":"Callback implementation for Gettext.Backend.gettext_noop/1 .","ref":"LightningWeb.Gettext.html#gettext_noop/1"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_bindings/2","doc":"Callback implementation for Gettext.Backend.handle_missing_bindings/2 .","ref":"LightningWeb.Gettext.html#handle_missing_bindings/2"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_plural_translation/7","doc":"Callback implementation for Gettext.Backend.handle_missing_plural_translation/7 .","ref":"LightningWeb.Gettext.html#handle_missing_plural_translation/7"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_translation/5","doc":"Callback implementation for Gettext.Backend.handle_missing_translation/5 .","ref":"LightningWeb.Gettext.html#handle_missing_translation/5"},{"type":"function","title":"LightningWeb.Gettext.lgettext/5","doc":"","ref":"LightningWeb.Gettext.html#lgettext/5"},{"type":"function","title":"LightningWeb.Gettext.lngettext/7","doc":"","ref":"LightningWeb.Gettext.html#lngettext/7"},{"type":"macro","title":"LightningWeb.Gettext.ngettext/4","doc":"Callback implementation for Gettext.Backend.ngettext/4 .","ref":"LightningWeb.Gettext.html#ngettext/4"},{"type":"macro","title":"LightningWeb.Gettext.ngettext_noop/2","doc":"Callback implementation for Gettext.Backend.ngettext_noop/2 .","ref":"LightningWeb.Gettext.html#ngettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pgettext/3","doc":"Callback implementation for Gettext.Backend.pgettext/3 .","ref":"LightningWeb.Gettext.html#pgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.pgettext_noop/2","doc":"","ref":"LightningWeb.Gettext.html#pgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pngettext/5","doc":"Callback implementation for Gettext.Backend.pngettext/5 .","ref":"LightningWeb.Gettext.html#pngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.pngettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#pngettext_noop/3"},{"type":"module","title":"LightningWeb.HealthCheck","doc":"","ref":"LightningWeb.HealthCheck.html"},{"type":"function","title":"LightningWeb.HealthCheck.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.HealthCheck.html#call/2"},{"type":"function","title":"LightningWeb.HealthCheck.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.HealthCheck.html#init/1"},{"type":"module","title":"LightningWeb.Hooks","doc":"LiveView Hooks","ref":"LightningWeb.Hooks.html"},{"type":"function","title":"LightningWeb.Hooks.on_mount/4","doc":"Finds and assigns a project to the socket, if a user doesn't have access they are redirected and shown a 'No Access' screen via a :nav flash message. There is a fallthru function, when there is no project_id in the params - this is for liveviews that may or may not have a project_id depending on usage - like DashboardLive .","ref":"LightningWeb.Hooks.html#on_mount/4"},{"type":"module","title":"LightningWeb.InitAssigns","doc":"Ensures common assigns are applied to all LiveViews attaching this hook.","ref":"LightningWeb.InitAssigns.html"},{"type":"function","title":"LightningWeb.InitAssigns.on_mount/4","doc":"","ref":"LightningWeb.InitAssigns.html#on_mount/4"},{"type":"module","title":"LightningWeb.JobLive.AdaptorPicker","doc":"Component allowing selecting an adaptor and it's version","ref":"LightningWeb.JobLive.AdaptorPicker.html"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.display_name_for_adaptor/1","doc":"Converts standard adaptor names into "label","value" lists and returns non-standard names as merely "value"; both can be passed directly into a select option list.","ref":"LightningWeb.JobLive.AdaptorPicker.html#display_name_for_adaptor/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.get_adaptor_version_options/1","doc":"","ref":"LightningWeb.JobLive.AdaptorPicker.html#get_adaptor_version_options/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.render/1","doc":"Attributes form ( :map ) (required) on_change ( :any ) - Defaults to nil . disabled ( :boolean ) - Defaults to false .","ref":"LightningWeb.JobLive.AdaptorPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CredentialPicker","doc":"Component allowing selecting a credential or creating a new one via a modal.","ref":"LightningWeb.JobLive.CredentialPicker.html"},{"type":"function","title":"LightningWeb.JobLive.CredentialPicker.render/1","doc":"Attributes form ( :map ) (required) disabled ( :boolean ) - Defaults to false . credentials ( :list ) (required) on_change ( :any ) - Defaults to nil .","ref":"LightningWeb.JobLive.CredentialPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CronSetupComponent","doc":"A live component for managing cron setup in a form. The CronSetupComponent provides an interactive form for configuring cron settings. It includes fields for specifying the frequency, minute, hour, weekday, monthday, and cron expression. Usage Include the CronSetupComponent in your live view or template. Pass the necessary assigns to the component, such as form , on_change , and disabled . Handle the cron_expression_change event to capture changes in the form inputs. Use the updated cron_expression in your application logic.","ref":"LightningWeb.JobLive.CronSetupComponent.html"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.build_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#build_cron_expression/2"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.frequency_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#frequency_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.get_cron_data/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#get_cron_data/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.hour_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#hour_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.minute_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#minute_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.monthday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#monthday_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.render/1","doc":"Attributes id ( :string ) (required) form ( :map ) (required) on_change ( :any ) (required) disabled ( :boolean ) (required)","ref":"LightningWeb.JobLive.CronSetupComponent.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.time_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#time_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.weekday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#weekday_field/1"},{"type":"module","title":"LightningWeb.JobLive.JobBuilder","doc":"Job Builder Panel","ref":"LightningWeb.JobLive.JobBuilder.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.delete_title/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#delete_title/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.follow_run/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#follow_run/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.render/1","doc":"Attributes return_to ( :string ) (required) params ( :map ) - Defaults to %{} . can_edit_job ( :boolean ) (required) can_run_job ( :boolean ) (required)","ref":"LightningWeb.JobLive.JobBuilder.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_adaptor/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_adaptor/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_credential/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_credential/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.update_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#update_cron_expression/2"},{"type":"module","title":"LightningWeb.JobLive.JobBuilderComponents","doc":"","ref":"LightningWeb.JobLive.JobBuilderComponents.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.job_editor_component/1","doc":"Attributes adaptor ( :string ) (required) disabled ( :boolean ) - Defaults to false . source ( :string ) (required) change_event ( :string ) - Defaults to "job_body_changed" . Global attributes are accepted.","ref":"LightningWeb.JobLive.JobBuilderComponents.html#job_editor_component/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.trigger_picker/1","doc":"Attributes form ( :map ) (required) upstream_jobs ( :list ) (required) on_cron_change ( :any ) (required) disabled ( :boolean ) - Defaults to true .","ref":"LightningWeb.JobLive.JobBuilderComponents.html#trigger_picker/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.when_invalid/1","doc":"Attributes changeset ( :map ) (required) field ( :atom ) (required) Slots inner_block (required)","ref":"LightningWeb.JobLive.JobBuilderComponents.html#when_invalid/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.render/1","doc":"Attributes job ( :map ) (required) on_run ( :any ) (required) user ( :map ) (required) selected_dataclip_id ( :string ) (required)","ref":"LightningWeb.JobLive.ManualRunComponent.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.changeset/2","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html#changeset/2"},{"type":"module","title":"LightningWeb.LiveHelpers","doc":"General purpose LiveView helper functions","ref":"LightningWeb.LiveHelpers.html"},{"type":"function","title":"LightningWeb.LiveHelpers.display_short_uuid/1","doc":"","ref":"LightningWeb.LiveHelpers.html#display_short_uuid/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_in/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_in/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_out/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_out/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_error_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_error_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_info_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_info_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_nav_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_nav_block/1"},{"type":"module","title":"LightningWeb.ModalPortal","doc":"Component for rendering content inside layout without full DOM patch.","ref":"LightningWeb.ModalPortal.html"},{"type":"function","title":"LightningWeb.ModalPortal.close_modal/0","doc":"","ref":"LightningWeb.ModalPortal.html#close_modal/0"},{"type":"function","title":"LightningWeb.ModalPortal.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.ModalPortal.html#handle_event/3"},{"type":"function","title":"LightningWeb.ModalPortal.on_hide/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_hide/2"},{"type":"function","title":"LightningWeb.ModalPortal.on_show/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_show/2"},{"type":"function","title":"LightningWeb.ModalPortal.open_modal/2","doc":"","ref":"LightningWeb.ModalPortal.html#open_modal/2"},{"type":"function","title":"LightningWeb.ModalPortal.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ModalPortal.html#render/1"},{"type":"function","title":"LightningWeb.ModalPortal.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.ModalPortal.html#update/2"},{"type":"module","title":"LightningWeb.OauthCredentialHelper","doc":"A set of helper functions to encodes state and coordinate OAuth callbacks back to a LiveView component.","ref":"LightningWeb.OauthCredentialHelper.html"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast/2","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#broadcast/2"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast_forward/3","doc":"Broadcast a message specifically for forwarding a message to a component. It expects a subscription_id , the module of the component and opts being a keyword list containing an :id key of the specific component. See: Phoenix.LiveView.send_update/3 for more info. A corresponding LiveView (that is subscribed) is expected to have a matching handle_info/2 that looks like this: def handle_info ( { :forward , mod , opts } , socket ) do send_update ( mod , opts ) { :noreply , socket } end","ref":"LightningWeb.OauthCredentialHelper.html#broadcast_forward/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.build_state/3","doc":"Encode and encrypt the callback data which will be sent so a provider as the state key in the request. The values are: subscription_id The same ID used to subscribe. The component module The LiveView component that is going to receive update The component id The ID of the component","ref":"LightningWeb.OauthCredentialHelper.html#build_state/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.decode_state/1","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#decode_state/1"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.subscribe/1","doc":"Subscribe to the oauth_credential topic. It expects the a unique ID for the topic, usually the LiveView's socket.id .","ref":"LightningWeb.OauthCredentialHelper.html#subscribe/1"},{"type":"module","title":"LightningWeb.OidcController","doc":"","ref":"LightningWeb.OidcController.html"},{"type":"function","title":"LightningWeb.OidcController.new/2","doc":"Once the user has completed the authorization flow from above, they are returned here, and the authorization code is used to log them in.","ref":"LightningWeb.OidcController.html#new/2"},{"type":"function","title":"LightningWeb.OidcController.show/2","doc":"Given a known provider, redirect them to the authorize url on the provider","ref":"LightningWeb.OidcController.html#show/2"},{"type":"module","title":"LightningWeb.PageView","doc":"","ref":"LightningWeb.PageView.html"},{"type":"function","title":"LightningWeb.PageView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.PageView.html#__resource__/0"},{"type":"function","title":"LightningWeb.PageView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.PageView.html#render/2"},{"type":"function","title":"LightningWeb.PageView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.PageView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.Pagination","doc":"Pagination Components This has been extracted and adapted from scrivener_html . See: https://github.com/mgwidmann/scrivener_html","ref":"LightningWeb.Pagination.html"},{"type":"function","title":"LightningWeb.Pagination.page_link/1","doc":"","ref":"LightningWeb.Pagination.html#page_link/1"},{"type":"function","title":"LightningWeb.Pagination.pagination_bar/1","doc":"","ref":"LightningWeb.Pagination.html#pagination_bar/1"},{"type":"function","title":"LightningWeb.Pagination.raw_pagination_links/2","doc":"Returns the raw data in order to generate the proper HTML for pagination links. Data is returned in a {text, page_number} format where text is intended to be the text of the link and page_number is the page it should go to. Defaults are already supplied and they are as follows: [ distance : 5 , next : :next , previous : :previous , first : true , last : true , ellipsis : :ellipsis ] distance must be a positive non-zero integer or an exception is raised. next and previous should be strings but can be anything you want as long as it is truthy, falsey values will remove them from the output. first and last are only booleans, and they just include/remove their respective link from output. An example of the data returned: iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 10 , page_number : 5 } ) [ { "<<" , 4 } , { 1 , 1 } , { 2 , 2 } , { 3 , 3 } , { 4 , 4 } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { ">>" , 6 } ] iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 20 , page_number : 10 } , first : [ "←" ] , last : [ "→" ] ) [ { "<<" , 9 } , { [ "←" ] , 1 } , { :ellipsis , { :safe , "&hellip;" } } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { 11 , 11 } , { 12 , 12 } , { 13 , 13 } , { 14 , 14 } , { 15 , 15 } , { :ellipsis , { :safe , "&hellip;" } } , { [ "→" ] , 20 } , { ">>" , 11 } ] Simply loop and pattern match over each item and transform it to your custom HTML.","ref":"LightningWeb.Pagination.html#raw_pagination_links/2"},{"type":"module","title":"LightningWeb.Plugs.FirstSetup","doc":"Plug to redirect HTTP requests to /first_setup if there are no superusers in the system yet.","ref":"LightningWeb.Plugs.FirstSetup.html"},{"type":"module","title":"LightningWeb.ProfileLive.Edit","doc":"LiveView for user profile page.","ref":"LightningWeb.ProfileLive.Edit.html"},{"type":"function","title":"LightningWeb.ProfileLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProfileLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.FormComponent","doc":"Form component update profile email and password","ref":"LightningWeb.ProfileLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.MfaComponent","doc":"Component to enable MFA on a User's account","ref":"LightningWeb.ProfileLive.MfaComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.MfaComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.MfaComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.ProjectLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProjectLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProjectLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.ProjectLive.Index.html"},{"type":"function","title":"LightningWeb.ProjectLive.Index.delete_action/1","doc":"","ref":"LightningWeb.ProjectLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.ProjectLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Settings","doc":"Index Liveview for Runs","ref":"LightningWeb.ProjectLive.Settings.html"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.digest/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#digest/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.failure_alert/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#failure_alert/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Settings.html#render/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.role/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#role/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.user/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#user/1"},{"type":"module","title":"LightningWeb.RouteHelpers","doc":"Convenience functions for generating paths.","ref":"LightningWeb.RouteHelpers.html"},{"type":"function","title":"LightningWeb.RouteHelpers.oidc_callback_url/0","doc":"","ref":"LightningWeb.RouteHelpers.html#oidc_callback_url/0"},{"type":"function","title":"LightningWeb.RouteHelpers.show_run_url/2","doc":"","ref":"LightningWeb.RouteHelpers.html#show_run_url/2"},{"type":"module","title":"LightningWeb.Router","doc":"","ref":"LightningWeb.Router.html"},{"type":"function","title":"LightningWeb.Router.api/2","doc":"","ref":"LightningWeb.Router.html#api/2"},{"type":"function","title":"LightningWeb.Router.browser/2","doc":"","ref":"LightningWeb.Router.html#browser/2"},{"type":"function","title":"LightningWeb.Router.call/2","doc":"Callback invoked by Plug on every request.","ref":"LightningWeb.Router.html#call/2"},{"type":"function","title":"LightningWeb.Router.init/1","doc":"Callback required by Plug that initializes the router for serving web requests.","ref":"LightningWeb.Router.html#init/1"},{"type":"function","title":"LightningWeb.Router.storybook_assets/2","doc":"","ref":"LightningWeb.Router.html#storybook_assets/2"},{"type":"function","title":"LightningWeb.Router.storybook_browser/2","doc":"","ref":"LightningWeb.Router.html#storybook_browser/2"},{"type":"module","title":"LightningWeb.RunLive.Index","doc":"Index Liveview for Runs","ref":"LightningWeb.RunLive.Index.html"},{"type":"function","title":"LightningWeb.RunLive.Index.checked/2","doc":"","ref":"LightningWeb.RunLive.Index.html#checked/2"},{"type":"function","title":"LightningWeb.RunLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.RunLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.RunLive.RerunJobComponent","doc":"Rerun job component","ref":"LightningWeb.RunLive.RerunJobComponent.html"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#handle_event/3"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#render/1"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#update/2"},{"type":"module","title":"LightningWeb.RunLive.RunViewerLive","doc":"","ref":"LightningWeb.RunLive.RunViewerLive.html"},{"type":"function","title":"LightningWeb.RunLive.RunViewerLive.handle_info/2","doc":"Reload the run when any update messages arrive.","ref":"LightningWeb.RunLive.RunViewerLive.html#handle_info/2"},{"type":"module","title":"LightningWeb.RunLive.Show","doc":"Show page for individual runs.","ref":"LightningWeb.RunLive.Show.html"},{"type":"function","title":"LightningWeb.RunLive.Show.apply_action/3","doc":"","ref":"LightningWeb.RunLive.Show.html#apply_action/3"},{"type":"module","title":"LightningWeb.RunLive.WorkOrderComponent","doc":"Workorder component","ref":"LightningWeb.RunLive.WorkOrderComponent.html"},{"type":"function","title":"LightningWeb.RunLive.WorkOrderComponent.render/1","doc":"Attributes show_details ( :boolean ) - Defaults to false . entry_selected ( :boolean ) - Defaults to false .","ref":"LightningWeb.RunLive.WorkOrderComponent.html#render/1"},{"type":"module","title":"LightningWeb.Telemetry","doc":"Assorted metrics to collect during runtime. See https://hexdocs.pm/phoenix/telemetry.html","ref":"LightningWeb.Telemetry.html"},{"type":"function","title":"LightningWeb.Telemetry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"LightningWeb.Telemetry.html#child_spec/1"},{"type":"function","title":"LightningWeb.Telemetry.metrics/0","doc":"","ref":"LightningWeb.Telemetry.html#metrics/0"},{"type":"function","title":"LightningWeb.Telemetry.start_link/1","doc":"","ref":"LightningWeb.Telemetry.html#start_link/1"},{"type":"module","title":"LightningWeb.TokensLive.Index","doc":"LiveView for listing and managing tokens","ref":"LightningWeb.TokensLive.Index.html"},{"type":"function","title":"LightningWeb.TokensLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.TokensLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserAuth","doc":"The UserAuth controller.","ref":"LightningWeb.UserAuth.html"},{"type":"function","title":"LightningWeb.UserAuth.authenticate_bearer/2","doc":"","ref":"LightningWeb.UserAuth.html#authenticate_bearer/2"},{"type":"function","title":"LightningWeb.UserAuth.fetch_current_user/2","doc":"Authenticates the user by looking into the session and remember me token.","ref":"LightningWeb.UserAuth.html#fetch_current_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_in_user/2","doc":"Logs the user in by creating a new session token.","ref":"LightningWeb.UserAuth.html#log_in_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_out_user/1","doc":"Logs the user out. It clears all session data for safety. See renew_session.","ref":"LightningWeb.UserAuth.html#log_out_user/1"},{"type":"function","title":"LightningWeb.UserAuth.mark_totp_pending/1","doc":"","ref":"LightningWeb.UserAuth.html#mark_totp_pending/1"},{"type":"function","title":"LightningWeb.UserAuth.new_session/2","doc":"Assigns the token to a new session. It renews the session ID and clears the whole session to avoid fixation attacks. See the renew_session function to customize this behaviour. It also sets a :live_socket_id key in the session, so LiveView sessions are identified and automatically disconnected on log out. The line can be safely removed if you are not using LiveView.","ref":"LightningWeb.UserAuth.html#new_session/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_if_user_is_authenticated/2","doc":"Used for routes that require the user to not be authenticated.","ref":"LightningWeb.UserAuth.html#redirect_if_user_is_authenticated/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_with_return_to/2","doc":"Returns to or redirects to the dashboard and potentially set remember_me token.","ref":"LightningWeb.UserAuth.html#redirect_with_return_to/2"},{"type":"function","title":"LightningWeb.UserAuth.require_authenticated_user/2","doc":"Used for routes that require the user to be authenticated. If you want to enforce the user email is confirmed before they use the application at all, here would be a good place.","ref":"LightningWeb.UserAuth.html#require_authenticated_user/2"},{"type":"function","title":"LightningWeb.UserAuth.totp_pending?/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_pending?/1"},{"type":"function","title":"LightningWeb.UserAuth.totp_validated/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_validated/1"},{"type":"module","title":"LightningWeb.UserConfirmationController","doc":"","ref":"LightningWeb.UserConfirmationController.html"},{"type":"function","title":"LightningWeb.UserConfirmationController.confirm_email/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#confirm_email/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.create/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#create/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.edit/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#edit/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.new/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#new/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.update/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#update/2"},{"type":"module","title":"LightningWeb.UserLive.Edit","doc":"LiveView for editing a single job, which inturn uses LightningWeb.JobLive.BigFormComponent for common functionality.","ref":"LightningWeb.UserLive.Edit.html"},{"type":"function","title":"LightningWeb.UserLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.FormComponent","doc":"Form component for creating and editing users","ref":"LightningWeb.UserLive.FormComponent.html"},{"type":"function","title":"LightningWeb.UserLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.UserLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.Index","doc":"Index page for listing users","ref":"LightningWeb.UserLive.Index.html"},{"type":"function","title":"LightningWeb.UserLive.Index.delete_action/1","doc":"","ref":"LightningWeb.UserLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.UserLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserRegistrationController","doc":"","ref":"LightningWeb.UserRegistrationController.html"},{"type":"function","title":"LightningWeb.UserRegistrationController.create/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#create/2"},{"type":"function","title":"LightningWeb.UserRegistrationController.new/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#new/2"},{"type":"module","title":"LightningWeb.UserResetPasswordController","doc":"","ref":"LightningWeb.UserResetPasswordController.html"},{"type":"function","title":"LightningWeb.UserResetPasswordController.create/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#create/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.edit/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#edit/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.new/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#new/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.update/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#update/2"},{"type":"module","title":"LightningWeb.UserSessionController","doc":"","ref":"LightningWeb.UserSessionController.html"},{"type":"function","title":"LightningWeb.UserSessionController.auth_handler_url/0","doc":"","ref":"LightningWeb.UserSessionController.html#auth_handler_url/0"},{"type":"function","title":"LightningWeb.UserSessionController.create/2","doc":"","ref":"LightningWeb.UserSessionController.html#create/2"},{"type":"function","title":"LightningWeb.UserSessionController.delete/2","doc":"","ref":"LightningWeb.UserSessionController.html#delete/2"},{"type":"function","title":"LightningWeb.UserSessionController.exchange_token/2","doc":"","ref":"LightningWeb.UserSessionController.html#exchange_token/2"},{"type":"function","title":"LightningWeb.UserSessionController.new/2","doc":"","ref":"LightningWeb.UserSessionController.html#new/2"},{"type":"module","title":"LightningWeb.UserTOTPController","doc":"","ref":"LightningWeb.UserTOTPController.html"},{"type":"function","title":"LightningWeb.UserTOTPController.create/2","doc":"","ref":"LightningWeb.UserTOTPController.html#create/2"},{"type":"function","title":"LightningWeb.UserTOTPController.new/2","doc":"","ref":"LightningWeb.UserTOTPController.html#new/2"},{"type":"module","title":"LightningWeb.WebhooksController","doc":"","ref":"LightningWeb.WebhooksController.html"},{"type":"function","title":"LightningWeb.WebhooksController.create/2","doc":"","ref":"LightningWeb.WebhooksController.html#create/2"},{"type":"module","title":"LightningWeb.WorkflowLive.EditorPane","doc":"","ref":"LightningWeb.WorkflowLive.EditorPane.html"},{"type":"function","title":"LightningWeb.WorkflowLive.EditorPane.render/1","doc":"Attributes id ( :string ) (required) disabled ( :boolean ) - Defaults to false . class ( :string ) - Defaults to "" . on_change ( :any ) (required) adaptor ( :string ) (required) source ( :string ) (required) job_id ( :string ) (required)","ref":"LightningWeb.WorkflowLive.EditorPane.html#render/1"},{"type":"module","title":"LightningWeb.WorkflowLive.JobView","doc":"","ref":"LightningWeb.WorkflowLive.JobView.html"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.container/1","doc":"Attributes id ( :string ) (required) Slots top column - Accepts attributes: class ( :string ) - Extra CSS classes for the column. bottom","ref":"LightningWeb.WorkflowLive.JobView.html#container/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.input_pane/1","doc":"Attributes job ( :map ) (required) user ( :map ) (required) project ( :map ) (required) on_run ( :any ) (required) - Callback to run a job manually. can_run_job ( :boolean ) - Defaults to true .","ref":"LightningWeb.WorkflowLive.JobView.html#input_pane/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.job_edit_view/1","doc":"Attributes job ( :map ) (required) form ( :map ) (required) - A form built from a job. current_user ( :map ) (required) project ( :map ) (required) close_url ( :any ) (required) socket ( :any ) (required) on_run ( :any ) (required) - Callback to run a job manually. follow_run_id ( :any ) - Defaults to nil . Slots footer","ref":"LightningWeb.WorkflowLive.JobView.html#job_edit_view/1"},{"type":"module","title":"LightningWeb.WorkflowNewLive.WorkflowParams","doc":"Various function for reconciling changes to a workflow params map. The front end editor uses JSON patches to represent changes to the workflow.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_form_params/2","doc":"Produce a new set of params by applying the given form params to the current params.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_form_params/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_patches/2","doc":"Produce a new set of params by applying the given patches to the current parms","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_patches/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_map/1","doc":"Convert a changeset to a serializable map of workflow params, suitable for sending to the front end editor. It uses Lightning.Helpers.json_safe/1 to ensure that the map is safe to serialize to JSON. This is necessary because the underlying model may contain atom values.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_map/1"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_patches/2","doc":"Produce a set of patches that represent the difference between the initial params and the target params. This usually is used to produce a set of patches that represent the changes introduced by a changeset.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_patches/2"},{"type":"task","title":"Mix.Tasks.Lightning.GenEncryptionKey","doc":"Helper to generate a unique encryption key for Vault","ref":"Mix.Tasks.Lightning.GenEncryptionKey.html"},{"type":"task","title":"Mix.Tasks.Lightning.InstallRuntime","doc":"Installs the following NodeJS packages: core language-common","ref":"Mix.Tasks.Lightning.InstallRuntime.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.packages/0","doc":"","ref":"Mix.Tasks.Lightning.InstallRuntime.html#packages/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallRuntime.html#run/1"},{"type":"task","title":"Mix.Tasks.Lightning.InstallSchemas","doc":"Install the credential json schemas Use --exclude language-package1, language-package2 to exclude specific packages","ref":"Mix.Tasks.Lightning.InstallSchemas.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.fetch_schemas/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#fetch_schemas/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.parse_excluded/1","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#parse_excluded/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.persist_schema/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#persist_schema/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_options/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_params/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_chunk/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request!/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#run/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#start/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#stream_next/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.write_schema/3","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#write_schema/3"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.body/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:body/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.headers/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:headers/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.method/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:method/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.options/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:options/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.params/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:params/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.request/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:request/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.url/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:url/0"},{"type":"module","title":"ObanPruner","doc":"The Oban Pruner removes completed Oban jobs. It leaves everything else for manual inspection.","ref":"ObanPruner.html"},{"type":"function","title":"ObanPruner.perform/1","doc":"Deletes completed Oban jobs, leaving discarded for manual inspection.","ref":"ObanPruner.html#perform/1"},{"type":"module","title":"Storybook.Root","doc":"","ref":"Storybook.Root.html"},{"type":"module","title":"Lightning.Accounts","doc":"The Accounts context.","ref":"Lightning.Accounts.html"},{"type":"function","title":"Lightning.Accounts.apply_user_email/3","doc":"Emulates that the email will change without actually changing it in the database. Examples iex> apply_user_email ( user , "valid password" , %{ email : ... } ) { :ok , % User { } } role : :superuser iex> apply_user_email ( user , "invalid password" , %{ email : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#apply_user_email/3"},{"type":"function","title":"Lightning.Accounts.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Accounts.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Accounts.change_scheduled_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the user scheduled_deletion. Examples iex> change_scheduled_deletion ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_scheduled_deletion/2"},{"type":"function","title":"Lightning.Accounts.change_superuser_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking superuser changes. Examples iex> change_superuser_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_superuser_registration/1"},{"type":"function","title":"Lightning.Accounts.change_user_details/2","doc":"","ref":"Lightning.Accounts.html#change_user_details/2"},{"type":"function","title":"Lightning.Accounts.change_user_email/2","doc":"Returns an %Ecto.Changeset{} for changing the user email. Examples iex> change_user_email ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_email/2"},{"type":"function","title":"Lightning.Accounts.change_user_password/2","doc":"Returns an %Ecto.Changeset{} for changing the user password. Examples iex> change_user_password ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_password/2"},{"type":"function","title":"Lightning.Accounts.change_user_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking user changes. Examples iex> change_user_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_registration/1"},{"type":"function","title":"Lightning.Accounts.confirm_user/1","doc":"Confirms a user by the given token. If the token matches, the user account is marked as confirmed and the token is deleted.","ref":"Lightning.Accounts.html#confirm_user/1"},{"type":"function","title":"Lightning.Accounts.delete_auth_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_auth_token/1"},{"type":"function","title":"Lightning.Accounts.delete_session_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_session_token/1"},{"type":"function","title":"Lightning.Accounts.delete_token/1","doc":"Deletes a token. Examples iex> delete_token ( token ) { :ok , % UserToken { } } iex> delete_token ( token ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_token/1"},{"type":"function","title":"Lightning.Accounts.delete_user/1","doc":"Deletes a user. Examples iex> delete_user ( user ) { :ok , % User { } } iex> delete_user ( user ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_user/1"},{"type":"function","title":"Lightning.Accounts.delete_user_totp/1","doc":"Deletes the given user's TOTP","ref":"Lightning.Accounts.html#delete_user_totp/1"},{"type":"function","title":"Lightning.Accounts.deliver_update_email_instructions/3","doc":"Delivers the update email instructions to the given user. Examples iex> deliver_update_email_instructions ( user , current_email , & Routes . user_update_email_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_update_email_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/2","doc":"Delivers the confirmation email instructions to the given user. Examples iex> deliver_user_confirmation_instructions ( user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } } iex> deliver_user_confirmation_instructions ( confirmed_user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :error , :already_confirmed }","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/3","doc":"","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_reset_password_instructions/2","doc":"Delivers the reset password email to the given user. Examples iex> deliver_user_reset_password_instructions ( user , & Routes . user_reset_password_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_user_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.exchange_auth_token/1","doc":"Exchanges an auth token for a session token. The auth token is removed from the database if successful.","ref":"Lightning.Accounts.html#exchange_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_api_token/1","doc":"Generates an API token for a user.","ref":"Lightning.Accounts.html#generate_api_token/1"},{"type":"function","title":"Lightning.Accounts.generate_auth_token/1","doc":"Generates an auth token.","ref":"Lightning.Accounts.html#generate_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_user_session_token/1","doc":"Generates a session token.","ref":"Lightning.Accounts.html#generate_user_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_token!/1","doc":"Gets a single token. Raises Ecto.NoResultsError if the UserToken does not exist. Examples iex> get_token! ( 123 ) % UserToken { } iex> get_token! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_token!/1"},{"type":"function","title":"Lightning.Accounts.get_user!/1","doc":"Gets a single user. Raises Ecto.NoResultsError if the User does not exist. Examples iex> get_user! ( 123 ) % User { } iex> get_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_user!/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_api_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_api_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_auth_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_auth_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email/1","doc":"Gets a user by email. Examples iex> get_user_by_email ( "foo@example.com" ) % User { } iex> get_user_by_email ( "unknown@example.com" ) nil","ref":"Lightning.Accounts.html#get_user_by_email/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email_and_password/2","doc":"Gets a user by email and password. Examples iex> get_user_by_email_and_password ( "foo@example.com" , "correct_password" ) % User { } iex> get_user_by_email_and_password ( "foo@example.com" , "invalid_password" ) nil","ref":"Lightning.Accounts.html#get_user_by_email_and_password/2"},{"type":"function","title":"Lightning.Accounts.get_user_by_reset_password_token/1","doc":"Gets the user by reset password token. Examples iex> get_user_by_reset_password_token ( "validtoken" ) % User { } iex> get_user_by_reset_password_token ( "invalidtoken" ) nil","ref":"Lightning.Accounts.html#get_user_by_reset_password_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_session_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_totp/1","doc":"Gets a single UserTOTP if any exists.","ref":"Lightning.Accounts.html#get_user_totp/1"},{"type":"function","title":"Lightning.Accounts.get_users_to_alert_for_project/1","doc":"Gets all users to alert of workflow failure for a project","ref":"Lightning.Accounts.html#get_users_to_alert_for_project/1"},{"type":"function","title":"Lightning.Accounts.has_activity_in_projects?/1","doc":"","ref":"Lightning.Accounts.html#has_activity_in_projects?/1"},{"type":"function","title":"Lightning.Accounts.has_one_superuser?/0","doc":"Used to determine if there is at least one Superuser in the system. This triggers the setup page on fresh installs.","ref":"Lightning.Accounts.html#has_one_superuser?/0"},{"type":"function","title":"Lightning.Accounts.list_api_tokens/1","doc":"Lists all user tokens","ref":"Lightning.Accounts.html#list_api_tokens/1"},{"type":"function","title":"Lightning.Accounts.list_users/0","doc":"Returns the list of users. Examples iex> list_users ( ) [ % User { } , ... ]","ref":"Lightning.Accounts.html#list_users/0"},{"type":"function","title":"Lightning.Accounts.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find users that are ready for permanent deletion and purge them.","ref":"Lightning.Accounts.html#perform/1"},{"type":"function","title":"Lightning.Accounts.purge_user/1","doc":"","ref":"Lightning.Accounts.html#purge_user/1"},{"type":"function","title":"Lightning.Accounts.register_superuser/1","doc":"Registers a superuser. Examples iex> register_superuser ( %{ field : value } ) { :ok , % User { } } iex> register_superuser ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_superuser/1"},{"type":"function","title":"Lightning.Accounts.register_user/1","doc":"Registers a user. Examples iex> register_user ( %{ field : value } ) { :ok , % User { } } iex> register_user ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_user/1"},{"type":"function","title":"Lightning.Accounts.reset_user_password/2","doc":"Resets the user password. Examples iex> reset_user_password ( user , %{ password : "new long password" , password_confirmation : "new long password" } ) { :ok , % User { } } iex> reset_user_password ( user , %{ password : "valid" , password_confirmation : "not the same" } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#reset_user_password/2"},{"type":"function","title":"Lightning.Accounts.schedule_user_deletion/2","doc":"Given a user and a confirmation email, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic user purge cronjob will never run. (Note that subsequent logins will be blocked for users pending deletion.)","ref":"Lightning.Accounts.html#schedule_user_deletion/2"},{"type":"function","title":"Lightning.Accounts.update_user_details/2","doc":"","ref":"Lightning.Accounts.html#update_user_details/2"},{"type":"function","title":"Lightning.Accounts.update_user_email/2","doc":"Updates the user email using the given token. If the token matches, the user email is updated and the token is deleted. The confirmed_at date is also updated to the current time.","ref":"Lightning.Accounts.html#update_user_email/2"},{"type":"function","title":"Lightning.Accounts.update_user_password/3","doc":"Updates the user password. Examples iex> update_user_password ( user , "valid password" , %{ password : ... } ) { :ok , % User { } } iex> update_user_password ( user , "invalid password" , %{ password : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#update_user_password/3"},{"type":"function","title":"Lightning.Accounts.upsert_user_totp/2","doc":"Updates or Inserts the user's TOTP","ref":"Lightning.Accounts.html#upsert_user_totp/2"},{"type":"function","title":"Lightning.Accounts.valid_user_totp?/2","doc":"Validates if the given TOTP code is valid.","ref":"Lightning.Accounts.html#valid_user_totp?/2"},{"type":"function","title":"Lightning.Accounts.validate_change_user_email/2","doc":"","ref":"Lightning.Accounts.html#validate_change_user_email/2"},{"type":"module","title":"Lightning.Accounts.User","doc":"The User model.","ref":"Lightning.Accounts.User.html"},{"type":"function","title":"Lightning.Accounts.User.confirm_changeset/1","doc":"Confirms the account by setting confirmed_at .","ref":"Lightning.Accounts.User.html#confirm_changeset/1"},{"type":"function","title":"Lightning.Accounts.User.details_changeset/2","doc":"A user changeset for user details: email first_name last_name role","ref":"Lightning.Accounts.User.html#details_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.email_changeset/2","doc":"A user changeset for changing the email. It requires the email to change otherwise an error is added.","ref":"Lightning.Accounts.User.html#email_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.password_changeset/3","doc":"A user changeset for changing the password. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#password_changeset/3"},{"type":"function","title":"Lightning.Accounts.User.scheduled_deletion_changeset/2","doc":"A user changeset for changing the scheduled_deletion property.","ref":"Lightning.Accounts.User.html#scheduled_deletion_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.superuser_registration_changeset/2","doc":"A superuser changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#superuser_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.user_registration_changeset/2","doc":"A user changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#user_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.valid_password?/2","doc":"Verifies the password. If there is no user or the user doesn't have a password, we call Bcrypt.no_user_verify/0 to avoid timing attacks.","ref":"Lightning.Accounts.User.html#valid_password?/2"},{"type":"function","title":"Lightning.Accounts.User.validate_current_password/2","doc":"Validates the current password otherwise adds an error to the changeset.","ref":"Lightning.Accounts.User.html#validate_current_password/2"},{"type":"type","title":"Lightning.Accounts.User.t/0","doc":"","ref":"Lightning.Accounts.User.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.User.RolesEnum","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.create_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Accounts.User.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schema/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Accounts.User.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Accounts.User.RolesEnum.t/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserNotifier","doc":"The UserNotifier module.","ref":"Lightning.Accounts.UserNotifier.html"},{"type":"function","title":"Lightning.Accounts.UserNotifier.build_digest_url/3","doc":"","ref":"Lightning.Accounts.UserNotifier.html#build_digest_url/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/2","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/3","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_addition_notification/2","doc":"Deliver email to notify user of his addition of a project.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_addition_notification/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_digest/2","doc":"Deliver a project digest of daily/weekly or monthly activity to a user.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_digest/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_reset_password_instructions/2","doc":"Deliver instructions to reset a user password.","ref":"Lightning.Accounts.UserNotifier.html#deliver_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_instructions/2","doc":"Deliver instructions to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_warning/2","doc":"Deliver warning to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_warning/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.notify_project_deletion/2","doc":"","ref":"Lightning.Accounts.UserNotifier.html#notify_project_deletion/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.send_deletion_notification_email/1","doc":"Deliver an email to notify the user about their account being deleted","ref":"Lightning.Accounts.UserNotifier.html#send_deletion_notification_email/1"},{"type":"module","title":"Lightning.Accounts.UserTOTP","doc":"User Time based OTPs schema","ref":"Lightning.Accounts.UserTOTP.html"},{"type":"function","title":"Lightning.Accounts.UserTOTP.changeset/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserTOTP.valid_totp?/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#valid_totp?/2"},{"type":"type","title":"Lightning.Accounts.UserTOTP.t/0","doc":"","ref":"Lightning.Accounts.UserTOTP.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserToken","doc":"The UserToken model. The reason why we store session tokens in the database, even though Phoenix already provides a session cookie, is because Phoenix' default session cookies are not persisted, they are simply signed and potentially encrypted. This means they are valid indefinitely, unless you change the signing/encryption salt. Therefore, storing them allows individual user sessions to be expired. The token system can also be extended to store additional data, such as the device used for logging in. You could then use this information to display all valid sessions and devices in the UI and allow users to explicitly expire any session they deem invalid.","ref":"Lightning.Accounts.UserToken.html"},{"type":"function","title":"Lightning.Accounts.UserToken.build_email_token/3","doc":"Builds a token and its hash to be delivered to the user's email. The non-hashed token is sent to the user email while the hashed part is stored in the database. The original token cannot be reconstructed, which means anyone with read-only access to the database cannot directly use the token in the application to gain access. Furthermore, if the user changes their email in the system, the tokens sent to the previous email are no longer valid. Users can easily adapt the existing code to provide other types of delivery methods, for example, by phone numbers.","ref":"Lightning.Accounts.UserToken.html#build_email_token/3"},{"type":"function","title":"Lightning.Accounts.UserToken.build_token/2","doc":"Generates a token that will be stored in a signed place, such as session or cookie. As they are signed, those tokens do not need to be hashed.","ref":"Lightning.Accounts.UserToken.html#build_token/2"},{"type":"function","title":"Lightning.Accounts.UserToken.changeset/2","doc":"","ref":"Lightning.Accounts.UserToken.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign/2","doc":"Combines generate_claims/1 and encode_and_sign/2","ref":"Lightning.Accounts.UserToken.html#generate_and_sign/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign!/2","doc":"Same as generate_and_sign/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#generate_and_sign!/2"},{"type":"function","title":"Lightning.Accounts.UserToken.last_used_changeset/1","doc":"Update when the api token was last used by setting last_used_at .","ref":"Lightning.Accounts.UserToken.html#last_used_changeset/1"},{"type":"function","title":"Lightning.Accounts.UserToken.token_and_context_query/2","doc":"Returns the token struct for the given token value and context.","ref":"Lightning.Accounts.UserToken.html#token_and_context_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.user_and_contexts_query/2","doc":"Gets all tokens for the given user for the given contexts.","ref":"Lightning.Accounts.UserToken.html#user_and_contexts_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate/3","doc":"Combines verify/2 and validate/2","ref":"Lightning.Accounts.UserToken.html#verify_and_validate/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate!/3","doc":"Same as verify_and_validate/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#verify_and_validate!/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_change_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. This is used to validate requests to change the user email. It is different from verify_email_token_query/2 precisely because verify_email_token_query/2 validates the email has not changed, which is the starting point by this function. The given token is valid if it matches its hashed counterpart in the database and if it has not expired (after @change_email_validity_in_days). The context must always start with "change:".","ref":"Lightning.Accounts.UserToken.html#verify_change_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The given token is valid if it matches its hashed counterpart in the database and the user email has not changed. This function also checks if the token is being used within a certain period, depending on the context. The default contexts supported by this function are either "confirm", for account confirmation emails, and "reset_password", for resetting the password. For verifying requests to change the email, see verify_change_email_token_query/2 .","ref":"Lightning.Accounts.UserToken.html#verify_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The token is valid if it matches the value in the database and it has not expired (after @auth_validity_in_seconds or @session_validity_in_days).","ref":"Lightning.Accounts.UserToken.html#verify_token_query/2"},{"type":"module","title":"Lightning.Credentials","doc":"The Credentials context.","ref":"Lightning.Credentials.html"},{"type":"function","title":"Lightning.Credentials.change_credential/2","doc":"Returns an %Ecto.Changeset{} for tracking credential changes. Examples iex> change_credential ( credential ) % Ecto.Changeset { data : % Credential { } }","ref":"Lightning.Credentials.html#change_credential/2"},{"type":"function","title":"Lightning.Credentials.create_credential/1","doc":"Creates a credential. Examples iex> create_credential ( %{ field : value } ) { :ok , % Credential { } } iex> create_credential ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#create_credential/1"},{"type":"function","title":"Lightning.Credentials.delete_credential/1","doc":"Deletes a credential. Examples iex> delete_credential ( credential ) { :ok , % Credential { } } iex> delete_credential ( credential ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#delete_credential/1"},{"type":"function","title":"Lightning.Credentials.get_credential!/1","doc":"Gets a single credential. Raises Ecto.NoResultsError if the Credential does not exist. Examples iex> get_credential! ( 123 ) % Credential { } iex> get_credential! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Credentials.html#get_credential!/1"},{"type":"function","title":"Lightning.Credentials.invalid_projects_for_user/2","doc":"Given a credential and a user, returns a list of invalid projects—i.e., those that the credential is shared with but that the user does not have access to. This is used to generate a validation error when a credential cannot be transferred. Examples iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ ] iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ "52ea8758-6ce5-43d7-912f-6a1e1f11dc55" ]","ref":"Lightning.Credentials.html#invalid_projects_for_user/2"},{"type":"function","title":"Lightning.Credentials.list_credentials/0","doc":"Returns the list of credentials. Examples iex> list_credentials ( ) [ % Credential { } , ... ]","ref":"Lightning.Credentials.html#list_credentials/0"},{"type":"function","title":"Lightning.Credentials.list_credentials/1","doc":"","ref":"Lightning.Credentials.html#list_credentials/1"},{"type":"function","title":"Lightning.Credentials.list_credentials_for_user/1","doc":"Returns the list of credentials for a given user. Examples iex> list_credentials_for_user ( 123 ) [ % Credential { user_id : 123 } , % Credential { user_id : 123 } , ... ]","ref":"Lightning.Credentials.html#list_credentials_for_user/1"},{"type":"function","title":"Lightning.Credentials.maybe_refresh_token/1","doc":"","ref":"Lightning.Credentials.html#maybe_refresh_token/1"},{"type":"function","title":"Lightning.Credentials.sensitive_values_for/1","doc":"","ref":"Lightning.Credentials.html#sensitive_values_for/1"},{"type":"function","title":"Lightning.Credentials.update_credential/2","doc":"Updates a credential. Examples iex> update_credential ( credential , %{ field : new_value } ) { :ok , % Credential { } } iex> update_credential ( credential , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#update_credential/2"},{"type":"module","title":"Lightning.Credentials.Audit","doc":"Model for storing changes to Credentials","ref":"Lightning.Credentials.Audit.html"},{"type":"function","title":"Lightning.Credentials.Audit.event/4","doc":"","ref":"Lightning.Credentials.Audit.html#event/4"},{"type":"function","title":"Lightning.Credentials.Audit.save/1","doc":"","ref":"Lightning.Credentials.Audit.html#save/1"},{"type":"module","title":"Lightning.Credentials.Credential","doc":"The Credential model.","ref":"Lightning.Credentials.Credential.html"},{"type":"type","title":"Lightning.Credentials.Credential.t/0","doc":"","ref":"Lightning.Credentials.Credential.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.Schema","doc":"Structure that can parse JsonSchemas (using ExJsonSchema ) and validate changesets for a given schema.","ref":"Lightning.Credentials.Schema.html"},{"type":"function","title":"Lightning.Credentials.Schema.new/2","doc":"","ref":"Lightning.Credentials.Schema.html#new/2"},{"type":"function","title":"Lightning.Credentials.Schema.properties/2","doc":"","ref":"Lightning.Credentials.Schema.html#properties/2"},{"type":"function","title":"Lightning.Credentials.Schema.required?/2","doc":"","ref":"Lightning.Credentials.Schema.html#required?/2"},{"type":"function","title":"Lightning.Credentials.Schema.validate/2","doc":"","ref":"Lightning.Credentials.Schema.html#validate/2"},{"type":"type","title":"Lightning.Credentials.Schema.t/0","doc":"","ref":"Lightning.Credentials.Schema.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.SchemaDocument","doc":"Provides facilities to dynamically create and validate a changeset for a given Schema","ref":"Lightning.Credentials.SchemaDocument.html"},{"type":"function","title":"Lightning.Credentials.SchemaDocument.changeset/3","doc":"","ref":"Lightning.Credentials.SchemaDocument.html#changeset/3"},{"type":"module","title":"Lightning.Credentials.SensitiveValues","doc":"Functions to pull out sensitive values inside a credential. These values are used to scrub logs for leaked secrets.","ref":"Lightning.Credentials.SensitiveValues.html"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.flatten_map/1","doc":"Keys that are not considered sensitive","ref":"Lightning.Credentials.SensitiveValues.html#flatten_map/1"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.secret_values/1","doc":"Given a map, find all values allowed (via @safe_keys ) and return them as a list.","ref":"Lightning.Credentials.SensitiveValues.html#secret_values/1"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:pairs/0"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.raw_pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:raw_pairs/0"},{"type":"module","title":"Lightning.Invocation","doc":"The Invocation context.","ref":"Lightning.Invocation.html"},{"type":"function","title":"Lightning.Invocation.change_dataclip/2","doc":"Returns an %Ecto.Changeset{} for tracking dataclip changes. Examples iex> change_dataclip ( dataclip ) % Ecto.Changeset { data : % Dataclip { } }","ref":"Lightning.Invocation.html#change_dataclip/2"},{"type":"function","title":"Lightning.Invocation.change_run/2","doc":"Returns an %Ecto.Changeset{} for tracking run changes. Examples iex> change_run ( run ) % Ecto.Changeset { data : % Run { } }","ref":"Lightning.Invocation.html#change_run/2"},{"type":"function","title":"Lightning.Invocation.create_dataclip/1","doc":"Creates a dataclip. Examples iex> create_dataclip ( %{ field : value } ) { :ok , % Dataclip { } } iex> create_dataclip ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_dataclip/1"},{"type":"function","title":"Lightning.Invocation.create_log_line/2","doc":"","ref":"Lightning.Invocation.html#create_log_line/2"},{"type":"function","title":"Lightning.Invocation.create_run/1","doc":"Creates a run. Examples iex> create_run ( %{ field : value } ) { :ok , % Run { } } iex> create_run ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_run/1"},{"type":"function","title":"Lightning.Invocation.delete_dataclip/1","doc":"Deletes a dataclip. Examples iex> delete_dataclip ( dataclip ) { :ok , % Dataclip { } } iex> delete_dataclip ( dataclip ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_dataclip/1"},{"type":"function","title":"Lightning.Invocation.delete_run/1","doc":"Deletes a run. Examples iex> delete_run ( run ) { :ok , % Run { } } iex> delete_run ( run ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_run/1"},{"type":"function","title":"Lightning.Invocation.filter_run_body_and_logs_where/2","doc":"","ref":"Lightning.Invocation.html#filter_run_body_and_logs_where/2"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_before_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_status_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_status_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workflow_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workflow_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_before_where/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip/1","doc":"Gets a single dataclip given one of: a Dataclip uuid a Run model Returns nil if the Dataclip does not exist. Examples iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) % Dataclip { } iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) nil iex> get_dataclip ( % Run { id : "a uuid" } ) % Dataclip { }","ref":"Lightning.Invocation.html#get_dataclip/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip!/1","doc":"Gets a single dataclip. Raises Ecto.NoResultsError if the Dataclip does not exist. Examples iex> get_dataclip! ( 123 ) % Dataclip { } iex> get_dataclip! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_dataclip!/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip_query/1","doc":"Query for retrieving the dataclip that a runs starting dataclip.","ref":"Lightning.Invocation.html#get_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_result_dataclip_query/1","doc":"Query for retrieving the dataclip that was the result of a successful run.","ref":"Lightning.Invocation.html#get_result_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_run!/1","doc":"Gets a single run. Raises Ecto.NoResultsError if the Run does not exist. Examples iex> get_run! ( 123 ) % Run { } iex> get_run! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_run!/1"},{"type":"function","title":"Lightning.Invocation.get_run_with_job!/1","doc":"Fetches a run and preloads the job via the run's event.","ref":"Lightning.Invocation.html#get_run_with_job!/1"},{"type":"function","title":"Lightning.Invocation.get_workorders_by_ids/1","doc":"","ref":"Lightning.Invocation.html#get_workorders_by_ids/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips/0","doc":"Returns the list of dataclips. Examples iex> list_dataclips ( ) [ % Dataclip { } , ... ]","ref":"Lightning.Invocation.html#list_dataclips/0"},{"type":"function","title":"Lightning.Invocation.list_dataclips/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_for_job/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_for_job/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_query/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_query/1"},{"type":"function","title":"Lightning.Invocation.list_runs/0","doc":"Returns the list of runs. Examples iex> list_runs ( ) [ % Run { } , ... ]","ref":"Lightning.Invocation.html#list_runs/0"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project/2","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project/2"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project_query/1","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project_query/1"},{"type":"function","title":"Lightning.Invocation.list_work_orders_for_project_query/2","doc":"","ref":"Lightning.Invocation.html#list_work_orders_for_project_query/2"},{"type":"function","title":"Lightning.Invocation.search_workorders/1","doc":"","ref":"Lightning.Invocation.html#search_workorders/1"},{"type":"function","title":"Lightning.Invocation.search_workorders/3","doc":"","ref":"Lightning.Invocation.html#search_workorders/3"},{"type":"function","title":"Lightning.Invocation.update_dataclip/2","doc":"Updates a dataclip. Examples iex> update_dataclip ( dataclip , %{ field : new_value } ) { :ok , % Dataclip { } } iex> update_dataclip ( dataclip , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_dataclip/2"},{"type":"function","title":"Lightning.Invocation.update_run/2","doc":"Updates a run. Examples iex> update_run ( run , %{ field : new_value } ) { :ok , % Run { } } iex> update_run ( run , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_run/2"},{"type":"function","title":"Lightning.Invocation.with_attempts/1","doc":"","ref":"Lightning.Invocation.html#with_attempts/1"},{"type":"module","title":"Lightning.Invocation.Dataclip","doc":"Ecto model for Dataclips. Dataclips represent some data that arrived in the system, and records both the data and the source of the data. Types :http_request The data arrived via a webhook. :global Was created manually, and is intended to be used multiple times. When repetitive static data is needed to be maintained, instead of hard-coding into a Job - a more convenient solution is to create a :global Dataclip and access it inside the Job. :run_result The final state of a successful run. :saved_input An arbitrary input, created by a user. (Only configuration will be overwritten.)","ref":"Lightning.Invocation.Dataclip.html"},{"type":"function","title":"Lightning.Invocation.Dataclip.get_types/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#get_types/0"},{"type":"function","title":"Lightning.Invocation.Dataclip.new/1","doc":"","ref":"Lightning.Invocation.Dataclip.html#new/1"},{"type":"function","title":"Lightning.Invocation.Dataclip.validate_by_type/1","doc":"Append validations based on the type of the Dataclip. :run_result must have an associated Run model.","ref":"Lightning.Invocation.Dataclip.html#validate_by_type/1"},{"type":"type","title":"Lightning.Invocation.Dataclip.source_type/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:source_type/0"},{"type":"type","title":"Lightning.Invocation.Dataclip.t/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.LogLine","doc":"Ecto model for run logs.","ref":"Lightning.Invocation.LogLine.html"},{"type":"function","title":"Lightning.Invocation.LogLine.validate/1","doc":"","ref":"Lightning.Invocation.LogLine.html#validate/1"},{"type":"type","title":"Lightning.Invocation.LogLine.t/0","doc":"","ref":"Lightning.Invocation.LogLine.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.Query","doc":"Query functions for working with Runs and Dataclips","ref":"Lightning.Invocation.Query.html"},{"type":"function","title":"Lightning.Invocation.Query.last_run_for_job/1","doc":"The last run for a job","ref":"Lightning.Invocation.Query.html#last_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.last_successful_run_for_job/1","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#last_successful_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_for/1","doc":"Runs for a specific user","ref":"Lightning.Invocation.Query.html#runs_for/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_with_code/2","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#runs_with_code/2"},{"type":"module","title":"Lightning.Invocation.Run","doc":"Ecto model for Runs. A run represents the work initiated for a Job with an input dataclip. Once completed (successfully) it will have an output_dataclip associated with it as well.","ref":"Lightning.Invocation.Run.html"},{"type":"function","title":"Lightning.Invocation.Run.new/1","doc":"","ref":"Lightning.Invocation.Run.html#new/1"},{"type":"function","title":"Lightning.Invocation.Run.new_from/1","doc":"Creates a new Run changeset, but copies over certain fields. This is used to create new runs for retrys.","ref":"Lightning.Invocation.Run.html#new_from/1"},{"type":"type","title":"Lightning.Invocation.Run.t/0","doc":"","ref":"Lightning.Invocation.Run.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReason","doc":"Ecto model for InvocationReasons.","ref":"Lightning.InvocationReason.html"},{"type":"function","title":"Lightning.InvocationReason.new/1","doc":"","ref":"Lightning.InvocationReason.html#new/1"},{"type":"function","title":"Lightning.InvocationReason.validate_by_trigger_type/1","doc":"","ref":"Lightning.InvocationReason.html#validate_by_trigger_type/1"},{"type":"type","title":"Lightning.InvocationReason.source_type/0","doc":"","ref":"Lightning.InvocationReason.html#t:source_type/0"},{"type":"type","title":"Lightning.InvocationReason.t/0","doc":"","ref":"Lightning.InvocationReason.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReasons","doc":"The InvocationReasons context.","ref":"Lightning.InvocationReasons.html"},{"type":"function","title":"Lightning.InvocationReasons.build/2","doc":"","ref":"Lightning.InvocationReasons.html#build/2"},{"type":"function","title":"Lightning.InvocationReasons.create_reason/1","doc":"Creates a reason. Examples iex> create_reason ( %{ field : value } ) { :ok , % InvocationReason { } } iex> create_reason ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.InvocationReasons.html#create_reason/1"},{"type":"type","title":"Lightning.InvocationReasons.reason_type/0","doc":"","ref":"Lightning.InvocationReasons.html#t:reason_type/0"},{"type":"module","title":"Lightning.Pipeline","doc":"Service class to coordinate the running of jobs, and their downstream jobs.","ref":"Lightning.Pipeline.html"},{"type":"function","title":"Lightning.Pipeline.assemble_logs_for_run/1","doc":"Return all logs for a run as a string of text, separated by new line breaks","ref":"Lightning.Pipeline.html#assemble_logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.logs_for_run/1","doc":"Return all logs for a run as a list","ref":"Lightning.Pipeline.html#logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.process/1","doc":"","ref":"Lightning.Pipeline.html#process/1"},{"type":"module","title":"Lightning.Pipeline.Runner","doc":"Job running entrypoint","ref":"Lightning.Pipeline.Runner.html"},{"type":"function","title":"Lightning.Pipeline.Runner.create_dataclip_from_result/2","doc":"Creates a dataclip linked to the run that just finished. If either the file doesn't exist or there is a JSON decoding error, it logs and returns an error tuple.","ref":"Lightning.Pipeline.Runner.html#create_dataclip_from_result/2"},{"type":"function","title":"Lightning.Pipeline.Runner.find_or_install_adaptor/1","doc":"Make sure an adaptor matching the name is available. If it is available, return it's Engine.Adaptor struct - if not then install it.","ref":"Lightning.Pipeline.Runner.html#find_or_install_adaptor/1"},{"type":"function","title":"Lightning.Pipeline.Runner.scrub_result/1","doc":"Scrubs values from all keys in configuration, will be replaced by extensions to scrubber.ex, which is currently only used for logs.","ref":"Lightning.Pipeline.Runner.html#scrub_result/1"},{"type":"function","title":"Lightning.Pipeline.Runner.start/2","doc":"Execute a Run. Given a valid run: Persist the Dataclip and the Job's body to disk Create a blank output file on disk Build up a %Lightning.Runtime.Runspec{} with the paths, and adaptor module name And start it via Handler.start/2 . The callbacks implemented on Handler ( c:Handler.on_start/1 and c:Handler.on_finish/2 ) update the run when a Run is started and when it's finished, attaching the exit_code and log when they are available.","ref":"Lightning.Pipeline.Runner.html#start/2"},{"type":"module","title":"Lightning.Pipeline.Runner.Handler","doc":"Custom handler callbacks for Lightnings use of Engine to execute runs.","ref":"Lightning.Pipeline.Runner.Handler.html"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.on_start/1","doc":"The on_start handler updates the run, setting the started_at time and stamping the run with the ID of the credential that was used, if any, to facilitate easier auditing.","ref":"Lightning.Pipeline.Runner.Handler.html#on_start/1"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.stop/1","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#stop/1"},{"type":"type","title":"Lightning.Pipeline.Runner.Handler.handler_opts/0","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#t:handler_opts/0"},{"type":"module","title":"Lightning.Pipeline.StateAssembler","doc":"Facilities for building the state for a Run How state is assembled For the most common jobs, an inbound webhook will store an :http_request type dataclip. The reason that is created is associated with the dataclip. At runtime, the initial state for a Run will be in the shape of: { "data" : < the dataclip > , "configuration" : < the job ' s credential > } Saved inputs Saved custom inputs will only have state.configuration changed, everything else will remain as displayed. Flow Jobs When a Job is triggered by a previous Jobs success or failure these are the rules for constructing that Jobs state: For jobs that trigger on it's upstream jobs failure, the event will have the previous runs input dataclip as its input dataclip. The state will also have the log of the previous run attached on the error key. For Jobs triggered by a previous success, the run will have the previous runs output dataclip as its input dataclip. :run_result dataclips are expected to already have a data key, and are merged into the root.","ref":"Lightning.Pipeline.StateAssembler.html"},{"type":"function","title":"Lightning.Pipeline.StateAssembler.assemble/1","doc":"Assemble state for use in a Run.","ref":"Lightning.Pipeline.StateAssembler.html#assemble/1"},{"type":"module","title":"Lightning.Jobs","doc":"The Jobs context.","ref":"Lightning.Jobs.html"},{"type":"function","title":"Lightning.Jobs.change_job/2","doc":"Returns an %Ecto.Changeset{} for tracking job changes. Examples iex> change_job ( job ) % Ecto.Changeset { data : % Job { } }","ref":"Lightning.Jobs.html#change_job/2"},{"type":"function","title":"Lightning.Jobs.create_job/1","doc":"Creates a job. Examples iex> create_job ( %{ field : value } ) { :ok , % Job { } } iex> create_job ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#create_job/1"},{"type":"function","title":"Lightning.Jobs.delete_job/1","doc":"Deletes a job. Examples iex> delete_job ( job ) { :ok , % Job { } } iex> delete_job ( job ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#delete_job/1"},{"type":"function","title":"Lightning.Jobs.get_downstream_jobs_for/2","doc":"Returns the list of downstream jobs for a given job, optionally matching a specific trigger type. When downstream_jobs_for is called without a trigger that means its between jobs when it called with a trigger that means we are starting from outside the pipeline","ref":"Lightning.Jobs.html#get_downstream_jobs_for/2"},{"type":"function","title":"Lightning.Jobs.get_job/1","doc":"","ref":"Lightning.Jobs.html#get_job/1"},{"type":"function","title":"Lightning.Jobs.get_job!/1","doc":"Gets a single job. Raises Ecto.NoResultsError if the Job does not exist. Examples iex> get_job! ( 123 ) % Job { } iex> get_job! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Jobs.html#get_job!/1"},{"type":"function","title":"Lightning.Jobs.get_upstream_jobs_for/1","doc":"Returns the list of jobs excluding the one given.","ref":"Lightning.Jobs.html#get_upstream_jobs_for/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project_query/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project_query/1"},{"type":"function","title":"Lightning.Jobs.list_active_cron_jobs/0","doc":"","ref":"Lightning.Jobs.html#list_active_cron_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs/0","doc":"Returns the list of jobs.","ref":"Lightning.Jobs.html#list_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs_for_workflow/1","doc":"","ref":"Lightning.Jobs.html#list_jobs_for_workflow/1"},{"type":"function","title":"Lightning.Jobs.update_job/2","doc":"Updates a job. Examples iex> update_job ( job , %{ field : new_value } ) { :ok , % Job { } } iex> update_job ( job , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#update_job/2"},{"type":"module","title":"Lightning.Jobs.Job","doc":"Ecto model for Jobs. A Job contains the fields for defining a job. body The expression/javascript code name A plain text identifier adaptor An NPM style string that contains both the module name and it's version. E.g. @openfn/language-http@v1.2.3 or @openfn/language-foo@latest . While the version suffix isn't enforced here as it's not strictly necessary in this context, the front end will ensure a version is stated ( @latest being the default).","ref":"Lightning.Jobs.Job.html"},{"type":"function","title":"Lightning.Jobs.Job.new/1","doc":"","ref":"Lightning.Jobs.Job.html#new/1"},{"type":"function","title":"Lightning.Jobs.Job.put_project_credential/2","doc":"","ref":"Lightning.Jobs.Job.html#put_project_credential/2"},{"type":"function","title":"Lightning.Jobs.Job.put_workflow/2","doc":"Attaches a workflow to a job, this is useful when you have an unpersisted Workflow changeset - and want it to be created at the same time as a Job. Example: workflow = Ecto.Changeset . cast ( % Lightning.Workflows.Workflow { } , %{ "project_id" => attrs [ :project_id ] , "id" => Ecto.UUID . generate ( ) } , [ :project_id , :id ] ) job = % Job { } |> Ecto.Changeset . change ( ) |> Job . put_workflow ( workflow ) |> Job . changeset ( attrs )","ref":"Lightning.Jobs.Job.html#put_workflow/2"},{"type":"function","title":"Lightning.Jobs.Job.validate/1","doc":"","ref":"Lightning.Jobs.Job.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Job.t/0","doc":"","ref":"Lightning.Jobs.Job.html#t:t/0"},{"type":"module","title":"Lightning.Jobs.Query","doc":"Query module for finding Jobs.","ref":"Lightning.Jobs.Query.html"},{"type":"function","title":"Lightning.Jobs.Query.enabled_cron_jobs_by_edge/0","doc":"Returns active jobs with their cron triggers for use in the cron scheduling service.","ref":"Lightning.Jobs.Query.html#enabled_cron_jobs_by_edge/0"},{"type":"function","title":"Lightning.Jobs.Query.jobs_for/1","doc":"Returns all jobs accessible to a user, via their projects or all jobs in a given project.","ref":"Lightning.Jobs.Query.html#jobs_for/1"},{"type":"module","title":"Lightning.Jobs.Scheduler","doc":"The Scheduler is responsible for finding jobs that are ready to run based on their cron schedule, and then running them.","ref":"Lightning.Jobs.Scheduler.html"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/0","doc":"Find and start any cronjobs that are scheduled to run for a given time (defaults to the current time).","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/0"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/1","doc":"","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/1"},{"type":"module","title":"Lightning.Jobs.Trigger","doc":"Ecto model for Triggers. Triggers represent the criteria in which a Job might be invoked. Types Webhook (default) A webhook trigger allows a Job to invoked (via Lightning.Invocation ) when it's endpoint is called.","ref":"Lightning.Jobs.Trigger.html"},{"type":"function","title":"Lightning.Jobs.Trigger.new/1","doc":"","ref":"Lightning.Jobs.Trigger.html#new/1"},{"type":"function","title":"Lightning.Jobs.Trigger.validate/1","doc":"","ref":"Lightning.Jobs.Trigger.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Trigger.t/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:t/0"},{"type":"type","title":"Lightning.Jobs.Trigger.trigger_type/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:trigger_type/0"},{"type":"module","title":"Lightning.Projects","doc":"The Projects context.","ref":"Lightning.Projects.html"},{"type":"function","title":"Lightning.Projects.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Projects.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Projects.change_project/2","doc":"Returns an %Ecto.Changeset{} for tracking project changes. Examples iex> change_project ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#change_project/2"},{"type":"function","title":"Lightning.Projects.create_project/1","doc":"Creates a project. Examples iex> create_project ( %{ field : value } ) { :ok , % Project { } } iex> create_project ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#create_project/1"},{"type":"function","title":"Lightning.Projects.delete_project/1","doc":"Deletes a project and its related data, including workflows, work orders, runs, jobs, attempts, triggers, project users, project credentials, and dataclips Examples iex> delete_project ( project ) { :ok , % Project { } } iex> delete_project ( project ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#delete_project/1"},{"type":"function","title":"Lightning.Projects.export_project/2","doc":"Exports a project as yaml. Examples iex> export_project ( :yaml , project_id ) { :ok , string }","ref":"Lightning.Projects.html#export_project/2"},{"type":"function","title":"Lightning.Projects.get_project/1","doc":"","ref":"Lightning.Projects.html#get_project/1"},{"type":"function","title":"Lightning.Projects.get_project!/1","doc":"Gets a single project. Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project!/1"},{"type":"function","title":"Lightning.Projects.get_project_credential/2","doc":"","ref":"Lightning.Projects.html#get_project_credential/2"},{"type":"function","title":"Lightning.Projects.get_project_user/1","doc":"","ref":"Lightning.Projects.html#get_project_user/1"},{"type":"function","title":"Lightning.Projects.get_project_user/2","doc":"","ref":"Lightning.Projects.html#get_project_user/2"},{"type":"function","title":"Lightning.Projects.get_project_user!/1","doc":"Gets a single project_user. Raises Ecto.NoResultsError if the ProjectUser does not exist. Examples iex> get_project_user! ( 123 ) % ProjectUser { } iex> get_project_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_user!/1"},{"type":"function","title":"Lightning.Projects.get_project_user_role/2","doc":"Returns the role of a user in a project. Possible roles are :admin, :viewer, :editor, and :owner Examples iex> get_project_user_role ( user , project ) :admin iex> get_project_user_role ( user , project ) :viewer iex> get_project_user_role ( user , project ) :editor iex> get_project_user_role ( user , project ) :owner","ref":"Lightning.Projects.html#get_project_user_role/2"},{"type":"function","title":"Lightning.Projects.get_project_with_users!/1","doc":"Gets a single project with it's members via project_users . Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_with_users!/1"},{"type":"function","title":"Lightning.Projects.get_projects_for_user/1","doc":"","ref":"Lightning.Projects.html#get_projects_for_user/1"},{"type":"function","title":"Lightning.Projects.is_member_of?/2","doc":"","ref":"Lightning.Projects.html#is_member_of?/2"},{"type":"function","title":"Lightning.Projects.list_project_credentials/1","doc":"","ref":"Lightning.Projects.html#list_project_credentials/1"},{"type":"function","title":"Lightning.Projects.list_projects/0","doc":"Returns the list of projects. Examples iex> list_projects ( ) [ % Project { } , ... ]","ref":"Lightning.Projects.html#list_projects/0"},{"type":"function","title":"Lightning.Projects.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find projects that are ready for permanent deletion and purge them.","ref":"Lightning.Projects.html#perform/1"},{"type":"function","title":"Lightning.Projects.project_attempt_run_query/1","doc":"","ref":"Lightning.Projects.html#project_attempt_run_query/1"},{"type":"function","title":"Lightning.Projects.project_attempts_query/1","doc":"","ref":"Lightning.Projects.html#project_attempts_query/1"},{"type":"function","title":"Lightning.Projects.project_credentials_query/1","doc":"","ref":"Lightning.Projects.html#project_credentials_query/1"},{"type":"function","title":"Lightning.Projects.project_dataclip_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_dataclip_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_dataclips_query/1","doc":"","ref":"Lightning.Projects.html#project_dataclips_query/1"},{"type":"function","title":"Lightning.Projects.project_jobs_query/1","doc":"","ref":"Lightning.Projects.html#project_jobs_query/1"},{"type":"function","title":"Lightning.Projects.project_run_invocation_reasons/1","doc":"","ref":"Lightning.Projects.html#project_run_invocation_reasons/1"},{"type":"function","title":"Lightning.Projects.project_runs_query/1","doc":"","ref":"Lightning.Projects.html#project_runs_query/1"},{"type":"function","title":"Lightning.Projects.project_trigger_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_trigger_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_triggers_query/1","doc":"","ref":"Lightning.Projects.html#project_triggers_query/1"},{"type":"function","title":"Lightning.Projects.project_user_role_query/2","doc":"","ref":"Lightning.Projects.html#project_user_role_query/2"},{"type":"function","title":"Lightning.Projects.project_users_query/1","doc":"","ref":"Lightning.Projects.html#project_users_query/1"},{"type":"function","title":"Lightning.Projects.project_workflows_query/1","doc":"","ref":"Lightning.Projects.html#project_workflows_query/1"},{"type":"function","title":"Lightning.Projects.project_workorders_query/1","doc":"","ref":"Lightning.Projects.html#project_workorders_query/1"},{"type":"function","title":"Lightning.Projects.projects_for_user_query/1","doc":"","ref":"Lightning.Projects.html#projects_for_user_query/1"},{"type":"function","title":"Lightning.Projects.schedule_project_deletion/1","doc":"Given a project, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic project purge cronjob will never run. (Note that subsequent logins will be blocked for projects pending deletion.)","ref":"Lightning.Projects.html#schedule_project_deletion/1"},{"type":"function","title":"Lightning.Projects.select_first_project_for_user/1","doc":"","ref":"Lightning.Projects.html#select_first_project_for_user/1"},{"type":"function","title":"Lightning.Projects.update_project/2","doc":"Updates a project. Examples iex> update_project ( project , %{ field : new_value } ) { :ok , % Project { } } iex> update_project ( project , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project/2"},{"type":"function","title":"Lightning.Projects.update_project_user/2","doc":"Updates a project user. Examples iex> update_project_user ( project_user , %{ field : new_value } ) { :ok , % ProjectUser { } } iex> update_project_user ( projectUser , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project_user/2"},{"type":"function","title":"Lightning.Projects.url_safe_project_name/1","doc":"","ref":"Lightning.Projects.html#url_safe_project_name/1"},{"type":"function","title":"Lightning.Projects.validate_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the project scheduled_deletion. Examples iex> validate_for_deletion ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#validate_for_deletion/2"},{"type":"module","title":"Lightning.Projects.Project","doc":"Project model","ref":"Lightning.Projects.Project.html"},{"type":"function","title":"Lightning.Projects.Project.deletion_changeset/2","doc":"Changeset to validate a project deletion request, the user must enter the projects name to confirm.","ref":"Lightning.Projects.Project.html#deletion_changeset/2"},{"type":"function","title":"Lightning.Projects.Project.validate/1","doc":"","ref":"Lightning.Projects.Project.html#validate/1"},{"type":"type","title":"Lightning.Projects.Project.t/0","doc":"","ref":"Lightning.Projects.Project.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectCredential","doc":"Join table to assign credentials to a project","ref":"Lightning.Projects.ProjectCredential.html"},{"type":"type","title":"Lightning.Projects.ProjectCredential.t/0","doc":"","ref":"Lightning.Projects.ProjectCredential.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser","doc":"Join table to assign users to a project","ref":"Lightning.Projects.ProjectUser.html"},{"type":"type","title":"Lightning.Projects.ProjectUser.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.DigestEnum","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.DigestEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.RolesEnum","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.RolesEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.Provisioner","doc":"Provides functions for importing projects. This module is used by the provisioning HTTP API. When providing a project to import, all records must have an id field. It's up to the caller to ensure that the id is unique and generated ahead of time in the case of new records.","ref":"Lightning.Projects.Provisioner.html"},{"type":"function","title":"Lightning.Projects.Provisioner.import_document/3","doc":"Import a project.","ref":"Lightning.Projects.Provisioner.html#import_document/3"},{"type":"function","title":"Lightning.Projects.Provisioner.load_project/1","doc":"Load a project by ID, including all workflows and their associated jobs, triggers and edges. Returns nil if the project does not exist.","ref":"Lightning.Projects.Provisioner.html#load_project/1"},{"type":"function","title":"Lightning.Projects.Provisioner.parse_document/2","doc":"","ref":"Lightning.Projects.Provisioner.html#parse_document/2"},{"type":"function","title":"Lightning.Projects.Provisioner.validate_extraneous_params/1","doc":"Validate that there are no extraneous parameters in the changeset. For all params in the changeset, ensure that the param is in the list of known fields in the schema.","ref":"Lightning.Projects.Provisioner.html#validate_extraneous_params/1"},{"type":"module","title":"Lightning.Runtime.ChildProcess","doc":"Provides an interface between a RunSpec and the shell. Internally it calls node , and more specifically the OpenFn core CLI.","ref":"Lightning.Runtime.ChildProcess.html"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_command/1","doc":"Builds up a string for shell execution based on the RunSpec","ref":"Lightning.Runtime.ChildProcess.html#build_command/1"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_env/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#build_env/2"},{"type":"function","title":"Lightning.Runtime.ChildProcess.run/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#run/2"},{"type":"behaviour","title":"Lightning.Runtime.Handler","doc":"A strategy for executing things via ChildProcess. This module handles the dirty bits, setting up processes and coordinating results (and logs) as they arrive. Since it is a macro, see Lightning.Pipeline.Runner.Handler for a usage example.","ref":"Lightning.Runtime.Handler.html"},{"type":"function","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.log_callback/3","doc":"","ref":"Lightning.Runtime.Handler.html#c:log_callback/3"},{"type":"function","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_finish/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_finish/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_log_emit/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_log_emit/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_start/1","doc":"","ref":"Lightning.Runtime.Handler.html#on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.on_start/1","doc":"Called with context, if any - when the Run has been started.","ref":"Lightning.Runtime.Handler.html#c:on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.start/2","doc":"The entrypoint for executing a run.","ref":"Lightning.Runtime.Handler.html#c:start/2"},{"type":"type","title":"Lightning.Runtime.Handler.t/0","doc":"","ref":"Lightning.Runtime.Handler.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.LogAgent","doc":"Agent facility to consume STDOUT/STDERR byte by byte. Since it works on a byte by byte basis, you will need to perform line-splitting yourself. Usage: { :ok , log } = LogAgent . start_link ( ) "foo" = LogAgent . process_chunk ( log , { :stdout , "foo" } ) "foobar" = LogAgent . process_chunk ( log , { :stdout , "bar" } )","ref":"Lightning.Runtime.LogAgent.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Runtime.LogAgent.html#child_spec/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.start_link/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#start_link/1"},{"type":"type","title":"Lightning.Runtime.LogAgent.logline/0","doc":"","ref":"Lightning.Runtime.LogAgent.html#t:logline/0"},{"type":"module","title":"Lightning.Runtime.LogAgent.StringBuffer","doc":"Internal datastructure to hold and process new bytes for a list of characters. By checking the if the buffer is a complete grapheme, emitting the buffer once valid and returning nil otherwise. In the case of emojis and other language character sets, a character (in UTF-8) can be between 1-4 bytes - when streaming logs for example it's quite easy to receive less than the whole character which can result in crashes or corrupt text.","ref":"Lightning.Runtime.LogAgent.StringBuffer.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.new/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#new/0"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.reduce_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#reduce_chunk/2"},{"type":"type","title":"Lightning.Runtime.LogAgent.StringBuffer.t/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.Result","doc":"Data structure used to represent the result of a Run executed by Lightning.Runtime.ChildProcess .","ref":"Lightning.Runtime.Result.html"},{"type":"function","title":"Lightning.Runtime.Result.new/1","doc":"","ref":"Lightning.Runtime.Result.html#new/1"},{"type":"type","title":"Lightning.Runtime.Result.t/0","doc":"","ref":"Lightning.Runtime.Result.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.RunSpec","doc":"A struct containing all the parameters required to execute a Job.","ref":"Lightning.Runtime.RunSpec.html"},{"type":"function","title":"Lightning.Runtime.RunSpec.new/1","doc":"","ref":"Lightning.Runtime.RunSpec.html#new/1"},{"type":"type","title":"Lightning.Runtime.RunSpec.t/0","doc":"","ref":"Lightning.Runtime.RunSpec.html#t:t/0"},{"type":"exception","title":"Lightning.MetadataService.Error","doc":"","ref":"Lightning.MetadataService.Error.html"},{"type":"function","title":"Lightning.MetadataService.Error.message/1","doc":"Callback implementation for Exception.message/1 .","ref":"Lightning.MetadataService.Error.html#message/1"},{"type":"function","title":"Lightning.MetadataService.Error.new/1","doc":"","ref":"Lightning.MetadataService.Error.html#new/1"},{"type":"type","title":"Lightning.MetadataService.Error.t/0","doc":"","ref":"Lightning.MetadataService.Error.html#t:t/0"},{"type":"extras","title":"Lightning","doc":"OpenFn/Lightning is a fully open source, workflow automation platform that's used to automate critical business processes and integrate information systems. From last-mile services to national-level reporting, it boosts efficiency & effectiveness while enabling secure, stable, scalable interoperability at all levels. Read more about Lightning on OpenFn/Docs . Lightning is the latest "v2" of OpenFn, the Digital Public Good for workflow automation . Use Lightning to visually build, execute and manage workflows. The latest version of the OpenFn technology - first launched in 2014, now tried and tested by NGOs in over 40 countries Fully open source (no premium features or community edition, you get the same product whether using SaaS or self-hosted) Recognised as a Digital Public Good by the DPGA and a Global Good for Health by Digital Square","ref":"readme.html"},{"type":"extras","title":"Lightning - Contents","doc":"Demo Sign up for the BETA Features Getting started Run Lightning via Docker Deploy Lightning on Docker or Kubernetes Run Lightning on your local machine (contributors) Troubleshooting Contribute Project Configuration Quickstart Generate the documentation Security and standards Server specs for self-hosting Questions or feedback?","ref":"readme.html#contents"},{"type":"extras","title":"Lightning - Demo","doc":"Watch a quick demo here: https://www.youtube.com/watch?v=BNaxlHAWb5I Explore our demo app* with username: demo@openfn.org , password: welcome123 , or read through the features section to view screenshots of the app. *Note that the demo app refreshes daily, so do not configure workflows you want to save.","ref":"readme.html#demo"},{"type":"extras","title":"Lightning - Register for a Beta account","doc":"Register for a Beta account at app.openfn.org and go through the quick-start guide to get familiar with the app.","ref":"readme.html#register-for-a-beta-account"},{"type":"extras","title":"Lightning - Features","doc":"Build Plan and build workflows using Lightning's visual interface to quickly define when, where and what you want your automation to do. Use our CLI to quickly build, edit and deploy projects from the comfort of your own code editor. Monitor Monitor all workflow activity in one place. Filter and search runs to identify issues that need addressing and follow how a specific request has been processed Configure alerts to be notified on run failures Receive a project digest for a daily/weekly/monthly summary of your project activity Manage Manage users and access by project Roles and permissions Authorization is a central part of Lightning. As such, users are given different roles which determine what level of access they have for resources in the application. For more details about roles and permissions in Lightning, please refer to our documentation . Roadmap View our public roadmap here .","ref":"readme.html#features"},{"type":"extras","title":"Lightning - Getting Started","doc":"If you only want to RUN Lightning on your own server, we recommend using Docker. If you want to DEPLOY Lightning, we recommend Docker builds and Kubernetes. If you want to CONTRIBUTE to the project, we recommend running Lightning on your local machine .","ref":"readme.html#getting-started"},{"type":"extras","title":"Lightning - Run via Docker","doc":"Install the latest version of Docker Clone this repo using git Copy the .env.example file to .env Run docker compose run --rm web mix ecto.migrate By default the application will be running at localhost:4000 . You can then rebuild and run with docker compose build and docker compose up . See "Problems with Docker" for additional troubleshooting help. Note that you can also create your own docker-compose.yml file, configuring a postgres database and using a pre-built image from Dockerhub.","ref":"readme.html#run-via-docker"},{"type":"extras","title":"Lightning - Deploy on external infrastructure","doc":"See Deployment for more detailed information.","ref":"readme.html#deploy-on-external-infrastructure"},{"type":"extras","title":"Lightning - Run Lightning locally","doc":"Clone the repo and optionally set ENVs git clone git@github.com:OpenFn/Lightning.git # or from YOUR fork! cd Lightning cp .env.example .env # and adjust as necessary! Take note of database names and ports in particular—they've got to match across your Postgres setup and your ENVs. You can run lightning without any ENVs assuming a vanilla postgres setup (see below), but you may want to make adjustments. Database Setup If you're already using Postgres locally, create a new database called lightning_dev , for example. If you'd rather use Docker to set up a Postgres DB, create a new volume and image: docker volume create lightning-postgres-data docker create \\ --name lightning-postgres \\ --mount source=lightning-postgres-data,target=/var/lib/postgresql/data \\ --publish 5432:5432 \\ -e POSTGRES_PASSWORD=postgres \\ postgres:15.3-alpine docker start lightning-postgres Elixir & Ecto Setup We use asdf to configure our local environments. Included in the repo is a .tool-versions file that is read by asdf in order to dynamically make the specified versions of Elixir and Erlang available. You'll need asdf plugins for Erlang , NodeJs Elixir and k6 . asdf install # Install language versions mix local.hex mix deps.get mix local.rebar --force mix ecto.create # Create a development database in Postgres mix ecto.migrate [[ $(uname -m) == 'arm64' ]] && mix compile.rambo # Force compile rambo if on M1 mix lightning.install_runtime mix lightning.install_schemas npm install --prefix assets Run the app Lightning is a web app. To run it in interactive Elixir mode, start the development server by running with your environment variables by running: iex -S mix phx.server or if you have set up custom environment variables, run: env $(cat .env | grep -v "#" | xargs ) iex -S mix phx.server Once the server has started, head to localhost:4000 in your browser. Run the tests Before the first time running the tests, you need a test database setup. MIX_ENV=test mix ecto.create And then after that run the tests using: MIX_ENV=test mix test We also have test.watch installed which can be used to rerun the tests on file changes.","ref":"readme.html#run-lightning-locally"},{"type":"extras","title":"Lightning - Benchmarking","doc":"We are using k6 to benchmark Lightning. Under benchmarking folder you can find a script for benchmarking Webhook Workflows. See Benchmarking for more detailed information.","ref":"readme.html#benchmarking"},{"type":"extras","title":"Lightning - Troubleshooting","doc":"Trouble with environment variables For troubleshooting custom environment variable configuration it's important to know how an Elixir app loads and modifies configuration. The order is as follows: Stuff in config.exs is loaded. That is then modified (think: overwritten ) by stuff your ENV-specific config: dev.exs , prod.exs or test.exs . That is then modified by runtime.exs which is where you are allowed to use System.env() Finally init/2 (if present in a child application) gets called (which takes the config which has been set in steps 1-3) when that child application is started during the parent app startup defined in application.ex . Problems with Postgres If you're having connecting issues with Postgres, check the database section of your .env to ensure the DB url is correctly set for your environment — note that composing a DB url out of other, earlier declared variables, does not work while using xargs . Problems with Debian If you're getting this error on debian == > earmark_parser Compiling 1 file ( . yrl ) / usr / lib / erlang / lib / parsetools - 2.3 . 1 / include / yeccpre . hrl : no such file or directory could not compile dependency :earmark_parser , "mix compile" failed . You can recompile this dependency with "mix deps.compile earmark_parser" , update it with "mix deps.update earmark_parser" or clean it with "mix deps.clean earmark_parser" You need to install erlang development environment sudo apt install erlang-dev refer to this issue Problems with Docker Versions The build may not work on old versions of Docker and Docker compose. It has been tested against: Docker version 20.10 . 17 , build 100 c701 Docker Compose version v2 . 6.0 Problems with rambo When running mix compile.rambo on Apple Silicon (an Apple M1/M2, macarm , aarch64-apple-darwin ) and encountering the following error: ** (RuntimeError) Rambo does not ship with binaries for your environment. aarch64 - apple - darwin22 . 3.0 detected Install the Rust compiler so a binary can be prepared for you . lib / mix / tasks / compile . rambo . ex : 89 : Mix.Tasks.Compile.Rambo . compile! / 0 lib / mix / tasks / compile . rambo . ex : 51 : Mix.Tasks.Compile.Rambo . run / 1 ( mix 1.14 . 2 ) lib / mix / task . ex : 421 : anonymous fn / 3 in Mix.Task . run_task / 4 ( mix 1.14 . 2 ) lib / mix / cli . ex : 84 : Mix.CLI . run_task / 2 You can resolve this error by installing the Rust compiler using Homebrew. Run the following command in your terminal: brew install rust If you have already compiled Rambo explicitly via mix compile.rambo , and you are still seeing the following error: sh : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : No such file or directory sh : line 0 : exec : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : cannot execute : No such file or directory You can try renaming deps/rambo/priv/rambo-mac to deps/rambo/priv/rambo . If neither of the approaches above work, please raise an issue. Starting from scratch If you're actively working with docker, you start experiencing issues, and you would like to start from scratch you can clean up everything and start over like this: # To remove any ignored files and reset your .env to it's example git clean -fdx && cp .env.example .env # You can skip the line below if you want to keep your database docker compose down --rmi all --volumes docker compose build --no-cache web && \\ docker compose create --force-recreate docker compose run --rm web mix ecto.migrate docker compose up","ref":"readme.html#troubleshooting"},{"type":"extras","title":"Lightning - Project Configuration Quickstart","doc":"For help getting started with your OpenFn/Lightning Project and Workflows configuration, check out the Lightning Quickstart Guidance on OpenFn/Docs.","ref":"readme.html#project-configuration-quickstart"},{"type":"extras","title":"Lightning - Contribute to this project","doc":"First, thanks for being here! You're contributing to a digital public good that will always be free and open source and aimed at serving innovative NGOs, governments, and social impact organizations the world over! You rock. ❤️ FYI, Lightning is built in Elixir , harnessing the Phoenix Framework . Currently, the only unbundled dependency is a PostgreSQL database. If you'd like to contribute to this projects, follow the steps below: Assign yourself to an issue Read through the existing issues , assign yourself to the issue you have chosen. Leave a comment on the issue to let us know you'll be working on it, and if you have any questions of clarifications that would help you get started ask them there - we will get back to you as soon as possible. If there isn't already an issue for the feature you would like to contribute, please start a discussion in our community forum . Open a pull request Clone the Lightning repository, then fork it . Run through setting up your environment and make your changes. Make sure you have written your tests and updated /CHANGELOG.md (in the 'Unreleased' section, add a short description of the changes you are making, along with a link to your issue). Open a draft pull request by clicking "Contribute > Open Pull Request" from your forked repository. Fill out the pull request template (this will be added automatically for you), then make sure to self-review your code and go through the 'Review checklist'. Don't worry about the QA checkbox, our product manager Amber will tick that once she has reviewed your PR. You can leave any notes for the reviewer in a comment. Once you're ready to submit a pull request, you can mark your draft PR as 'Ready for review' and assign @stuartc or @taylordowns2000.","ref":"readme.html#contribute-to-this-project"},{"type":"extras","title":"Lightning - Generating Documentation","doc":"You can generate the HTML and EPUB documentation locally using: mix docs and opening doc/index.html in your browser.","ref":"readme.html#generating-documentation"},{"type":"extras","title":"Lightning - Security and Standards","doc":"We use a host of common Elixir static analysis tools to help us avoid common pitfalls and make sure we keep everything clean and consistent. In addition to our test suite, you can run the following commands: mix format --check-formatted Code formatting checker, run again without the --check-formatted flag to have your code automatically changed. mix dialyzer Static analysis for type mismatches and other common warnings. See dialyxir . mix credo Static analysis for consistency, and coding standards. See Credo . mix sobelow Check for commonly known security exploits. See Sobelow . MIX_ENV=test mix coveralls Test coverage reporter. This command also runs the test suite, and can be used in place of mix test when checking everything before pushing your code. See excoveralls . For convenience there is a verify mix task that runs all of the above and defaults the MIX_ENV to test . For more guidance on security best practices for workflow automation implementations, check out OpenFn Docs: docs.openfn.org/documentation/getting-started/security","ref":"readme.html#security-and-standards"},{"type":"extras","title":"Lightning - Server Specs for Self-Hosting","doc":"For recommend server specifications for self-hosting of Lightning, check out this Community topic: community.openfn.org/t/specs-for-self-hosting-lightning/292","ref":"readme.html#server-specs-for-self-hosting"},{"type":"extras","title":"Lightning - Support","doc":"If you have any questions, feedback, or issues, please: Post on the OpenFn Community at community.openfn.org Open an issue directly on this Github Repo: github.com/OpenFn/Lightning/issues","ref":"readme.html#support"},{"type":"extras","title":"Deployment","doc":"","ref":"deployment.html"},{"type":"extras","title":"Deployment - Encryption","doc":"Lightning enforces encryption at rest for Credentials, for which an encryption key must be provided when running in production. The key is expected to be a randomized set of bytes, 32 long; and Base64 encoded when setting the environment variable. There is a mix task that can generate keys in the correct shape for use as an environment variable: mix lightning.gen_encryption_key 0bJ9w+hn4ebQrsCaWXuA9JY49fP9kbHmywGd5K7k+/s= Copy your key (NOT THIS ONE) and set it as PRIMARY_ENCRYPTION_KEY in your environment.","ref":"deployment.html#encryption"},{"type":"extras","title":"Deployment - Environment Variables","doc":"Note that for secure deployments, it's recommended to use a combination of secrets and configMaps to generate secure environment variables. ADAPTORS_PATH - where you store your locally installed adaptors DISABLE_DB_SSL - in production the use of an SSL conntection to Postgres is required by default, setting this to "true" allows unencrypted connections to the database. This is strongly discouraged in real production environment. K8S_HEADLESS_SERVICE - this environment variable is automatically set if you're running on GKE and it is used to establish an Erlang node cluster. Note that if you're not using Kubernetes, the "gossip" strategy is used for establish clusters. LISTEN_ADDRESS " - the address the web server should bind to, defaults to 127.0.0.1 to block access from other machines. LOG_LEVEL - how noisy you want the logs to be (e.g. debug , info ) MAX_RUN_DURATION - the maximum time (in milliseconds) that jobs are allowed to run (keep this below your termination_grace_period if using kubernetes) MIX_ENV - your mix env, likely prod for deployment NODE_ENV - node env, likely production for deployment ORIGINS - the allowed origins for web traffic to the backend PORT - the port your Phoenix app runs on PRIMARY_ENCRYPTION_KEY - a base64 encoded 32 character long string. See Encryption . SCHEMAS_PATH - path to the credential schemas that provide forms for different adaptors SECRET_KEY_BASE - a secret key used as a base to generate secrets for encrypting and signing data. SENTRY_DSN - if using Sentry for error monitoring, your DSN URL_HOST - the host, used for writing urls (e.g., demo.openfn.org ) URL_PORT - the port, usually 443 for production URL_SCHEME - the scheme for writing urls, (e.g., https ) Google Using your Google Cloud account, provision a new OAuth 2.0 Client with the 'Web application' type. Set the callback url to: https://<ENDPOINT DOMAIN>/authenticate/callback . Replacing ENDPOINT DOMAIN with the host name of your instance. Once the client has been created, get/download the OAuth client JSON and set the following environment variables: GOOGLE_CLIENT_ID - Which is client_id from the client details. GOOGLE_CLIENT_SECRET - client_secret from the client details.","ref":"deployment.html#environment-variables"},{"type":"extras","title":"Benchmarking","doc":"Execute the following steps to run a benchmark on Lightning: Make sure you have k6 installed locally. If you're using asdf you can run asdf install in the project root. Spin up your Lightning local instance Run the demo setup script: mix run --no-start priv/repo/demo.exs The webhookURL is already set to default to the webhook created in the demo data In another terminal (do not stop the Lightning server) run the benchmarking/script.js file using the following command k6 run benchmarking/script.js If the script exits succesfully, this means the app met the defined performance thresholds. To collect the benchmarking data in a CSV file, run the previous command with the --out filename option. k6 run --out csv=test_results.csv benchmarking/script.js See results output for other available output formats.","ref":"benchmarking.html"},{"type":"extras","title":"Provisioning","doc":"Lightning offers the ability to configure projects via the HTTP API. By providing a JSON document with the desired configuration, the project can be configured to your liking.","ref":"provisioning.html"},{"type":"extras","title":"Provisioning - Using the API","doc":"The API is available at /api/provision , and expects an application/json Content-Type. Authentication The API requires a valid auth token to be provided in the Authorization header. Example Request curl -X POST \\ -d @project.json \\ -H "Authorization: Bearer $TOKEN" \\ -H "Content-Type: application/json" \\ $ENDPOINT/api/provision","ref":"provisioning.html#using-the-api"},{"type":"extras","title":"Provisioning - Document Structure","doc":"The provisioning document is a JSON document with the project at the root. All entities must have an id field, which is a UUIDv4 string. In the case of new entities, this must be generated by the client. The API is idempotent, and the distinction between creating and updating is determined by the presence of the id field. { "id": "<<project-id>>", "name": "<<project-name>>", "workflows": [ { "id": "<<workflow-id>>", "name": "<<workflow-name>>", "jobs": [ { "id": "<<job-id>>", "name": "<<job-name>>", "body": "<<job-body>>", "adaptor": "<<adaptor-name>>", "enabled": true } // ... more jobs ], "triggers": [ { "id": "<<trigger-id>>", "name": "<<trigger-name>>", "type": "webhook" } // ... more triggers ], "edges": [ { "id": "<<edge-id>>", "source_trigger_id": "<<trigger-id>>", "target_job_id": "<<job-id>>" } // ... more edges ] } // ... more workflows ] }","ref":"provisioning.html#document-structure"},{"type":"extras","title":"Provisioning - API Behaviour","doc":"The API expects all existing entities to be provided in the provisioning document. If the document provided is out of date (e.g. a new job was added on the server), a new reference document should be fetched and the changes applied to it. Deleting Entities Entities can be deleted by setting the disabled key to true . Example: { "id": "<<project-id>>", "workflows": [ { "id": "<<workflow-id>>", "jobs": [ { "id": "<<job-id>>", "delete": true // <== delete this job } ] } ] }","ref":"provisioning.html#api-behaviour"},{"type":"extras","title":"Provisioning - Relationship with Projects as Code","doc":"The Projects as Code spec is a superset of the provisioning API. Projects as Code allows for the user to specify a key for each entity, which makes it easier to manage the project in the future. For example: name: my-project workflows: workflow-one: jobs: job-one: body: | console.log("Hello World"); adaptor: '@openfn/language-common' enabled: true triggers: trigger-one: type: webhook edges: - source_trigger: trigger-one target_job: job-one The above YAML document illustrates the use of keys being used to identify entities. Allowing the user to provision the same project to multiple environments. The API is unaware of 'keys', and expects IDs to be provided by the client. In order to convert the above YAML document to a provisioning document, the CLI uses a local state file (if available) to map the keys to UUIDs. Using the example above a state file might look like this: { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": {}, "triggers": {} } } } The state file shows that the project and workflow already exist, but the job, trigger and edge do not. In order to create these new entities, IDs will be applied them. On a successful application of the provisioning document, the state file will be updated to reflect the new IDs and entities. { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": { "job-one": { "id": "18ed71de-caf8-4822-aefc-5b19351f4016" } }, "triggers": { "trigger-one": { "id": "e0b9f357-9cf9-4206-9924-4d5674aad830" } }, "edges": [ { "id": "c239d994-6662-4637-90f8-0293c924b461", "source_trigger_id": "e0b9f357-9cf9-4206-9924-4d5674aad830", "target_job_id": "18ed71de-caf8-4822-aefc-5b19351f4016" } ] } } }","ref":"provisioning.html#relationship-with-projects-as-code"},{"type":"extras","title":"Changelog","doc":"All notable changes to this project will be documented in this file. The format is based on Keep a Changelog , and this project adheres to Semantic Versioning .","ref":"changelog.html"},{"type":"extras","title":"Changelog - [Unreleased]","doc":"Added Changed Moved to Elixir 1.15 and Erlang 26.0.2 to sort our an annoying ElixirLS issue that was slowing down our engineers. Update Debian base to use bookworm (Debian 12) for our Docker images Fixed Fix issue creating a new credential from the Job editor where the new credential was not being set on the job. #951","ref":"changelog.html#unreleased"},{"type":"extras","title":"Changelog - [v0.7.0-pre5] - 2023-07-28","doc":"Added Changed Unless otherwise specified, only show workorders with activity in last 14 days #968 Fixed","ref":"changelog.html#v0-7-0-pre5-2023-07-28"},{"type":"extras","title":"Changelog - [v0.7.0-pre4] - 2023-07-27","doc":"Added Changed Don't add cast fragments if the search_term is nil #968 Fixed","ref":"changelog.html#v0-7-0-pre4-2023-07-27"},{"type":"extras","title":"Changelog - [v0.7.0-pre3] - 2023-07-26","doc":"Added Changed Fixed Fixed an issue with newly created edges that prevented downstream jobs 977","ref":"changelog.html#v0-7-0-pre3-2023-07-26"},{"type":"extras","title":"Changelog - [v0.7.0-pre2] - 2023-07-26","doc":"Note that this is a pre-release with a couple of known bugs that are tracked in the Nodes and Edges epic . Added Added ability for a user to enable MFA on their account; using 2FA apps like Authy, Google Authenticator etc #890 Write/run sql script to convert triggers #875 Export projects as .yaml via UI #249 Changed In v0.7.0 we change the underlying workflow building and execution infrastructure to align with a standard "nodes and edges" design for directed acyclic graphs (DAGs). Make sure to run the migrations! 793 Fixed Propagate url pushState/changes to Workflow Diagram selection #944 Fix issue when deleting nodes from the workflow editor #830 Fix issue when clicking a trigger on a new/unsaved workflow #954","ref":"changelog.html#v0-7-0-pre2-2023-07-26"},{"type":"extras","title":"Changelog - [0.6.7] - 2023-07-13","doc":"Added Add feature to bulk rerun work orders from a specific step in their workflow; e.g., "rerun these 50 work orders, starting each at step 4." #906 Changed Fixed Oban exception: "value too long" when log lines are longer than 255 chars #929","ref":"changelog.html#0-6-7-2023-07-13"},{"type":"extras","title":"Changelog - [0.6.6] - 2023-06-30","doc":"Added Add public API token to the demo site setup script Check and renew OAuth credentials when running a job #646 Fixed Remove google sheets from adaptors list until supporting oauth flow #792 Remove duplicate google sheets adaptor display on credential type picklist #663 Fix demo setup script for calling from outside the app on Kubernetes deployments #917","ref":"changelog.html#0-6-6-2023-06-30"},{"type":"extras","title":"Changelog - [0.6.5] - 2023-06-22","doc":"Added Ability to rerun workorders from start by selecting one of more of them from the History page and clicking the "Rerun" button. #659 Fixed Example runs for demo incorrect #856","ref":"changelog.html#0-6-5-2023-06-22"},{"type":"extras","title":"Changelog - [0.6.3] - 2023-06-15","doc":"Fixed Prevent saving null log lines to the database, fix issue with run display #866","ref":"changelog.html#0-6-3-2023-06-15"},{"type":"extras","title":"Changelog - [0.6.2] - 2023-06-09","doc":"Fixed Fixed viewer permissions for delete workflow Fixed bug with workflow cards #859","ref":"changelog.html#0-6-2-2023-06-09"},{"type":"extras","title":"Changelog - [0.6.1] - 2023-06-08","doc":"Fixed Fixed bug with run logs #864 Correctly stagger demo runs to maintain order #856 Remove Timex use from SetupUtils in favor of DateTime to fix issue when calling it in escript.","ref":"changelog.html#0-6-1-2023-06-08"},{"type":"extras","title":"Changelog - [0.6.0]- 2023-04-12","doc":"Added Create sample runs when generating sample workflow #821 Added a provisioning api for creating and updating projects and their workflows See: PROVISIONING.md #641 Add ability for a superuser to schedule deletion, cancel deletion, and delete projects #757 Add ability for a project owner to schedule deletion, cancel deletion, and delete projects #746 Changed Ability to store run log lines as rows in a separate table #514 Fixed Incorrect project digest queries #768 ] Fix issue when purging deleted users #747 Generate a random name for Workflows when creating one via the UI. #828 Handle error when deleting a job with runs. #814","ref":"changelog.html#0-6-0-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.2]","doc":"Added Add workflow_edges table in preparation for new workflow editor implementation #794 Stamped credential_id on run directly for easier auditing of the history interface. Admins can now see which credential was used to run a run. #800 Better errors when using magic functions: "no magic yet" and "check credential" #812 Changed The delete-project function now delete all associated activities #759 Fixed","ref":"changelog.html#0-5-2"},{"type":"extras","title":"Changelog - [0.5.1] - 2023-04-12","doc":"Added Added ability to create and revoke personal API tokens #147 Add last-used at to API tokens #722 Improved "save" for job builder; users can now press Ctrl + S or ⌘ + S to save new or updated jobs job panel will not close. (Click elsewhere in the canvas or click the "Close" button to close.) #568 Add filtered search params to the history page URL #660 Changed The secret scrubber now ignores booleans 690 Fixed The secret scrubber now properly handles integer secrets from credentials 690 Updated describe-package dependency, fixing sparkles in adaptor-docs 657 Clicks on the workflow canvas were not lining up with the nodes users clicked on; they are now 733 Job panel behaves better when collapsed 774","ref":"changelog.html#0-5-1-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.0] - 2023-04-03","doc":"Added Magic functions that fetch real metadata from connected systems via credentials and suggest completions in the job builder (e.g., pressing control-space when setting the orgUnit attribute for a DHIS2 create operation will pull the actual list of orgUnits with human readable labels and fill in their orgUnit codes upon enter.) 670 A "metadata explorer" to browse actual system metadata for connected instances. 658 Resizable job builder panel for the main canvas/workflow view. 681 Changed Display timezone for cron schedule—it is always UTC. #716 Instance administrators can now configure the interval between when a project owner or user requests deletion and when these records are purged from the database. It defaults to 7, but by providing a PURGE_DELETED_AFTER_DAYS environment variable the grace period can be altered. Note that setting this variable to 0 will make automatic purging never occur but will still make "deleted" projects and users unavailable. This has been requested by certain organizations that must retain audit logs in a Lightning instance. 758 Fixed Locked CLI version to @openfn/cli@0.0.35 . #761","ref":"changelog.html#0-5-0-2023-04-03"},{"type":"extras","title":"Changelog - [0.4.8] - 2023-03-29","doc":"Added Added a test harness for monitoring critical parts of the app using Telemetry #654 Changed Set log level to info for runs. Most of the debug logging is useful for the CLI, but not for Lightning. In the future the log level will be configurable at instance > project > job level by the superuser and any project admin . Renamed license file so that automagic github icon is less confusing Fixed Broken links in failure alert email #732 Registration Submission on app.openfn.org shows internal server error in browser #686 Run the correct runtime install mix task in Dockerfile-dev #541 Users not disabled when scheduled for deletion #719","ref":"changelog.html#0-4-8-2023-03-29"},{"type":"extras","title":"Changelog - [0.4.6] - 2023-03-23","doc":"Added Implement roles and permissions across entire app #645 Fix webhook URL ( https://<<HOST_URL>>/i/cae544ab-03dc-4ccc-a09c-fb4edb255d7a ) for the OpenHIE demo workflow 448 Phoenix Storybook for improved component development Load test for webhook endpoint performance #645 Notify user via email when they're added to a project #306 Added notify user via email when their account is created #307 Changed Improved errors when decoding encryption keys for use with Cloak. #684 Allow users to run ANY job with a custom input. #629 Fixed Ensure JSON schema form inputs are in the same order as they are written in the schema #685","ref":"changelog.html#0-4-6-2023-03-23"},{"type":"extras","title":"Changelog - [0.4.4] - 2023-03-10","doc":"Added Users can receive a digest email reporting on a specified project. #638 #585 Changed Fixed","ref":"changelog.html#0-4-4-2023-03-10"},{"type":"extras","title":"Changelog - [0.4.3] - 2023-03-06","doc":"Added Tooltips on Job Builder panel #650 Changed Upgraded to Phoenix 1.7 (3945856) Fixed Issue with FailureAlerter configuration missing in prod mode.","ref":"changelog.html#0-4-3-2023-03-06"},{"type":"extras","title":"Changelog - [0.4.2] - 2023-02-24","doc":"Added A user can change their own email #247 Added a SCHEMAS_PATH environment variable to override the default folder location for credential schemas #604 Added the ability to configure Google Sheets credentials #536 Function to import a project #574 Changed Users cannot register if they have not selected the terms and conditions #531 Fixed Jobs panel slow for first open after restart #567","ref":"changelog.html#0-4-2-2023-02-24"},{"type":"extras","title":"Changelog - [0.4.0] - 2023-02-08","doc":"Added Added a Delete job button in Inspector Filter workflow runs by text/value in run logs or input body Drop "configuration" key from Run output dataclips after completion Ability to 'rerun' a run from the Run list Attempts and Runs update themselves in the Runs list Configure a project and workflow for a new registering user Run a job with a custom input Added plausible analytics Allow user to click on Webhook Trigger Node to copy webhook URL on workflow diagram Allow any user to delete a credential that they own Create any credential through a form except for OAuth Refit all diagram nodes on browser and container resize Enable distributed Erlang, allowing any number of redundant Lightning nodes to communicate with each other. Users can set up realtime alerts for a project Changed Better code-assist and intelliense in the Job Editor Updated @openfn/workflow-diagram to 0.4.0 Make plus button part of job nodes in Workflow Diagram Updated @openfn/adaptor-docs to 0.0.5 Updated @openfn/describe-package to 0.0.10 Create an follow a manual Run from the Job Inspector View all workflows in a project on the workflows index page Move @openfn/workflow-diagram into the application, the NPM module is now deprecated. Remove workflow name from first node Move the used parts of @openfn/engine into the application. [BREAKING CHANGE] Ported mix openfn.install.runtime into application, use mix lightning.install_runtime . [BREAKING CHANGE] Introduced @openfn/cli as the new runtime for Jobs Rename a workflow through the page heading Hide the dataclips tab for beta Make adaptor default to common@latest Remove jobs list page Better error handling in the docs panel Disable credential ownership transfer in dev and prod environments Add project settings page Change Workorder filters to apply to the aggregate state of the workorder and not the run directly Enable jobs by default Set log level to info Add Beta checkbox to register page User roles and permissions Fixed Don't consider disabled jobs when calculating subsequent runs Fixed overflow on Job Editor Tooltips Fixed auto-scroll when adding a new snippet in the Job Editor Fixed common operation typings in Job Editor","ref":"changelog.html#0-4-0-2023-02-08"},{"type":"extras","title":"Changelog - [0.3.1] - 2022-11-22","doc":"Fixed Fixed bug that attempted to execute HTML scripts in dataclips Fixed bug that prevented workorders from displaying in the order of their last run, descending. Remove alerts after set timeout or close","ref":"changelog.html#0-3-1-2022-11-22"},{"type":"extras","title":"Changelog - [0.3.0] - 2022-11-21","doc":"Added Add seed data for demo site Create adaptor credentials through a form Configure cron expressions through a form View runs grouped by workorders and attempts Run an existing Job with any dataclip uuid from the Job form Changed Redirect users to projects list page when they click on Admin Settings menu Move job, project, input and output Dataclips to Run table Reverse the relationship between Jobs and Triggers. Triggers now can exist on their own; setting the stage for branching and merging workflows Updated Elixir and frontend dependencies [BREAKING CHANGE] Pipeline now uses WorkOrders, previous data is not compatible. Runs, Dataclips and Attempts now all correctly use usec resolution timestamps. Upgraded LiveView to 0.18.0 Upgraded Elixir to 1.14.1 and OTP 25 Workflow Job editor now behaves like a panel Split JobLive.InspectorFormComponent into different plug-able subcomponents Ensure new jobs with cron triggers receive a default frequency Webhooks are now referenced by the trigger id instead of job id. Filter runs by status Filter runs by workflow Filter runs by date View a job run from the runs history View latest matching inputs to run a job with","ref":"changelog.html#0-3-0-2022-11-21"},{"type":"extras","title":"Changelog - [0.2.0] - 2022-09-12","doc":"Changed [BREAKING CHANGE] Add Workflow model, Jobs now belong to a Workflow This is a breaking change to the schema. Use Node.js 18, soon to be in LTS. Visualize success/fail triggers in workflow diagram. Move WorkflowDiagram related actions from DashboardLive into WorkflowLive Move WorkflowDiagram component into liveview, so that we can subscribe to channels (i.e. updating of the diagram when someone changes something). Integrate @openfn/workflow-diagram@0.0.8 and use the new Store interface for updating it. Remove component_mounted event from WorkflowDiagram hook, using a MutationObserver and a Base64 encoded JSON payload. Fixed an issue where the compiler component would try and load a 'nothing adaptor', added a condition to check an adaptor is actually selected. Removed previous Workflow CTE queries, replaced by the introduction of the Workflow model, see ( https://github.com/OpenFn/Lightning/blob/53da6883483e7d8d078783f348da327d1dd72d20/lib/lightning/workflows.ex#L111-L119 ).","ref":"changelog.html#0-2-0-2022-09-12"},{"type":"extras","title":"Changelog - [0.1.13] - 2022-08-29","doc":"Added Allow administrators to configure OIDC providers for authentication (note that this is just for authenticating, not yet for creating new accounts via OIDC) Add Monaco editor to the step/job panel Allow users to delete their own accounts. Schedule their user and credentials data for deletion when they do. Allow superusers to delete a user account. Schedule the user's credentials and user data for deletion when they do. If a user is scheduled for deletion, disable their account and prevent them from logging in. The 'User profile' and 'Credentials' page now have a sidebar menu Changed Project users now have one of the following roles: viewer, editor, admin, owner Users only have the following roles: user, superuser","ref":"changelog.html#0-1-13-2022-08-29"},{"type":"extras","title":"Changelog - [0.1.12] - 2022-08-15","doc":"Added Transfer credential ownership to another user. Create credentials via a form interface* Show "projects with access" in credentials list view. Show job in runs list and run view. Added roles and permissions to workflows and history page #645 *The form is defined by a JSON schema provided by an adaptor, in most cases: e.g., language-dhis2 provides a single schema which defines the required attributes for state.configuration , while language-common provides multiple credential schemas like "oauth" or "basic auth" which define attributes for state.configuration and which might be used by lots of different jobs.) Fixed User menu (top right) appears on top of all other components. User profile screen integrated with the rest of the liveview app.","ref":"changelog.html#0-1-12-2022-08-15"},{"type":"extras","title":"Changelog - [0.1.11] - 2022-08-05","doc":"Fixed Fixed logging in Runner when :debug log level used; note that this caused crashes in Oban","ref":"changelog.html#0-1-11-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.10] - 2022-08-05","doc":"Added Credential auditing Build/version information display for easier debugging Fixed Fixed a bug that enqueued cron-triggered jobs even when they were disabled","ref":"changelog.html#0-1-10-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.9] - 2022-07-27","doc":"Added Navigate to user profile or credentials page and log out through the user icon dropdown Create and edit dataclips Add a production tag to credentials View a dropdown of operations and their description for the language-common v2.0.0-rc2 adaptor (this pattern to be rolled out across adaptors) Changed Navigate between projects through a project picker on the navbar Fixed Run Lightning with docker Security Sensitive credential values are scrubbed from run logs All credentials are encrypted at REST","ref":"changelog.html#0-1-9-2022-07-27"},{"type":"extras","title":"Changelog - [0.1.7] - 2022-06-24","doc":"Added Run a job with a cron trigger Queue jobs via Oban/Postgres Edit jobs via the workflow canvas","ref":"changelog.html#0-1-7-2022-06-24"},{"type":"extras","title":"Changelog - [0.1.6] - 2022-06-07","doc":"Added Register, log in and log out of an account Allow superusers and admin users to create projects Allow admin users to create or disable a user’s account Allow superusers for local deployments to create users and give them access to project spaces Create and edit a job with a webhook, flow/fail or cron trigger Create and edit credentials for a job Copy a job's webhook URL View all workflows in a project visually Deploy lightning locally with Docker Enable a job to automatically process incoming requests Run a job with a webhook or flow/fail trigger View job runs along with their logs, exit code, start and end time View data clips that have initiated job runs (http requests for webhooks, run results) Changed - Removed -","ref":"changelog.html#0-1-6-2022-06-07"}] \ No newline at end of file diff --git a/readme.html b/readme.html index 29054140b0..cb286e07ea 100644 --- a/readme.html +++ b/readme.html @@ -250,7 +250,7 @@

    --mount source=lightning-postgres-data,target=/var/lib/postgresql/data \ --publish 5432:5432 \ -e POSTGRES_PASSWORD=postgres \ - postgres:14.1-alpine + postgres:15.3-alpine docker start lightning-postgres

    @@ -335,7 +335,7 @@

    Problems with Debian

    If you're getting this error on debian

    ==> earmark_parser
    -Compiling 1 file (.yrl)
    +Compiling 1 file (.yrl)
     /usr/lib/erlang/lib/parsetools-2.3.1/include/yeccpre.hrl: no such file or directory
     could not compile dependency :earmark_parser, "mix compile" failed. You can recompile this dependency with "mix deps.compile earmark_parser", update it with "mix deps.update earmark_parser" or clean it with "mix deps.clean earmark_parser"

    You need to install erlang development environment sudo apt install erlang-dev refer to this issue

    @@ -361,8 +361,8 @@

    Versions

    The build may not work on old versions of Docker and Docker lib/mix/tasks/compile.rambo.ex:89: Mix.Tasks.Compile.Rambo.compile!/0 lib/mix/tasks/compile.rambo.ex:51: Mix.Tasks.Compile.Rambo.run/1 - (mix 1.14.2) lib/mix/task.ex:421: anonymous fn/3 in Mix.Task.run_task/4 - (mix 1.14.2) lib/mix/cli.ex:84: Mix.CLI.run_task/2

    You can resolve this error by installing the Rust compiler using Homebrew. Run + (mix 1.14.2) lib/mix/task.ex:421: anonymous fn/3 in Mix.Task.run_task/4 + (mix 1.14.2) lib/mix/cli.ex:84: Mix.CLI.run_task/2

    You can resolve this error by installing the Rust compiler using Homebrew. Run the following command in your terminal: brew install rust

    If you have already compiled Rambo explicitly via mix compile.rambo, and you are still seeing the following error:

    sh: /path_to_directory/Lightning/_build/dev/lib/rambo/priv/rambo: No such file or directory
     sh: line 0: exec: /path_to_directory/Lightning/_build/dev/lib/rambo/priv/rambo: cannot execute: No such file or directory

    You can try renaming deps/rambo/priv/rambo-mac to deps/rambo/priv/rambo.

    If neither of the approaches above work, please raise an issue.

    Starting from scratch

    If you're actively working with docker, you start experiencing issues, and you diff --git a/search.html b/search.html index d9ccbd68d9..936ed64ffa 100644 --- a/search.html +++ b/search.html @@ -107,7 +107,7 @@

    - +