From 03da1b2d7c1117925d2c3353805dc6a868ad2708 Mon Sep 17 00:00:00 2001 From: stuartc Date: Fri, 1 Sep 2023 07:44:25 +0000 Subject: [PATCH] =?UTF-8?q?Deploying=20to=20gh-pages=20from=20@=20OpenFn/L?= =?UTF-8?q?ightning@8dc2676f41c0fd98a46f2b9be6c12f6bcb40fcf1=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.VersionControl.GithubClient.html | 154 +++++++++--------- Lightning.WorkOrderService.html | 8 +- Lightning.Workflows.html | 42 ++--- Lightning.epub | Bin 379399 -> 379439 bytes LightningWeb.Components.NewInputs.html | 4 +- LightningWeb.Gettext.html | 8 +- LightningWeb.OauthCredentialHelper.html | 8 +- LightningWeb.Pagination.html | 10 +- Mix.Tasks.Lightning.InstallSchemas.html | 10 +- changelog.html | 3 +- ...s-C25C3791.js => search_items-684BF541.js} | 2 +- readme.html | 6 +- search.html | 2 +- 32 files changed, 370 insertions(+), 369 deletions(-) rename dist/{search_items-C25C3791.js => search_items-684BF541.js} (93%) diff --git a/.build b/.build index f514d35e10..e7e5079c2c 100644 --- a/.build +++ b/.build @@ -205,7 +205,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-C25C3791.js +dist/search_items-684BF541.js dist/sidebar_items-068BD01D.js index.html provisioning.html diff --git a/Lightning.Accounts.html b/Lightning.Accounts.html index 4390df0f76..fd9878fba4 100644 --- a/Lightning.Accounts.html +++ b/Lightning.Accounts.html @@ -702,10 +702,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{}}
@@ -757,8 +757,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{}}
@@ -794,8 +794,8 @@

change_superuser_registration(attrs \\ %{}) Examples

-
iex> change_superuser_registration(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_superuser_registration(user)
+%Ecto.Changeset{data: %User{}}
@@ -849,8 +849,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{}}
@@ -880,8 +880,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{}}
@@ -911,8 +911,8 @@

change_user_registration(attrs \\ %{})

Examples -
iex> change_user_registration(user)
-%Ecto.Changeset{data: %User{}}
+
iex> change_user_registration(user)
+%Ecto.Changeset{data: %User{}}
@@ -1029,11 +1029,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{}}
@@ -1061,11 +1061,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{}}
@@ -1122,8 +1122,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: ...}}
@@ -1151,11 +1151,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}
@@ -1205,8 +1205,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: ...}}
@@ -1344,10 +1344,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)
@@ -1376,10 +1376,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)
@@ -1452,10 +1452,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
@@ -1484,10 +1484,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
@@ -1516,10 +1516,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
@@ -1716,8 +1716,8 @@

list_users()

Examples -
iex> list_users()
-[%User{}, ...]
+
iex> list_users()
+[%User{}, ...]
@@ -1824,11 +1824,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{}}
@@ -1866,11 +1866,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{}}
@@ -1898,11 +1898,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{}}
@@ -2024,11 +2024,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 45c2ac0871..c81ab106ce 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 7275d29522..f7acfb9e14 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 dc5c2aa071..53b09e2e46 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 1a8a15ec60..04c59808c6 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 3a53801da9..234020f545 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 4cccc51ccd..adddc5249f 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 9fefe04735..835de58f6e 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 b5522a0ebb..d3cf35bb2b 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 592be5e63f..0249112f2d 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 eb36f4598a..d9b3f2586c 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 b23313177d..55f3d6a4a3 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 4e338c6941..ae289bc35a 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 35294f36d8..9ad2c5c1be 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 3b3cb4be16..aa7f589e40 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 c3c70dfcd5..464c57df8b 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 063fdaabde..dc9b61e824 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 930bf747b2..14c980d243 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.VersionControl.GithubClient.html b/Lightning.VersionControl.GithubClient.html index 435a5f6c8e..7218372db8 100644 --- a/Lightning.VersionControl.GithubClient.html +++ b/Lightning.VersionControl.GithubClient.html @@ -446,11 +446,11 @@

delete(client, url, opts)

-

Perform a DELETE request.

See request/1 or request/2 for options definition.

delete("/users")
-delete("/users", query: [scope: "admin"])
-delete(client, "/users")
-delete(client, "/users", query: [scope: "admin"])
-delete(client, "/users", body: %{name: "Jon"})
+

Perform a DELETE request.

See request/1 or request/2 for options definition.

delete("/users")
+delete("/users", query: [scope: "admin"])
+delete(client, "/users")
+delete(client, "/users", query: [scope: "admin"])
+delete(client, "/users", body: %{name: "Jon"})
@@ -479,11 +479,11 @@

delete!(client, url, opts)

-

Perform a DELETE request.

See request!/1 or request!/2 for options definition.

delete!("/users")
-delete!("/users", query: [scope: "admin"])
-delete!(client, "/users")
-delete!(client, "/users", query: [scope: "admin"])
-delete!(client, "/users", body: %{name: "Jon"})
+

Perform a DELETE request.

See request!/1 or request!/2 for options definition.

delete!("/users")
+delete!("/users", query: [scope: "admin"])
+delete!(client, "/users")
+delete!(client, "/users", query: [scope: "admin"])
+delete!(client, "/users", body: %{name: "Jon"})
@@ -533,11 +533,11 @@

get(client, url, opts)

-

Perform a GET request.

See request/1 or request/2 for options definition.

get("/users")
-get("/users", query: [scope: "admin"])
-get(client, "/users")
-get(client, "/users", query: [scope: "admin"])
-get(client, "/users", body: %{name: "Jon"})
+

Perform a GET request.

See request/1 or request/2 for options definition.

get("/users")
+get("/users", query: [scope: "admin"])
+get(client, "/users")
+get(client, "/users", query: [scope: "admin"])
+get(client, "/users", body: %{name: "Jon"})
@@ -566,11 +566,11 @@

get!(client, url, opts)

-

Perform a GET request.

See request!/1 or request!/2 for options definition.

get!("/users")
-get!("/users", query: [scope: "admin"])
-get!(client, "/users")
-get!(client, "/users", query: [scope: "admin"])
-get!(client, "/users", body: %{name: "Jon"})
+

Perform a GET request.

See request!/1 or request!/2 for options definition.

get!("/users")
+get!("/users", query: [scope: "admin"])
+get!(client, "/users")
+get!(client, "/users", query: [scope: "admin"])
+get!(client, "/users", body: %{name: "Jon"})
@@ -620,11 +620,11 @@

head(client, url, opts)

-

Perform a HEAD request.

See request/1 or request/2 for options definition.

head("/users")
-head("/users", query: [scope: "admin"])
-head(client, "/users")
-head(client, "/users", query: [scope: "admin"])
-head(client, "/users", body: %{name: "Jon"})
+

Perform a HEAD request.

See request/1 or request/2 for options definition.

head("/users")
+head("/users", query: [scope: "admin"])
+head(client, "/users")
+head(client, "/users", query: [scope: "admin"])
+head(client, "/users", body: %{name: "Jon"})
@@ -653,11 +653,11 @@

head!(client, url, opts)

-

Perform a HEAD request.

See request!/1 or request!/2 for options definition.

head!("/users")
-head!("/users", query: [scope: "admin"])
-head!(client, "/users")
-head!(client, "/users", query: [scope: "admin"])
-head!(client, "/users", body: %{name: "Jon"})
+

Perform a HEAD request.

See request!/1 or request!/2 for options definition.

head!("/users")
+head!("/users", query: [scope: "admin"])
+head!(client, "/users")
+head!(client, "/users", query: [scope: "admin"])
+head!(client, "/users", body: %{name: "Jon"})
@@ -707,11 +707,11 @@

options(client, url, opts)

-

Perform a OPTIONS request.

See request/1 or request/2 for options definition.

options("/users")
-options("/users", query: [scope: "admin"])
-options(client, "/users")
-options(client, "/users", query: [scope: "admin"])
-options(client, "/users", body: %{name: "Jon"})
+

Perform a OPTIONS request.

See request/1 or request/2 for options definition.

options("/users")
+options("/users", query: [scope: "admin"])
+options(client, "/users")
+options(client, "/users", query: [scope: "admin"])
+options(client, "/users", body: %{name: "Jon"})
@@ -740,11 +740,11 @@

options!(client, url, opts)

-

Perform a OPTIONS request.

See request!/1 or request!/2 for options definition.

options!("/users")
-options!("/users", query: [scope: "admin"])
-options!(client, "/users")
-options!(client, "/users", query: [scope: "admin"])
-options!(client, "/users", body: %{name: "Jon"})
+

Perform a OPTIONS request.

See request!/1 or request!/2 for options definition.

options!("/users")
+options!("/users", query: [scope: "admin"])
+options!(client, "/users")
+options!(client, "/users", query: [scope: "admin"])
+options!(client, "/users", body: %{name: "Jon"})
@@ -773,10 +773,10 @@

patch(client, url, body, opts)

-

Perform a PATCH request.

See request/1 or request/2 for options definition.

patch("/users", %{name: "Jon"})
-patch("/users", %{name: "Jon"}, query: [scope: "admin"])
-patch(client, "/users", %{name: "Jon"})
-patch(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
+

Perform a PATCH request.

See request/1 or request/2 for options definition.

patch("/users", %{name: "Jon"})
+patch("/users", %{name: "Jon"}, query: [scope: "admin"])
+patch(client, "/users", %{name: "Jon"})
+patch(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
@@ -805,10 +805,10 @@

patch!(client, url, body, opts)

-

Perform a PATCH request.

See request!/1 or request!/2 for options definition.

patch!("/users", %{name: "Jon"})
-patch!("/users", %{name: "Jon"}, query: [scope: "admin"])
-patch!(client, "/users", %{name: "Jon"})
-patch!(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
+

Perform a PATCH request.

See request!/1 or request!/2 for options definition.

patch!("/users", %{name: "Jon"})
+patch!("/users", %{name: "Jon"}, query: [scope: "admin"])
+patch!(client, "/users", %{name: "Jon"})
+patch!(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
@@ -837,10 +837,10 @@

post(client, url, body, opts)

-

Perform a POST request.

See request/1 or request/2 for options definition.

post("/users", %{name: "Jon"})
-post("/users", %{name: "Jon"}, query: [scope: "admin"])
-post(client, "/users", %{name: "Jon"})
-post(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
+

Perform a POST request.

See request/1 or request/2 for options definition.

post("/users", %{name: "Jon"})
+post("/users", %{name: "Jon"}, query: [scope: "admin"])
+post(client, "/users", %{name: "Jon"})
+post(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
@@ -869,10 +869,10 @@

post!(client, url, body, opts)

-

Perform a POST request.

See request!/1 or request!/2 for options definition.

post!("/users", %{name: "Jon"})
-post!("/users", %{name: "Jon"}, query: [scope: "admin"])
-post!(client, "/users", %{name: "Jon"})
-post!(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
+

Perform a POST request.

See request!/1 or request!/2 for options definition.

post!("/users", %{name: "Jon"})
+post!("/users", %{name: "Jon"}, query: [scope: "admin"])
+post!(client, "/users", %{name: "Jon"})
+post!(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
@@ -901,10 +901,10 @@

put(client, url, body, opts)

-

Perform a PUT request.

See request/1 or request/2 for options definition.

put("/users", %{name: "Jon"})
-put("/users", %{name: "Jon"}, query: [scope: "admin"])
-put(client, "/users", %{name: "Jon"})
-put(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
+

Perform a PUT request.

See request/1 or request/2 for options definition.

put("/users", %{name: "Jon"})
+put("/users", %{name: "Jon"}, query: [scope: "admin"])
+put(client, "/users", %{name: "Jon"})
+put(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
@@ -933,10 +933,10 @@

put!(client, url, body, opts)

-

Perform a PUT request.

See request!/1 or request!/2 for options definition.

put!("/users", %{name: "Jon"})
-put!("/users", %{name: "Jon"}, query: [scope: "admin"])
-put!(client, "/users", %{name: "Jon"})
-put!(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
+

Perform a PUT request.

See request!/1 or request!/2 for options definition.

put!("/users", %{name: "Jon"})
+put!("/users", %{name: "Jon"}, query: [scope: "admin"])
+put!(client, "/users", %{name: "Jon"})
+put!(client, "/users", %{name: "Jon"}, query: [scope: "admin"])
@@ -978,11 +978,11 @@

request(client \\ %Tesla.Client{}, options) Examples

-
ExampleApi.request(method: :get, url: "/users/path")
+
ExampleApi.request(method: :get, url: "/users/path")
 
 # use shortcut methods
-ExampleApi.get("/users/1")
-ExampleApi.post(client, "/users", %{name: "Jon"})
+
ExampleApi.get("/users/1") +ExampleApi.post(client, "/users", %{name: "Jon"})
@@ -1064,11 +1064,11 @@

trace(client, url, opts)

-

Perform a TRACE request.

See request/1 or request/2 for options definition.

trace("/users")
-trace("/users", query: [scope: "admin"])
-trace(client, "/users")
-trace(client, "/users", query: [scope: "admin"])
-trace(client, "/users", body: %{name: "Jon"})
+

Perform a TRACE request.

See request/1 or request/2 for options definition.

trace("/users")
+trace("/users", query: [scope: "admin"])
+trace(client, "/users")
+trace(client, "/users", query: [scope: "admin"])
+trace(client, "/users", body: %{name: "Jon"})
@@ -1097,11 +1097,11 @@

trace!(client, url, opts)

-

Perform a TRACE request.

See request!/1 or request!/2 for options definition.

trace!("/users")
-trace!("/users", query: [scope: "admin"])
-trace!(client, "/users")
-trace!(client, "/users", query: [scope: "admin"])
-trace!(client, "/users", body: %{name: "Jon"})
+

Perform a TRACE request.

See request!/1 or request!/2 for options definition.

trace!("/users")
+trace!("/users", query: [scope: "admin"])
+trace!(client, "/users")
+trace!(client, "/users", query: [scope: "admin"])
+trace!(client, "/users", body: %{name: "Jon"})
diff --git a/Lightning.WorkOrderService.html b/Lightning.WorkOrderService.html index e3f0157d9e..d951122346 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 473e1f3fea..6cf1069c14 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 b0cd9d83da056db9864faf78b32e3c1d5933a4c1..7e345e7c3203558ba8e9ad715e8b6b7abe460b3f 100644 GIT binary patch delta 104114 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 XvRUB2+j4|}_D>LTc)<{ delta 103504 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 Examples -
<.input field={@form[:email]} type="email" />
-<.input name="my-input" errors={["oh no!"]} />

+
<.input field={@form[:email]} type="email" />
+<.input name="my-input" errors={["oh no!"]} />

attributes

diff --git a/LightningWeb.Gettext.html b/LightningWeb.Gettext.html index c13c76e6a0..3b437d7922 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 e8642aaf7d..151642ef6f 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 766a8f099a..c67b1d990e 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 b190a11205..51f8b8873d 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 800cb9148c..57170aab42 100644 --- a/changelog.html +++ b/changelog.html @@ -208,7 +208,8 @@

#958
  • Improve feedback when a Workflow name is invalid #961
  • Show that the jobs' body is invalid #957
  • Reimplement skipped CredentialLive tests -#962
  • Show GitHub installation ID and repo link to help setup/debugging for version +#962
  • Reimplement skipped WorkflowLive.IndexTest test +#964
  • Show GitHub installation ID and repo link to help setup/debugging for version control 1059
  • changed-3

    diff --git a/dist/search_items-C25C3791.js b/dist/search_items-684BF541.js similarity index 93% rename from dist/search_items-C25C3791.js rename to dist/search_items-684BF541.js index 53b5b0507d..8c41d2d309 100644 --- a/dist/search_items-C25C3791.js +++ b/dist/search_items-684BF541.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.transact/2","doc":"A small wrapper around Repo.transaction/2'. Commits the transaction if the lambda returns :ok or , rolling it back if the lambda returns :error or `. In both cases, the function returns the result of the lambda.","ref":"Lightning.Repo.html#transact/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.create_users/1","doc":"","ref":"Lightning.SetupUtils.html#create_users/1"},{"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.VersionControl","doc":"Boundary module for handling Version control activities for project, jobs workflows etc Use this module to create, modify and delete connections as well as running any associated sync jobs","ref":"Lightning.VersionControl.html"},{"type":"function","title":"Lightning.VersionControl.add_github_installation_id/2","doc":"","ref":"Lightning.VersionControl.html#add_github_installation_id/2"},{"type":"function","title":"Lightning.VersionControl.add_github_repo_and_branch/3","doc":"","ref":"Lightning.VersionControl.html#add_github_repo_and_branch/3"},{"type":"function","title":"Lightning.VersionControl.create_github_connection/1","doc":"Creates a connection between a project and a github repo","ref":"Lightning.VersionControl.html#create_github_connection/1"},{"type":"function","title":"Lightning.VersionControl.fetch_installation_repos/1","doc":"","ref":"Lightning.VersionControl.html#fetch_installation_repos/1"},{"type":"function","title":"Lightning.VersionControl.fetch_repo_branches/2","doc":"","ref":"Lightning.VersionControl.html#fetch_repo_branches/2"},{"type":"function","title":"Lightning.VersionControl.get_repo_connection/1","doc":"","ref":"Lightning.VersionControl.html#get_repo_connection/1"},{"type":"function","title":"Lightning.VersionControl.github_enabled?/0","doc":"","ref":"Lightning.VersionControl.html#github_enabled?/0"},{"type":"function","title":"Lightning.VersionControl.remove_github_connection/1","doc":"Deletes a github connection used when re installing","ref":"Lightning.VersionControl.html#remove_github_connection/1"},{"type":"function","title":"Lightning.VersionControl.run_sync/2","doc":"","ref":"Lightning.VersionControl.html#run_sync/2"},{"type":"module","title":"Lightning.VersionControl.GithubClient","doc":"Tesla github http client we use this to make any network requests to github from Lightning","ref":"Lightning.VersionControl.GithubClient.html"},{"type":"function","title":"Lightning.VersionControl.GithubClient.delete/3","doc":"Perform a DELETE request. See request/1 or request/2 for options definition. delete ( "/users" ) delete ( "/users" , query : [ scope : "admin" ] ) delete ( client , "/users" ) delete ( client , "/users" , query : [ scope : "admin" ] ) delete ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#delete/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.delete!/3","doc":"Perform a DELETE request. See request!/1 or request!/2 for options definition. delete! ( "/users" ) delete! ( "/users" , query : [ scope : "admin" ] ) delete! ( client , "/users" ) delete! ( client , "/users" , query : [ scope : "admin" ] ) delete! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#delete!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.fire_repository_dispatch/3","doc":"","ref":"Lightning.VersionControl.GithubClient.html#fire_repository_dispatch/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.get/3","doc":"Perform a GET request. See request/1 or request/2 for options definition. get ( "/users" ) get ( "/users" , query : [ scope : "admin" ] ) get ( client , "/users" ) get ( client , "/users" , query : [ scope : "admin" ] ) get ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#get/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.get!/3","doc":"Perform a GET request. See request!/1 or request!/2 for options definition. get! ( "/users" ) get! ( "/users" , query : [ scope : "admin" ] ) get! ( client , "/users" ) get! ( client , "/users" , query : [ scope : "admin" ] ) get! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#get!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.get_repo_branches/2","doc":"","ref":"Lightning.VersionControl.GithubClient.html#get_repo_branches/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.head/3","doc":"Perform a HEAD request. See request/1 or request/2 for options definition. head ( "/users" ) head ( "/users" , query : [ scope : "admin" ] ) head ( client , "/users" ) head ( client , "/users" , query : [ scope : "admin" ] ) head ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#head/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.head!/3","doc":"Perform a HEAD request. See request!/1 or request!/2 for options definition. head! ( "/users" ) head! ( "/users" , query : [ scope : "admin" ] ) head! ( client , "/users" ) head! ( client , "/users" , query : [ scope : "admin" ] ) head! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#head!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.installation_repos/1","doc":"","ref":"Lightning.VersionControl.GithubClient.html#installation_repos/1"},{"type":"function","title":"Lightning.VersionControl.GithubClient.options/3","doc":"Perform a OPTIONS request. See request/1 or request/2 for options definition. options ( "/users" ) options ( "/users" , query : [ scope : "admin" ] ) options ( client , "/users" ) options ( client , "/users" , query : [ scope : "admin" ] ) options ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#options/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.options!/3","doc":"Perform a OPTIONS request. See request!/1 or request!/2 for options definition. options! ( "/users" ) options! ( "/users" , query : [ scope : "admin" ] ) options! ( client , "/users" ) options! ( client , "/users" , query : [ scope : "admin" ] ) options! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#options!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.patch/4","doc":"Perform a PATCH request. See request/1 or request/2 for options definition. patch ( "/users" , %{ name : "Jon" } ) patch ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) patch ( client , "/users" , %{ name : "Jon" } ) patch ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#patch/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.patch!/4","doc":"Perform a PATCH request. See request!/1 or request!/2 for options definition. patch! ( "/users" , %{ name : "Jon" } ) patch! ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) patch! ( client , "/users" , %{ name : "Jon" } ) patch! ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#patch!/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.post/4","doc":"Perform a POST request. See request/1 or request/2 for options definition. post ( "/users" , %{ name : "Jon" } ) post ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) post ( client , "/users" , %{ name : "Jon" } ) post ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#post/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.post!/4","doc":"Perform a POST request. See request!/1 or request!/2 for options definition. post! ( "/users" , %{ name : "Jon" } ) post! ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) post! ( client , "/users" , %{ name : "Jon" } ) post! ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#post!/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.put/4","doc":"Perform a PUT request. See request/1 or request/2 for options definition. put ( "/users" , %{ name : "Jon" } ) put ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) put ( client , "/users" , %{ name : "Jon" } ) put ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#put/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.put!/4","doc":"Perform a PUT request. See request!/1 or request!/2 for options definition. put! ( "/users" , %{ name : "Jon" } ) put! ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) put! ( client , "/users" , %{ name : "Jon" } ) put! ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#put!/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.request/2","doc":"Perform a request. Options :method - the request method, one of [ :head , :get , :delete , :trace , :options , :post , :put , :patch ] :url - either full url e.g. " http://example.com/some/path" or just "/some/path" if using Tesla.Middleware.BaseUrl :query - a keyword list of query params, e.g. [page: 1, per_page: 100] :headers - a keyworld list of headers, e.g. [{"content-type", "text/plain"}] :body - depends on used middleware: by default it can be a binary if using e.g. JSON encoding middleware it can be a nested map if adapter supports it it can be a Stream with any of the above :opts - custom, per-request middleware or adapter options Examples ExampleApi . request ( method : :get , url : "/users/path" ) # use shortcut methods ExampleApi . get ( "/users/1" ) ExampleApi . post ( client , "/users" , %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#request/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.request!/2","doc":"Perform request and raise in case of error. This is similar to request/2 behaviour from Tesla 0.x See request/2 for list of available options.","ref":"Lightning.VersionControl.GithubClient.html#request!/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.send_sentry_error/2","doc":"","ref":"Lightning.VersionControl.GithubClient.html#send_sentry_error/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.trace/3","doc":"Perform a TRACE request. See request/1 or request/2 for options definition. trace ( "/users" ) trace ( "/users" , query : [ scope : "admin" ] ) trace ( client , "/users" ) trace ( client , "/users" , query : [ scope : "admin" ] ) trace ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#trace/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.trace!/3","doc":"Perform a TRACE request. See request!/1 or request!/2 for options definition. trace! ( "/users" ) trace! ( "/users" , query : [ scope : "admin" ] ) trace! ( client , "/users" ) trace! ( client , "/users" , query : [ scope : "admin" ] ) trace! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#trace!/3"},{"type":"type","title":"Lightning.VersionControl.GithubClient.option/0","doc":"Options that may be passed to a request function. See request/2 for detailed descriptions.","ref":"Lightning.VersionControl.GithubClient.html#t:option/0"},{"type":"module","title":"Lightning.VersionControl.GithubToken","doc":"A module that uses Joken to handle building and signing application tokens for communicating with github See: https://docs.github.com/en/apps/creating-github-apps/authenticating-with-a-github-app/generating-a-json-web-token-jwt-for-a-github-app#about-json-web-tokens-jwts","ref":"Lightning.VersionControl.GithubToken.html"},{"type":"function","title":"Lightning.VersionControl.GithubToken.build/2","doc":"","ref":"Lightning.VersionControl.GithubToken.html#build/2"},{"type":"function","title":"Lightning.VersionControl.GithubToken.generate_and_sign/2","doc":"Combines generate_claims/1 and encode_and_sign/2","ref":"Lightning.VersionControl.GithubToken.html#generate_and_sign/2"},{"type":"function","title":"Lightning.VersionControl.GithubToken.generate_and_sign!/2","doc":"Same as generate_and_sign/2 but raises if error","ref":"Lightning.VersionControl.GithubToken.html#generate_and_sign!/2"},{"type":"function","title":"Lightning.VersionControl.GithubToken.verify_and_validate/3","doc":"Combines verify/2 and validate/2","ref":"Lightning.VersionControl.GithubToken.html#verify_and_validate/3"},{"type":"function","title":"Lightning.VersionControl.GithubToken.verify_and_validate!/3","doc":"Same as verify_and_validate/2 but raises if error","ref":"Lightning.VersionControl.GithubToken.html#verify_and_validate!/3"},{"type":"module","title":"Lightning.VersionControl.ProjectRepoConnection","doc":"Ecto model for project repo connections","ref":"Lightning.VersionControl.ProjectRepoConnection.html"},{"type":"function","title":"Lightning.VersionControl.ProjectRepoConnection.changeset/2","doc":"","ref":"Lightning.VersionControl.ProjectRepoConnection.html#changeset/2"},{"type":"type","title":"Lightning.VersionControl.ProjectRepoConnection.t/0","doc":"","ref":"Lightning.VersionControl.ProjectRepoConnection.html#t:t/0"},{"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.WorkOrders.Manual","doc":"A model is used to build WorkOrders with custom input data.","ref":"Lightning.WorkOrders.Manual.html"},{"type":"function","title":"Lightning.WorkOrders.Manual.changeset/2","doc":"","ref":"Lightning.WorkOrders.Manual.html#changeset/2"},{"type":"type","title":"Lightning.WorkOrders.Manual.t/0","doc":"","ref":"Lightning.WorkOrders.Manual.html#t:t/0"},{"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":"Creates or updates a project based on a JSON payload that may or may not contain UUIDs for existing resources.","ref":"LightningWeb.API.ProvisioningController.html#create/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show/2","doc":"Returns a project "state.json", complete with UUIDs to enable idempotent project deployments and updates to existing projects via the CLI.","ref":"LightningWeb.API.ProvisioningController.html#show/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show_yaml/2","doc":"Returns a description of the project as yaml. Same as the export project to yaml button (see Downloads Controller) but made for the API.","ref":"LightningWeb.API.ProvisioningController.html#show_yaml/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.BackupCodesController","doc":"","ref":"LightningWeb.BackupCodesController.html"},{"type":"function","title":"LightningWeb.BackupCodesController.print/2","doc":"","ref":"LightningWeb.BackupCodesController.html#print/2"},{"type":"module","title":"LightningWeb.BackupCodesLive.Index","doc":"LiveView for user backup codes.","ref":"LightningWeb.BackupCodesLive.Index.html"},{"type":"function","title":"LightningWeb.BackupCodesLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.BackupCodesLive.Index.html#render/1"},{"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.Components.NewInputs","doc":"A temporary module that will serve as a place to put new inputs that conform with the newer CoreComponents conventions introduced in Phoenix 1.7.","ref":"LightningWeb.Components.NewInputs.html"},{"type":"function","title":"LightningWeb.Components.NewInputs.button/1","doc":"Renders a button. Examples < . button > Send ! < / . button > < . button phx - click = "go" class = "ml-2" > Send ! < / . button > Attributes type ( :string ) - Defaults to nil . class ( :string ) - Defaults to nil . Global attributes are accepted. Slots inner_block (required)","ref":"LightningWeb.Components.NewInputs.html#button/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.error/1","doc":"Generates a generic error message. Slots inner_block (required)","ref":"LightningWeb.Components.NewInputs.html#error/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.icon/1","doc":"Renders a Heroicon . Heroicons come in three styles – outline, solid, and mini. By default, the outline style is used, but solid and mini may be applied by using the -solid and -mini suffix. You can customize the size and colors of the icons by setting width, height, and background color classes. Icons are extracted from your assets/vendor/heroicons directory and bundled within your compiled app.css by the plugin in your assets/tailwind.config.js . Examples < . icon name = "hero-x-mark-solid" / > < . icon name = "hero-arrow-path" class = "ml-1 w-3 h-3 animate-spin" / > Attributes name ( :string ) (required) class ( :string ) - Defaults to nil .","ref":"LightningWeb.Components.NewInputs.html#icon/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.input/1","doc":"Renders an input with label and error messages. A Phoenix.HTML.FormField may be passed as argument, which is used to retrieve the input name, id, and values. Otherwise all attributes may be passed explicitly. Types This function accepts all HTML input types, considering that: You may also set type="select" to render a <select> tag type="checkbox" is used exclusively to render boolean values For live file uploads, see Phoenix.Component.live_file_input/1 See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input for more information. Examples < . input field = { @form [ :email ] } type = "email" / > < . input name = "my-input" errors = { [ "oh no!" ] } / > Attributes id ( :any ) - Defaults to nil . name ( :any ) label ( :string ) - Defaults to nil . value ( :any ) type ( :string ) - Defaults to "text" . field ( Phoenix.HTML.FormField ) - a form field struct retrieved from the form, for example: @form[:email]. errors ( :list ) - Defaults to [] . checked ( :boolean ) - the checked flag for checkbox inputs. prompt ( :string ) - the prompt for select inputs. Defaults to nil . options ( :list ) - the options to pass to Phoenix.HTML.Form.options_for_select/2. multiple ( :boolean ) - the multiple flag for select inputs. Defaults to false . Global attributes are accepted. Slots inner_block","ref":"LightningWeb.Components.NewInputs.html#input/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.label/1","doc":"Renders a label. Attributes for ( :string ) - Defaults to nil . Slots inner_block (required)","ref":"LightningWeb.Components.NewInputs.html#label/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.translate_error/1","doc":"Translates an error message using gettext.","ref":"LightningWeb.Components.NewInputs.html#translate_error/1"},{"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.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.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.MFARequired","doc":"Liveview for project access denied error messages","ref":"LightningWeb.ProjectLive.MFARequired.html"},{"type":"function","title":"LightningWeb.ProjectLive.MFARequired.on_mount/4","doc":"","ref":"LightningWeb.ProjectLive.MFARequired.html#on_mount/4"},{"type":"function","title":"LightningWeb.ProjectLive.MFARequired.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.MFARequired.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.ReAuthenticateLive.New","doc":"LiveView for re-authentication page.","ref":"LightningWeb.ReAuthenticateLive.New.html"},{"type":"function","title":"LightningWeb.ReAuthenticateLive.New.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ReAuthenticateLive.New.html#render/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.on_mount/4","doc":"Used for LiveView routes that require the user to be re-authenticated.","ref":"LightningWeb.UserAuth.html#on_mount/4"},{"type":"function","title":"LightningWeb.UserAuth.reauth_sudo_mode/2","doc":"Re-Authenticates the user by using the sudo token","ref":"LightningWeb.UserAuth.html#reauth_sudo_mode/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.require_sudo_user/2","doc":"Used for routes that require the user to be re-authenticated.","ref":"LightningWeb.UserAuth.html#require_sudo_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.VersionControlController","doc":"","ref":"LightningWeb.VersionControlController.html"},{"type":"function","title":"LightningWeb.VersionControlController.index/2","doc":"","ref":"LightningWeb.VersionControlController.html#index/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.Helpers","doc":"Helper functions for the Workflow LiveViews.","ref":"LightningWeb.WorkflowLive.Helpers.html"},{"type":"function","title":"LightningWeb.WorkflowLive.Helpers.create_user_workorder/1","doc":"","ref":"LightningWeb.WorkflowLive.Helpers.html#create_user_workorder/1"},{"type":"function","title":"LightningWeb.WorkflowLive.Helpers.save_and_run/2","doc":"","ref":"LightningWeb.WorkflowLive.Helpers.html#save_and_run/2"},{"type":"function","title":"LightningWeb.WorkflowLive.Helpers.save_workflow/1","doc":"","ref":"LightningWeb.WorkflowLive.Helpers.html#save_workflow/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 inner_block 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.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) follow_run_id ( :any ) - Defaults to nil . Slots footer column - Accepts attributes: class ( :string ) - Extra CSS classes for the column.","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_sudo_session_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_sudo_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_sudo_session_token/1","doc":"Generates a 2FA session token.","ref":"Lightning.Accounts.html#generate_sudo_session_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_user_backup_codes/1","doc":"Lists the user backup codes","ref":"Lightning.Accounts.html#list_user_backup_codes/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.regenerate_user_backup_codes/1","doc":"Regenerates the user backup codes","ref":"Lightning.Accounts.html#regenerate_user_backup_codes/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.sudo_session_token_valid?/2","doc":"Checks if the given sudo token for the user is valid","ref":"Lightning.Accounts.html#sudo_session_token_valid?/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_backup_code?/2","doc":"Validates if the given Backup code is valid.","ref":"Lightning.Accounts.html#valid_user_backup_code?/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.UserBackupCode","doc":"User backup codes schema","ref":"Lightning.Accounts.UserBackupCode.html"},{"type":"function","title":"Lightning.Accounts.UserBackupCode.changeset/2","doc":"","ref":"Lightning.Accounts.UserBackupCode.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserBackupCode.generate_backup_code/0","doc":"","ref":"Lightning.Accounts.UserBackupCode.html#generate_backup_code/0"},{"type":"type","title":"Lightning.Accounts.UserBackupCode.t/0","doc":"","ref":"Lightning.Accounts.UserBackupCode.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 Fixed Output incorrectly shows "this run failed" when the run hasn't yet finished #1048","ref":"changelog.html#unreleased"},{"type":"extras","title":"Changelog - [v0.8.2] - 2023-08-31","doc":"Added Changed Fixed Lack of differentiation between top of job editor modal and top menu was disorienting. Added shadow.","ref":"changelog.html#v0-8-2-2023-08-31"},{"type":"extras","title":"Changelog - [v0.8.1] - 2023-08-31","doc":"Added Changed Moved Save and Run button to bottom of the Job edit modal #1026 Allow a manual workorder to save the workflow before creating the workorder #959 Fixed","ref":"changelog.html#v0-8-1-2023-08-31"},{"type":"extras","title":"Changelog - [v0.8.0] - 2023-08-31","doc":"Added Introduces Github sync feature, users can now setup our github app on their instance and sync projects using our latest portability spec #970 Support Backup Codes for Multi-Factor Authentication 937 Log a warning in the console when the Editor/docs component is given latest #958 Improve feedback when a Workflow name is invalid #961 Show that the jobs' body is invalid #957 Reimplement skipped CredentialLive tests #962 Show GitHub installation ID and repo link to help setup/debugging for version control 1059 Changed Fixed Fixed issue where job names were being incorrectly hyphenated during project.yaml export #1050 Allows the demo script to set a project id during creation to help with cli deploy/pull/Github integration testing. Fixed demo project_repo_connection failing after nightly demo resets 1058 Fixed an issue where the monaco suggestion tooltip was offset from the main editor 1030","ref":"changelog.html#v0-8-0-2023-08-31"},{"type":"extras","title":"Changelog - [v0.7.3] - 2023-08-15","doc":"Added Changed Version control in project settings is now named Export your project #1015 Fixed Tooltip for credential select in Job Edit form is cut off #972 Dataclip type and state assembly notice for creating new dataclip dropped during refactor #975","ref":"changelog.html#v0-7-3-2023-08-15"},{"type":"extras","title":"Changelog - [v0.7.2] - 2023-08-10","doc":"Added Changed NodeJs security patch 1009 Fixed","ref":"changelog.html#v0-7-2-2023-08-10"},{"type":"extras","title":"Changelog - [v0.7.1] - 2023-08-04","doc":"Added Changed Fixed Fixed flickery icons on new workflow job creation.","ref":"changelog.html#v0-7-1-2023-08-04"},{"type":"extras","title":"Changelog - [v0.7.0] - 2023-08-04","doc":"Added Project owners can require MFA for their users 892 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 Change new credential modal to take up less space on the screen #931 Placeholder nodes are now purely handled client-side Fixed Fix issue creating a new credential from the Job editor where the new credential was not being set on the job. #951 Fix issue where checking a credential type radio button shows as unchecked on first click. #976 Return the pre-filled workflow names #971 Fix version reporting and external reset_demo() call via Application.spec() #1010 Fixed issue where entering a placeholder name through the form would result an in unsaveable workflow #1001 Ensure the DownloadController checks for authentication and authorisation.","ref":"changelog.html#v0-7-0-2023-08-04"},{"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.transact/2","doc":"A small wrapper around Repo.transaction/2'. Commits the transaction if the lambda returns :ok or , rolling it back if the lambda returns :error or `. In both cases, the function returns the result of the lambda.","ref":"Lightning.Repo.html#transact/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.create_users/1","doc":"","ref":"Lightning.SetupUtils.html#create_users/1"},{"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.VersionControl","doc":"Boundary module for handling Version control activities for project, jobs workflows etc Use this module to create, modify and delete connections as well as running any associated sync jobs","ref":"Lightning.VersionControl.html"},{"type":"function","title":"Lightning.VersionControl.add_github_installation_id/2","doc":"","ref":"Lightning.VersionControl.html#add_github_installation_id/2"},{"type":"function","title":"Lightning.VersionControl.add_github_repo_and_branch/3","doc":"","ref":"Lightning.VersionControl.html#add_github_repo_and_branch/3"},{"type":"function","title":"Lightning.VersionControl.create_github_connection/1","doc":"Creates a connection between a project and a github repo","ref":"Lightning.VersionControl.html#create_github_connection/1"},{"type":"function","title":"Lightning.VersionControl.fetch_installation_repos/1","doc":"","ref":"Lightning.VersionControl.html#fetch_installation_repos/1"},{"type":"function","title":"Lightning.VersionControl.fetch_repo_branches/2","doc":"","ref":"Lightning.VersionControl.html#fetch_repo_branches/2"},{"type":"function","title":"Lightning.VersionControl.get_repo_connection/1","doc":"","ref":"Lightning.VersionControl.html#get_repo_connection/1"},{"type":"function","title":"Lightning.VersionControl.github_enabled?/0","doc":"","ref":"Lightning.VersionControl.html#github_enabled?/0"},{"type":"function","title":"Lightning.VersionControl.remove_github_connection/1","doc":"Deletes a github connection used when re installing","ref":"Lightning.VersionControl.html#remove_github_connection/1"},{"type":"function","title":"Lightning.VersionControl.run_sync/2","doc":"","ref":"Lightning.VersionControl.html#run_sync/2"},{"type":"module","title":"Lightning.VersionControl.GithubClient","doc":"Tesla github http client we use this to make any network requests to github from Lightning","ref":"Lightning.VersionControl.GithubClient.html"},{"type":"function","title":"Lightning.VersionControl.GithubClient.delete/3","doc":"Perform a DELETE request. See request/1 or request/2 for options definition. delete ( "/users" ) delete ( "/users" , query : [ scope : "admin" ] ) delete ( client , "/users" ) delete ( client , "/users" , query : [ scope : "admin" ] ) delete ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#delete/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.delete!/3","doc":"Perform a DELETE request. See request!/1 or request!/2 for options definition. delete! ( "/users" ) delete! ( "/users" , query : [ scope : "admin" ] ) delete! ( client , "/users" ) delete! ( client , "/users" , query : [ scope : "admin" ] ) delete! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#delete!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.fire_repository_dispatch/3","doc":"","ref":"Lightning.VersionControl.GithubClient.html#fire_repository_dispatch/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.get/3","doc":"Perform a GET request. See request/1 or request/2 for options definition. get ( "/users" ) get ( "/users" , query : [ scope : "admin" ] ) get ( client , "/users" ) get ( client , "/users" , query : [ scope : "admin" ] ) get ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#get/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.get!/3","doc":"Perform a GET request. See request!/1 or request!/2 for options definition. get! ( "/users" ) get! ( "/users" , query : [ scope : "admin" ] ) get! ( client , "/users" ) get! ( client , "/users" , query : [ scope : "admin" ] ) get! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#get!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.get_repo_branches/2","doc":"","ref":"Lightning.VersionControl.GithubClient.html#get_repo_branches/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.head/3","doc":"Perform a HEAD request. See request/1 or request/2 for options definition. head ( "/users" ) head ( "/users" , query : [ scope : "admin" ] ) head ( client , "/users" ) head ( client , "/users" , query : [ scope : "admin" ] ) head ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#head/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.head!/3","doc":"Perform a HEAD request. See request!/1 or request!/2 for options definition. head! ( "/users" ) head! ( "/users" , query : [ scope : "admin" ] ) head! ( client , "/users" ) head! ( client , "/users" , query : [ scope : "admin" ] ) head! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#head!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.installation_repos/1","doc":"","ref":"Lightning.VersionControl.GithubClient.html#installation_repos/1"},{"type":"function","title":"Lightning.VersionControl.GithubClient.options/3","doc":"Perform a OPTIONS request. See request/1 or request/2 for options definition. options ( "/users" ) options ( "/users" , query : [ scope : "admin" ] ) options ( client , "/users" ) options ( client , "/users" , query : [ scope : "admin" ] ) options ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#options/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.options!/3","doc":"Perform a OPTIONS request. See request!/1 or request!/2 for options definition. options! ( "/users" ) options! ( "/users" , query : [ scope : "admin" ] ) options! ( client , "/users" ) options! ( client , "/users" , query : [ scope : "admin" ] ) options! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#options!/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.patch/4","doc":"Perform a PATCH request. See request/1 or request/2 for options definition. patch ( "/users" , %{ name : "Jon" } ) patch ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) patch ( client , "/users" , %{ name : "Jon" } ) patch ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#patch/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.patch!/4","doc":"Perform a PATCH request. See request!/1 or request!/2 for options definition. patch! ( "/users" , %{ name : "Jon" } ) patch! ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) patch! ( client , "/users" , %{ name : "Jon" } ) patch! ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#patch!/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.post/4","doc":"Perform a POST request. See request/1 or request/2 for options definition. post ( "/users" , %{ name : "Jon" } ) post ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) post ( client , "/users" , %{ name : "Jon" } ) post ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#post/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.post!/4","doc":"Perform a POST request. See request!/1 or request!/2 for options definition. post! ( "/users" , %{ name : "Jon" } ) post! ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) post! ( client , "/users" , %{ name : "Jon" } ) post! ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#post!/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.put/4","doc":"Perform a PUT request. See request/1 or request/2 for options definition. put ( "/users" , %{ name : "Jon" } ) put ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) put ( client , "/users" , %{ name : "Jon" } ) put ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#put/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.put!/4","doc":"Perform a PUT request. See request!/1 or request!/2 for options definition. put! ( "/users" , %{ name : "Jon" } ) put! ( "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] ) put! ( client , "/users" , %{ name : "Jon" } ) put! ( client , "/users" , %{ name : "Jon" } , query : [ scope : "admin" ] )","ref":"Lightning.VersionControl.GithubClient.html#put!/4"},{"type":"function","title":"Lightning.VersionControl.GithubClient.request/2","doc":"Perform a request. Options :method - the request method, one of [ :head , :get , :delete , :trace , :options , :post , :put , :patch ] :url - either full url e.g. " http://example.com/some/path" or just "/some/path" if using Tesla.Middleware.BaseUrl :query - a keyword list of query params, e.g. [page: 1, per_page: 100] :headers - a keyworld list of headers, e.g. [{"content-type", "text/plain"}] :body - depends on used middleware: by default it can be a binary if using e.g. JSON encoding middleware it can be a nested map if adapter supports it it can be a Stream with any of the above :opts - custom, per-request middleware or adapter options Examples ExampleApi . request ( method : :get , url : "/users/path" ) # use shortcut methods ExampleApi . get ( "/users/1" ) ExampleApi . post ( client , "/users" , %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#request/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.request!/2","doc":"Perform request and raise in case of error. This is similar to request/2 behaviour from Tesla 0.x See request/2 for list of available options.","ref":"Lightning.VersionControl.GithubClient.html#request!/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.send_sentry_error/2","doc":"","ref":"Lightning.VersionControl.GithubClient.html#send_sentry_error/2"},{"type":"function","title":"Lightning.VersionControl.GithubClient.trace/3","doc":"Perform a TRACE request. See request/1 or request/2 for options definition. trace ( "/users" ) trace ( "/users" , query : [ scope : "admin" ] ) trace ( client , "/users" ) trace ( client , "/users" , query : [ scope : "admin" ] ) trace ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#trace/3"},{"type":"function","title":"Lightning.VersionControl.GithubClient.trace!/3","doc":"Perform a TRACE request. See request!/1 or request!/2 for options definition. trace! ( "/users" ) trace! ( "/users" , query : [ scope : "admin" ] ) trace! ( client , "/users" ) trace! ( client , "/users" , query : [ scope : "admin" ] ) trace! ( client , "/users" , body : %{ name : "Jon" } )","ref":"Lightning.VersionControl.GithubClient.html#trace!/3"},{"type":"type","title":"Lightning.VersionControl.GithubClient.option/0","doc":"Options that may be passed to a request function. See request/2 for detailed descriptions.","ref":"Lightning.VersionControl.GithubClient.html#t:option/0"},{"type":"module","title":"Lightning.VersionControl.GithubToken","doc":"A module that uses Joken to handle building and signing application tokens for communicating with github See: https://docs.github.com/en/apps/creating-github-apps/authenticating-with-a-github-app/generating-a-json-web-token-jwt-for-a-github-app#about-json-web-tokens-jwts","ref":"Lightning.VersionControl.GithubToken.html"},{"type":"function","title":"Lightning.VersionControl.GithubToken.build/2","doc":"","ref":"Lightning.VersionControl.GithubToken.html#build/2"},{"type":"function","title":"Lightning.VersionControl.GithubToken.generate_and_sign/2","doc":"Combines generate_claims/1 and encode_and_sign/2","ref":"Lightning.VersionControl.GithubToken.html#generate_and_sign/2"},{"type":"function","title":"Lightning.VersionControl.GithubToken.generate_and_sign!/2","doc":"Same as generate_and_sign/2 but raises if error","ref":"Lightning.VersionControl.GithubToken.html#generate_and_sign!/2"},{"type":"function","title":"Lightning.VersionControl.GithubToken.verify_and_validate/3","doc":"Combines verify/2 and validate/2","ref":"Lightning.VersionControl.GithubToken.html#verify_and_validate/3"},{"type":"function","title":"Lightning.VersionControl.GithubToken.verify_and_validate!/3","doc":"Same as verify_and_validate/2 but raises if error","ref":"Lightning.VersionControl.GithubToken.html#verify_and_validate!/3"},{"type":"module","title":"Lightning.VersionControl.ProjectRepoConnection","doc":"Ecto model for project repo connections","ref":"Lightning.VersionControl.ProjectRepoConnection.html"},{"type":"function","title":"Lightning.VersionControl.ProjectRepoConnection.changeset/2","doc":"","ref":"Lightning.VersionControl.ProjectRepoConnection.html#changeset/2"},{"type":"type","title":"Lightning.VersionControl.ProjectRepoConnection.t/0","doc":"","ref":"Lightning.VersionControl.ProjectRepoConnection.html#t:t/0"},{"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.WorkOrders.Manual","doc":"A model is used to build WorkOrders with custom input data.","ref":"Lightning.WorkOrders.Manual.html"},{"type":"function","title":"Lightning.WorkOrders.Manual.changeset/2","doc":"","ref":"Lightning.WorkOrders.Manual.html#changeset/2"},{"type":"type","title":"Lightning.WorkOrders.Manual.t/0","doc":"","ref":"Lightning.WorkOrders.Manual.html#t:t/0"},{"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":"Creates or updates a project based on a JSON payload that may or may not contain UUIDs for existing resources.","ref":"LightningWeb.API.ProvisioningController.html#create/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show/2","doc":"Returns a project "state.json", complete with UUIDs to enable idempotent project deployments and updates to existing projects via the CLI.","ref":"LightningWeb.API.ProvisioningController.html#show/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show_yaml/2","doc":"Returns a description of the project as yaml. Same as the export project to yaml button (see Downloads Controller) but made for the API.","ref":"LightningWeb.API.ProvisioningController.html#show_yaml/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.BackupCodesController","doc":"","ref":"LightningWeb.BackupCodesController.html"},{"type":"function","title":"LightningWeb.BackupCodesController.print/2","doc":"","ref":"LightningWeb.BackupCodesController.html#print/2"},{"type":"module","title":"LightningWeb.BackupCodesLive.Index","doc":"LiveView for user backup codes.","ref":"LightningWeb.BackupCodesLive.Index.html"},{"type":"function","title":"LightningWeb.BackupCodesLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.BackupCodesLive.Index.html#render/1"},{"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.Components.NewInputs","doc":"A temporary module that will serve as a place to put new inputs that conform with the newer CoreComponents conventions introduced in Phoenix 1.7.","ref":"LightningWeb.Components.NewInputs.html"},{"type":"function","title":"LightningWeb.Components.NewInputs.button/1","doc":"Renders a button. Examples < . button > Send ! < / . button > < . button phx - click = "go" class = "ml-2" > Send ! < / . button > Attributes type ( :string ) - Defaults to nil . class ( :string ) - Defaults to nil . Global attributes are accepted. Slots inner_block (required)","ref":"LightningWeb.Components.NewInputs.html#button/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.error/1","doc":"Generates a generic error message. Slots inner_block (required)","ref":"LightningWeb.Components.NewInputs.html#error/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.icon/1","doc":"Renders a Heroicon . Heroicons come in three styles – outline, solid, and mini. By default, the outline style is used, but solid and mini may be applied by using the -solid and -mini suffix. You can customize the size and colors of the icons by setting width, height, and background color classes. Icons are extracted from your assets/vendor/heroicons directory and bundled within your compiled app.css by the plugin in your assets/tailwind.config.js . Examples < . icon name = "hero-x-mark-solid" / > < . icon name = "hero-arrow-path" class = "ml-1 w-3 h-3 animate-spin" / > Attributes name ( :string ) (required) class ( :string ) - Defaults to nil .","ref":"LightningWeb.Components.NewInputs.html#icon/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.input/1","doc":"Renders an input with label and error messages. A Phoenix.HTML.FormField may be passed as argument, which is used to retrieve the input name, id, and values. Otherwise all attributes may be passed explicitly. Types This function accepts all HTML input types, considering that: You may also set type="select" to render a <select> tag type="checkbox" is used exclusively to render boolean values For live file uploads, see Phoenix.Component.live_file_input/1 See https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input for more information. Examples < . input field = { @form [ :email ] } type = "email" / > < . input name = "my-input" errors = { [ "oh no!" ] } / > Attributes id ( :any ) - Defaults to nil . name ( :any ) label ( :string ) - Defaults to nil . value ( :any ) type ( :string ) - Defaults to "text" . field ( Phoenix.HTML.FormField ) - a form field struct retrieved from the form, for example: @form[:email]. errors ( :list ) - Defaults to [] . checked ( :boolean ) - the checked flag for checkbox inputs. prompt ( :string ) - the prompt for select inputs. Defaults to nil . options ( :list ) - the options to pass to Phoenix.HTML.Form.options_for_select/2. multiple ( :boolean ) - the multiple flag for select inputs. Defaults to false . Global attributes are accepted. Slots inner_block","ref":"LightningWeb.Components.NewInputs.html#input/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.label/1","doc":"Renders a label. Attributes for ( :string ) - Defaults to nil . Slots inner_block (required)","ref":"LightningWeb.Components.NewInputs.html#label/1"},{"type":"function","title":"LightningWeb.Components.NewInputs.translate_error/1","doc":"Translates an error message using gettext.","ref":"LightningWeb.Components.NewInputs.html#translate_error/1"},{"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.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.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.MFARequired","doc":"Liveview for project access denied error messages","ref":"LightningWeb.ProjectLive.MFARequired.html"},{"type":"function","title":"LightningWeb.ProjectLive.MFARequired.on_mount/4","doc":"","ref":"LightningWeb.ProjectLive.MFARequired.html#on_mount/4"},{"type":"function","title":"LightningWeb.ProjectLive.MFARequired.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.MFARequired.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.ReAuthenticateLive.New","doc":"LiveView for re-authentication page.","ref":"LightningWeb.ReAuthenticateLive.New.html"},{"type":"function","title":"LightningWeb.ReAuthenticateLive.New.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ReAuthenticateLive.New.html#render/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.on_mount/4","doc":"Used for LiveView routes that require the user to be re-authenticated.","ref":"LightningWeb.UserAuth.html#on_mount/4"},{"type":"function","title":"LightningWeb.UserAuth.reauth_sudo_mode/2","doc":"Re-Authenticates the user by using the sudo token","ref":"LightningWeb.UserAuth.html#reauth_sudo_mode/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.require_sudo_user/2","doc":"Used for routes that require the user to be re-authenticated.","ref":"LightningWeb.UserAuth.html#require_sudo_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.VersionControlController","doc":"","ref":"LightningWeb.VersionControlController.html"},{"type":"function","title":"LightningWeb.VersionControlController.index/2","doc":"","ref":"LightningWeb.VersionControlController.html#index/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.Helpers","doc":"Helper functions for the Workflow LiveViews.","ref":"LightningWeb.WorkflowLive.Helpers.html"},{"type":"function","title":"LightningWeb.WorkflowLive.Helpers.create_user_workorder/1","doc":"","ref":"LightningWeb.WorkflowLive.Helpers.html#create_user_workorder/1"},{"type":"function","title":"LightningWeb.WorkflowLive.Helpers.save_and_run/2","doc":"","ref":"LightningWeb.WorkflowLive.Helpers.html#save_and_run/2"},{"type":"function","title":"LightningWeb.WorkflowLive.Helpers.save_workflow/1","doc":"","ref":"LightningWeb.WorkflowLive.Helpers.html#save_workflow/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 inner_block 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.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) follow_run_id ( :any ) - Defaults to nil . Slots footer column - Accepts attributes: class ( :string ) - Extra CSS classes for the column.","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_sudo_session_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_sudo_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_sudo_session_token/1","doc":"Generates a 2FA session token.","ref":"Lightning.Accounts.html#generate_sudo_session_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_user_backup_codes/1","doc":"Lists the user backup codes","ref":"Lightning.Accounts.html#list_user_backup_codes/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.regenerate_user_backup_codes/1","doc":"Regenerates the user backup codes","ref":"Lightning.Accounts.html#regenerate_user_backup_codes/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.sudo_session_token_valid?/2","doc":"Checks if the given sudo token for the user is valid","ref":"Lightning.Accounts.html#sudo_session_token_valid?/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_backup_code?/2","doc":"Validates if the given Backup code is valid.","ref":"Lightning.Accounts.html#valid_user_backup_code?/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.UserBackupCode","doc":"User backup codes schema","ref":"Lightning.Accounts.UserBackupCode.html"},{"type":"function","title":"Lightning.Accounts.UserBackupCode.changeset/2","doc":"","ref":"Lightning.Accounts.UserBackupCode.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserBackupCode.generate_backup_code/0","doc":"","ref":"Lightning.Accounts.UserBackupCode.html#generate_backup_code/0"},{"type":"type","title":"Lightning.Accounts.UserBackupCode.t/0","doc":"","ref":"Lightning.Accounts.UserBackupCode.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 Fixed Output incorrectly shows "this run failed" when the run hasn't yet finished #1048","ref":"changelog.html#unreleased"},{"type":"extras","title":"Changelog - [v0.8.2] - 2023-08-31","doc":"Added Changed Fixed Lack of differentiation between top of job editor modal and top menu was disorienting. Added shadow.","ref":"changelog.html#v0-8-2-2023-08-31"},{"type":"extras","title":"Changelog - [v0.8.1] - 2023-08-31","doc":"Added Changed Moved Save and Run button to bottom of the Job edit modal #1026 Allow a manual workorder to save the workflow before creating the workorder #959 Fixed","ref":"changelog.html#v0-8-1-2023-08-31"},{"type":"extras","title":"Changelog - [v0.8.0] - 2023-08-31","doc":"Added Introduces Github sync feature, users can now setup our github app on their instance and sync projects using our latest portability spec #970 Support Backup Codes for Multi-Factor Authentication 937 Log a warning in the console when the Editor/docs component is given latest #958 Improve feedback when a Workflow name is invalid #961 Show that the jobs' body is invalid #957 Reimplement skipped CredentialLive tests #962 Reimplement skipped WorkflowLive.IndexTest test #964 Show GitHub installation ID and repo link to help setup/debugging for version control 1059 Changed Fixed Fixed issue where job names were being incorrectly hyphenated during project.yaml export #1050 Allows the demo script to set a project id during creation to help with cli deploy/pull/Github integration testing. Fixed demo project_repo_connection failing after nightly demo resets 1058 Fixed an issue where the monaco suggestion tooltip was offset from the main editor 1030","ref":"changelog.html#v0-8-0-2023-08-31"},{"type":"extras","title":"Changelog - [v0.7.3] - 2023-08-15","doc":"Added Changed Version control in project settings is now named Export your project #1015 Fixed Tooltip for credential select in Job Edit form is cut off #972 Dataclip type and state assembly notice for creating new dataclip dropped during refactor #975","ref":"changelog.html#v0-7-3-2023-08-15"},{"type":"extras","title":"Changelog - [v0.7.2] - 2023-08-10","doc":"Added Changed NodeJs security patch 1009 Fixed","ref":"changelog.html#v0-7-2-2023-08-10"},{"type":"extras","title":"Changelog - [v0.7.1] - 2023-08-04","doc":"Added Changed Fixed Fixed flickery icons on new workflow job creation.","ref":"changelog.html#v0-7-1-2023-08-04"},{"type":"extras","title":"Changelog - [v0.7.0] - 2023-08-04","doc":"Added Project owners can require MFA for their users 892 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 Change new credential modal to take up less space on the screen #931 Placeholder nodes are now purely handled client-side Fixed Fix issue creating a new credential from the Job editor where the new credential was not being set on the job. #951 Fix issue where checking a credential type radio button shows as unchecked on first click. #976 Return the pre-filled workflow names #971 Fix version reporting and external reset_demo() call via Application.spec() #1010 Fixed issue where entering a placeholder name through the form would result an in unsaveable workflow #1001 Ensure the DownloadController checks for authentication and authorisation.","ref":"changelog.html#v0-7-0-2023-08-04"},{"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 ad35d3c1ab..5d0f820747 100644 --- a/readme.html +++ b/readme.html @@ -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 ffdf3de367..c42b5009cd 100644 --- a/search.html +++ b/search.html @@ -107,7 +107,7 @@

    - +