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

apply_user_email(user, password, attrs)

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

change_scheduled_deletion(user, attrs \\ %{ Examples

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

change_superuser_registration(attrs \\ %{}) Examples

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

change_user_email(user, attrs \\ %{})

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

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

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

change_user_registration(attrs \\ %{})

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

delete_token(token)

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

delete_user(user)

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

deliver_update_email_instructions(user, cur Examples

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

deliver_user_confirmation_instructions(user Examples

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

deliver_user_reset_password_instructions(us Examples

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

get_token!(id)

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

get_user!(id)

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

get_user_by_email(email)

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

get_user_by_email_and_password(email, passw Examples

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

get_user_by_reset_password_token(token)

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

list_users()

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

register_superuser(attrs)

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

register_user(attrs)

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

reset_user_password(user, attrs)

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

update_user_password(user, password, attrs) Examples

-
iex> update_user_password(user, "valid password", %{password: ...})
-{:ok, %User{}}
+
iex> update_user_password(user, "valid password", %{password: ...})
+{:ok, %User{}}
 
-iex> update_user_password(user, "invalid password", %{password: ...})
-{:error, %Ecto.Changeset{}}
+
iex> update_user_password(user, "invalid password", %{password: ...}) +{:error, %Ecto.Changeset{}}
diff --git a/Lightning.AdaptorRegistry.Npm.html b/Lightning.AdaptorRegistry.Npm.html index c023b2f710..0c5b45f8e7 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 8f2dc13f85..0ec1176a57 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 2ba2cd2595..19c2c2bec9 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 cc7f104f88..9d9cdde354 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 0eaaf9517a..ec846f960d 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 d99b297fda..9f0ad34b7a 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 a11e400b04..632e2b6ab8 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 66e933b5c4..6649790e4b 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 5487bda169..51fb282da4 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 a0d6145ae1..0a11797f88 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 b7a68b53d3..2d810c6c3a 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 bf883bd376..efe8998feb 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 ac16dc344f..7fb53316a8 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 2f18c2b4e4..e40c321de7 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 125e28cb6b..bcb790eb01 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 0515ea6437..4ac735f18e 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 9673ad0d6c..0e6d55b6e1 100644 --- a/Lightning.Validators.html +++ b/Lightning.Validators.html @@ -192,10 +192,10 @@

validate_exclusive(changeset, fields, messa

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

Example:

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

diff --git a/Lightning.WorkOrderService.html b/Lightning.WorkOrderService.html index 5c0ab9edcb..76f7368c33 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 cb0a94339f..4e1b2d5270 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 3c89cb14b62a9a9d03aa4c5f962addefc3a2e02a..42de37a63a0d73d3cc803a80741e9d3c6714c815 100644 GIT binary patch delta 94272 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

H_IG= zzL}gmVuWy;JN=60B0*C!&xwJZRjO<-a`#>(&^`Lgq%b7MEfFsHs}16sEbP_;Qsp*N zvKWdhzzhHmPnuB?5nDFb2VEQ34l+F;F9!FU$XwJZp4w%Cmon`-IrXH#Xo<>c*Fus; z7`nrjRA*ahO{>rV)qKnffo9Y=pZ#E0ca!3!dNOn$`7kT=G8PV(+{Tb@8Fv$xE5RO9 zfoXjJ;W&MKAfX>XO*juFi3*xEou*5wNqyu|4FJ$ywekpd1b-}>Rqk3Duy5l2A+)Y8 zzl=dI%B#9lhCoIPqbB_M-nj{kYgh`y8FOo`!8XgMw-Lu)kkgBogWqYe0r8$W$fpT@rbBDN+x>5 zE{~2=A~ahb(j~pD?DBLj zI~S`9uVRATgtRPP%UkH0ANXXnY$ugk=v>!mlQse`dXU!i0g4j4@`3P8M?F3gGc*W4 z@49DhtdyBDuJlzZm+&kC_3rNJ3dV^i?P}e+mLIPH!B5_8kBp}N#a3*%N=C;wOG{Vh zDd9%^H5&|a?w?<&E}x$lU+dQwJNm2bBt*6L>stcWrjtIe78au`VdS?9?WR{nG)Vfd zr~%fmIFxh4GXNp=wVAyypj z-62gt425tk3Uv=Q%a&7+IxMk3LEPq}JscURRY}704Rj>AZ0L-ZR0zI=_;c_-p!?97 z?&qK;LGBV5IprL)R46v1(+Q1*so897#nh)TGdWp# zuGoY+n_b^zZ-us~yH1l)v!xl=r3cVSH%tLjKT*#y)@KA}{qQ`P`&tAezSf0r2h__> z+>gRcgI6fOdR?YYSXZ)csR>xV*!02|jW;7Moyky9jj#Hx-x9y@_NLJ2iG5QreTkrp zvuMZv!`nk7ZYur{Z(l-!uA{rhrAlkKt!afiMV^7}P)9tIDAZYnn!m81dQt_r*&s@n zCQ>~sx8+W#+YGO63TFoyEoU$Pw0^%!p3`rQ9NDTheB5CuU9$c~Z_H<6;hlF=T&z%> z-O}7LJ3DU}(31@AZ}V#Lp$b8Fd?#q&XZuIjBoQplC|v}ri9L*MiT8yT;`!cL4%12A zV0}{AKxB72^Yg(E%kMWIPwE8VX@g*JK>duI#nu+$(kol0HflGm{wYlEXEe5FChdjn z(L7?AuXqZ979`&QRdP@qow)mv_dqR-i+2Y@b(5cJw?R(LPNvHv1TUYRJwbL{_Sxc5 zVJr4^Tcd%Yu8)ff<{4-fugWFBqqXDK#p55> zXTPmiHL+FcOAWxWO9htNcAYdG$E7C4nMjw69?|SwPXS;KDAJ==EMW!{|1eJol2xWsAKa(cn|O!VAJrp&AR>s0}$Utl-i9Pv#mGJ_0dG|y;LpWAA6hsXO9rV4I)cksz-*&)P z?jqrInu8y*@=yu`JvqITYGjk7jo@={yf*{lHPDCof2;ONbi(*h#wo$Qd+qP}nD^A6>ZB%UAwry5iNlw1K zzw;m5tgE@k^URAm#^|m0_BMhsOF>Ec7Y0=DAOy>cKcZ8jwr$FKpWCurFW_X8O;bG~DHuah zHD3Q3e}|Xv-Ny7YzwA zYm!P~LwFjq`14%7T|yP(1Zv|S{Hl*ze2?wzIGw{Wf&!f4z@)z+l{LI=&dW(CCn0^S zuhB}w#DtMI8cp|WcSWF;IApv}!xFD3XkIlTNh}`7REgk`&+W7>va}`rcZZt~mD8%(5HWw4Ow|xmY_ujtm(*X35-3_1%KiKz8W>RzG#9mGFA$LOSD zt*6YQIgal`+31~x^MMunLY`jprYkTTeHUcOjt5J9o%K6g*GFP4OiO+?d9E76*MZ+{ zv(QcK3oG&HuDjLawd!CjClDlPMcKSr4qNV-uK2^wrZMcThf}$<2gL!;ctrT=*VaPl zs_isXN6j^$#JB%Yhd?qd^~(<3y6L4#-A7&uI_*PK3frwMB&^dRcU{lCI)>NEz<4Cm9^)7wtu~KtbHzSo$gKL21A+gW znCf2MxtWeBg=SBew#g`H>bPCCzjU{ClH#s+g(nfvivBV_tWL>@PT_t80?Y%hOqN

w{HV;{#q_WkCni)Mx@1>QKMP>Qs&~NwFt(FEGji&RQ~!>hCH_ zLXDRBFpg#oQeBVp1y-Z^gHvFQiTd7E7?#?WQ*{&VhhK%U)X7stw)ptaZFAg4&Ck;L za3f>@nHPH3m|&NJq8i_Lp=1g9T$D6F+ZyIbl+g!y@KD!^xj9-i(;U?CewP2wB51cVNt}8zCd!pnS2MCBfm52lcC6!(p z90PEr`M(eIzTc;DAjz2A()%9ubwsiO)!*4>eg44U&{{?$HOQ24Dwp5TGXi}Q&1MsB zwv;>ybiz8~ZJgW=Zx5G10==%T&ew}r>bMFX7>IyZphzBLCurs32q>Y4*UjAA7abA{ z4@buUbg>`U`I4cUOd-L7vpjJAAAsLvWJMI_)NU3!n_AL^2M5lBFi^tPh(%x{`Ae`) z1q{EGXh;;5H7d02VSy{q(NwXZYAq9}#5DK6hOa$>(T*;8`x2O;cxI~tJB+_J`QhUh ztkTUn$d%I&#lB(D0*H&&Eo?MD1!2I8eyvlTl-|(W*X(4y z0IYpL4ukvqdIdtPssgcMZv>MJ%av7adkbZ5B#qCjN`(au%oKaJL$j0x&t^HD_Too> zX;5>xFCAOPrEk3wyo0@Jox z>9T^&rIa^pZOJ6WRm1nZYSkb^*^dkFi)drcjm{XUUpfU#>~-gZ4OOA%r^n}aL{_n1 zOmGX0A(MJ#%rwsEjBrD%mkKf1@=Ev`Yq_e<+uklC_nS@7XR?7<0vUXC`T4*R8xqVI zv0Z`cZek}t_0P|x!~ooa9$B}<*^r`qZ7z9+r9eEhj$2)nvuZ9bg{h3gb6>azo9IL*_!FvtvQ zCMCDYZy1?|2hmt^GFTt*gs#1Un%H8C{#&2-9M(uM857AC1qXO+rY^M1#9K6-nQUri z5{_L|mKRdhn*RIi>wa7;k3LfqR``RxVnzvJ!$x&M zT&lhx;8tGhco&pjbn*NF?GG_NioBQCPbOA{gzToZSd)nbDM~Ib%*M7nWxeY3{Mg^n z$daqyrgt64`WV2MltyU$X_X4sAAK!LaB4}AM5wgGv}6v@FCg7Csm+TpGTn)>Q+K|^ z&*(f!pnF0ubwl1AWkjO>Nd|rE2xkcD$R-JbgL-J6a=0Ed!2I zeIGzM)@-+kHI)cwyqqA+2@Gn-!ByrTZ2mr1|_vGrEg+i_^L>YA!O9O_<92nH&iiq*qX zi-fALuBtu6^_v|g3#<&W)dwzs%oG^6(Dj@yPOA%D7FvNxlBvs@XTpUciN;T6aV8YA zakYX$n|hrOOP7aHZB-hK9%e?s&x?%<2Vn%`#26soFf_L<<;CIoSu8JYCDmO_68SD0 zn%Mp9jA^&Nfj2GXB;lct0K7jGdiui~hdW?T%!lmn!w(m|>!4}1Fa9&IMb75qIy_#0 zXav6V+PgOo&tE$q9o9Y)!_(SQqgf_~tabRGtbv(Wll+n2pfUEHp8r?6S2$ZL3Hsj4 z;qm_g#{6d^|DJNEEcBn6r7=*o$ zlb`ssDe{En-#o`~Y!m|78Q?#BBdHf_C&Mh*3|rF94sBOno13ApVIcuK0NoIhqhGMm zrUu-rZ1(YPTGhmI^elPeg8j0E>n=7Ngj&JlcKVb^a)z^P(^_aIF0B%8vzs01Kk086 zevrG%9ctWi^;iwt!QqC*-dS5QC+_sV1%`;x$pc^k6LS9TKlU7-rLOK1`b78 zbtt_SA*R;yCM=VPQ*)A1TPHAh5-Q(HfdjaLgV|B?92l?=J#BBKyrbPX3Xx9>Kz}Pc zxxPlAj`T+y3&E5mi4s^r{5g6%Y87%;O{W11xsF~Ac@H_)Ph%FmxN|-|v-l`f!rLqF z5p12_1q6Pn7L}~vHx|9#VdW`Y$`}9*;p75MgLeA|Olw8hr5e=>*a~1O z&&#QrB@m_y|sYrVyO{b<52?gu?Pjrb@E@jlFimayQE^a32{MGNEt{o?)DItuFis@B`yJTzjw z^~5|^zI=XU(H+^g&L-T+3V8cq2open(hqxp4>$Ou4CqZhFt=%EAg!(VUj5c}0FA8` z*v?5lFgo_wmwCB3F^NS!Q?z!I(uxwNU8k-w#(Nf_$H+fdh`%Ut{*L4$g6F9!B#XV6$oPP-=E?$o2 z(AIw;?StLOU+A$foLRUrr^}p#aWoXRtIwNCu3@>9OS$Gts}A%Cq>=(ar}Ly$&#lAa zvZs?OX2WI5_uoQg#_H~6Ems9nlWA9UIUC#-AxPwwMn1WF{A0`rQaLVmbUcD}Yf`&U zC36g0r=dt2Oaw-m&<)HI~6k`*^*yrh(p{tUDX?{mL7)73i_x5NI%jx{o zT+9X%ouB(O02=R|O%(;acr>YdrH0M_btM@0(PC#0=lhLVe++Mrc1Fp~v2V`-#*AKH zIWR2j-ZFfAyN;`ARmvV|9HxpqI?ZV@VUYS*I9@7|=EFxyvBFPuQYhUdoW+!rZZG(>yM=OZyg3BuM6px@-%n=Nkks68 zA-oHSnGKv?x)95se@$EvCsIA{q=%@u41b8Da`DSs4=p=6bgkU$^G;oqO-;z?R+prm zJMY%kl*3omP+i87ev4(GH(EkaDV4mPxLtwF6i)t|Y&HIB zQsz4?XaycB5h!2~G)iO*5jpq5J9Z}eJLAg(o-!zBhB3hdslfpWC8C3;nB8D3Fqc4= z``kMlew2?61Zm{^xR6Z$#k~O^9yNbdJ)`#Y47O0a7z2>Rl$5ftpf_%ARR4AAJu&Yq zM35d!V)zH0zvrDT?UC@-m}C$aIad+p^$iOQcf^VqmmE=o*f{y@3yeSDOJm#@wu^^! zHvwcS{HOs)yC3;n%;ZdpP#}wtnpn{P3|A_Ls?0hOqg4WtmKf5G5&J7x&ys9q`RDyz zO*ASNo-Y6?ZX^mreV1mBSghBFsErC0{pSO)Ua3!8bU$)DCJ6GRVK=4~+$G;+n|^8A zV`Hhbrg<3T+FcuQ&dSD0i1NL`u zHrJJV)wPi1Ib$(HSMd2mRw%+`dgU{t0Y}JWFkIQV>Jt0tB77re>Z=JVY}I_*Eu_U#%twuCB2C& zd|^vis3>1DD!j8|FZ<`zyHDgmVAvG>mi!6eOtmT^g+@1#k7eZUL zI;&WA{pw?!n*p{r#=LfM@}bsNmdX#QE-#R=k!2O~c4TW|RZWmLx4xF7VO$Z?IR$uO z{QB{RVDQ@`f??z-uX>3S`{wO(W-p-w4g@#{CMWyMOjDe^ zj+F>qrywhtHZtQRvZ&&-(+bFs;tm%5Np803k6kGfWlWYPM~J^mE%KeT2>8nQyF8tJwa}?| zziz5FG}AOgklrT_l?ge0|k@K5th_3sGuqoQhh%z`@stIINV!NW%C@@9%j7_mG`t9Yne)p65LC}Hp*t+W8`_yeEI4bCxxwruSB-d-LrQ3SloM}Uy?MV^vr@i{gwDQ2W z_hdPhp1@wa5&%OZE**=3X&Ur>$Td%1a|c$J(EUV@ddH9Gpb^nz<%jcwD1;)#?s6c6 zK^sfr^(RyUioZjQ2IW5pn9&p%GXgdueTqME!c`?^K8(RI&S8}ZlK0<`w%FGuq3#x) zHi?5U(+3M4YIQa|<4MR1j|=V1vRoJ!mf&K|o(SR01{CAEG1V|J@!+ofu#FGJVe4D3VQ=HATsO74a;-8(<66BPdhl59qwnk+11s*Uef(sNJIW+H!>OU3Z1(vT#^~8Q|!~LS;tKd@qrQWlRKyT7Uq)r8KE%5 zsH2CH^Umw|)QJ$$keHes4N3q29s5ozeA{zcl2oiLY4evuqLc{+u^>z z=r!oy19NfBl%KwyyqvsdrkCGWo&vk+-M{=iUnlkxNp5dSqi>bZqtxIl? zprogvu!O7O-_lDuX5pD5?VV#Vtlb` zO12w8{*{tcxPd)z2%@Xw9%;qGA;r@E*)=z#ON=r*;BR#@jgW_yFOu&ER`lz4&^`V&pF)WM z9{PFgKClY1r7L-HoaL;?#1DGu+dEVSy^xn)#=soMj5@|SqBsCCDl()?2#l;jU(&P= zY)3G7i`tnl?n)?l_ApWz>9+IfD$y*VFs!DiYqL5pDs$YA<5F2hh&%LS zm-(kUUZ@Ve34`@tK8qpnE9O!qZ-(^M)^(ey%RBT4yp^N0PZ(I(~L+NV7?z%W7y?pXvTSPMcK$RnLHAnV;1pgb&vmy5DUIHkAb~ zrN(vdzabU@#W9pjavsZxku{zMzt^_Rsbe*7T0PPFV`p`(8lXzQ+k~pN%>ynzjlQD}mD01)ZP=GM+zL_>W>WqpLA&psa zcwqzT1(t$*lGDxl{iv0KySezD-dEsCGMZ*__R3mJKE`2N!}{E$dptd8!&dnOL%MC~ zBx0pe*>lHlJ9c882*!5Yow^_dNbaikFmosMom1?R)0WIWamQ<)F23G1m-+v_h-gUL zZ*U-XzpEp{wmbMs)E}80nAqlrxdng#fZsr#=Fxx9%G0nh4Uv`)3T}MqDIsXr7n7;k z5Hxz%(MDzwon+eMlhAynRrbt|#+&qY#lMRANfhB4hjztDv8E%5{{7AiWeqnr8U7ZS zjO^ytFcBb9F$NwUB2#%QW4r87@)xM`y)Rwe)Q*jA9;SsrFT2XX1_2`HuAwS`M6@1u zq&H34!|Wv>jT7^Y=1hTiRt%@V7C+v?U~aiat3bL7kfeE7bZO=H%JD8O4`34cRW@Bzgv7N40(tFeb3}1 z%7%zgWOFYuWN}3g&TwDo_a46{&P|(Iuj6UMhgCOF;x|H<+!F#3>Vgjj_=H^Wqh8OI zQlRry&C(25S{O-6YmNBEcgBAaw%aaDr1t*5&$MqZce6ss0s9_vfJ z$T!hw@LIqswnZvNlQS-E`5YfL*A`mB7gkrd=GQ$b+qhX!Oy^&n5TNZxaV^fu*;BgW zF^#&;$-(Z*z3mas+q$kyip4(!XW2n3rS6N27_ z-h*ASc|>MP9X}ErMCRa_X3tSF`8>g|$xvuD&Y3kKokFCV1%!rF?BP&aM1xM@MaCuU z#p|PA(5c$I8Xl%2l+a)ZPda|%rlkcHFMn>+=@58*qG889MKaoP5j1SDy)icnFwCNYZmM>V@7;3iKoG83 zf%suj5HE3la*iI%-}ac?o#);wz)|Cyzp*3#3V7#DPuFkfUvt$)%v+@4$p8c)z##ER;lDjqxLVz?MhQue&b&&8*{AU`KY{MKTi|Sxm4_Dk<(oC)fL?Z91OtTTDHU*Vd^@TBa=XkWHyj3QF6xF)r~Ac)WF)vssb5pjt2B()Z)Ea&9rmVIv(aF> z*-w#Q+Gd({1N~Jsr~^ttr?qUeJ{drN-mEQXL`ptEw*mxovf#dgmX%sKCnYO^8)yb==tYF zvf7jl2vjeXK<1jxX^vkF?9;)U&yTdJVD}SRiwES+Lls2;nWxIsjxjJm(x=3rNrnK& zcrGA`Gr`j7M_+AR!uFaj4H?M}xK=8NtQosknO1C8-3H|V{+K!J_S|jdpt!LIZ{ss> zR(9|^_iL4CbMEf_xo?LzjRK(LFz?>|fF_rvFc1SVGsj3OoekZ;VQcd33NS2nWMHAg zv34OQf8H(vW-D*5nu9BQwk_sdjk}Z0JC5dB1t`lw(C%n!XeGUB-sN4A8MK`27AkXK+eD zVDH?EGqCm8N@x3}F=d~;Pm6G7rFrwBMGNkY_@KSs6YyGH(A|i5c;#!4w%94&ao=S- z>=b#lHKf6()}md!+G%>^?%q*sfNUchtPTL0=vrh|in(8V%xxaLh-uIdAJlNfRU9(-$1bTNMu%Z(FvR3)R3}f? zFr2Tk3FE&!FkFzN#MTPD$0Y^Qia3x*#g3d^wk~4Yc~{0x;rd|d#|E_!#W+$JTjB_q zp^(XzU*Bk+&<+eXuk2aGl5~kY6M<4mtj}JAG=3#o7(scWAR!?Gh+%St<&IIP^Mts@ z9WzB-WZQhbUacIS_O`-mj%mSQau-C1`+O5k|4fW5Ohmd5Tw@3fyD|~iE+`Umf=gY~ zz}v+U7y=89v8mpp15?7W*_M z;GG0YGCfj>1WN<})ef{BHHjV}0_3Ty1;RxWq{r+yaERpCP~0F2RYDTSN^s}-WH z5>77;O!+1o|2hBta3V+v5jR1FLdN=Y2B7K2!yD+{K#73{bGU1JMclbvhSDA6avpRA z5TsYX7-hHUpV%@e_<#UAsuG3F1r@~YS#>STXqhy?{4Qp>`9ctI!Ah{I69~2}5JJ)) zV-LmAOa!F`*dgbn%rNMO&yRP4b4G_>D@cvwbI;waWdW{{l-@j%sOuyZu3sGR9a zL4la}2}8&sFBycF?;?<3vR9TtMLaAC@*G`SHqDYT_$5u$gwQ^4XFD5TUY^T;XBzU* zLQ9(f9?EJRvO7sjL5p)S3g?DVb3*EwN-u-Ib@wx|()-f~aK2QA)q`0L76qDSUOaUc zzzqYXfb5{7$w|2!;F{Uvx7hgbS58@=nz!A~qr1hF0RcQTSSlEW73p%#LX%lNXX{JsQJSf=bb-TB zMl^eIWVkNg?&dvn5uZXi6&)G(CJQ`{X&6;!%!ZwvixM9nBKzv}g=BooAhQC}bSHuV zBvMk1UqvP%d_$Da(R|9lnpa-Ub$#kc>s~)MRY9iwaWbf|a$WOc*8Hs2taZv=@O_Qp zAZ$&I5?S@gkk(j=OH!Pz&sRh-KSn!@-8v-Uj)Z*tx{}E6JU<%~b*);V$o0V6NV|zG z1MA;gwVXOYR$=Bvzr(H#Rvt)mL=9j8q!+p9_v%aY0aa3y*i?n#@W{wwP|X-Tawyt; z{HM|019z2r00ZF@PAe3@xW!ai87rzCO-z?rjA*hq?$(NQFM=-p%rvklrlBR8S#C?m zPR7s{Y=+L%gI-s!uTePdtnHycSA4fAGMX>}?fWE^Rrlb-JlkyUICN#_&Mjhq#g3az z+jfNayW{H`oUW0arouS{UaJjz)<{N8`3COTF<+6tx5iF5!5b=p&00yYD<5B?i|gGwED zygJouUI?4yv+I1T%yH^1KWsmM$9?_VKkV&I&3}2q-+`ezF7dh|Y9vD~RfVax>#3qJ zMWXrDKp(Q!vb)Kl?4Fz=*z(9{Q_~LEGKc!$8pY=8n);9@HRsw z;rt9el8DU3*A_yOqbv<%<&(&fVeWt4)~}vX#2Mt+^0NSjd z623~TbY=WTdk(|Ef-%6m3sF{xUVMsGBFt1O^eFPgxxb`EWnB#A3!y{UK&@%6NrZbfKm8PqHg|+tL zEm{6Rbw0j8Os;cvqG~Go1t|49Vj6{>)-^L|I1)arDtr}T$S(HpkWE)$Txc~WD(^`= zIyy!|P&hoSN=^%Jlv}K}m2ZuyS0ts5Q+^Kv8*J&1c_o<-j11?EI2z4ivKtl)xr1x9 z>iauIzH+ymS~dY{2Vh^^CtNeM0Jmpq(~!RQjp{%v1B7)>P8)e|Inaw2N=Gv{&OtK;^izA-Td{l zR%KP_gd*K7=0VCd_*kj>#%_WncbZyO0E6XK)w#40#=(!zl6piMzdMloJgbHC4ZhJ) z3|ic1l~E%6tG89?rcu}y%o%h^s)UV>(6;tSr$vXGziVd#LP4`?sjJOOD?Vqq(MG6s zwBeb$q;>}R0RTl%=;Cj~g#krsCnh0lFr7@-)|!nnyEw1}#NiInA}ZL${rlt(Z_CAF z&GUN;Woh?;*T^n!12uZ@<{ND}?$RP>a-nr|q0{U{M&3!v8n?o6rrQZ-rhLmy_l_W7KCB>7XgQfY_`D9GUsQoQJ;KGSaAlO zQG+tmX~`-2bluvG8}}`{j_p()5BjWqW-22(`*h>x*Uc|Io8CHr>0c+A*%ORMU~>8^ zKI2<{YFp(VN5o4rc}uj%ozE1>kNhi;3)cDJaITseN?!7Q@0)N?r70W<&_nO0s-CwKeP@27D43>wPCt zV8|sae>%kmVEW=BxoMDUglS@N{tCK?xb&i>lMz4As7F}6N~zt;Os6ULlbDP3`ZhYq zHp$cl<3?f-Q<)hRR#{IGlC`oS7-*h)ApL>18zJ^UQ6Dx4z|fg3lSGj%XWjkTE^_HH844bHjrLXD9qW?7 zv2e0MZg=F^$ZJg_QxP?5u;^{NW;*Dr*_#xisA?jK`G50SGCs%l9{26KtX&RX=lYMrgkXUt z^vf1dQVM?0?)HyH?F120o|QJ&*L-cKF09y>sh@$2=AV(|9I1FYtHF zwLoL8(0Ov!_FqL5bMvi~SSBfVdH1(b+a!%rqua$d#V5@4?pHh7wjma(ED~(pFH%7A zcL&C`4TXo~tA^RNp?~YIQopaSEgc9i!px(D_EwU%w%P@%|MU_4=}UC!4)J;A@E{_596FFY#u_ zcNWC0trNo6d+EdG!02oQlL<35M?P&oCw{Al4kcE)dS|CDm4q>dOJRLm!)-%MsdWEO z`>0q_@!CN`5sRAmDMi=y$-)(d+~u|z1$DpF2!66~40e4J@8+uHP&t8>jX$7nL#)gm z;YvQ}v()ZkN2JUrb6J?ZpOn5%+t*?ozp;R7>FN8Y>pfyO+$?zxe z=XaI+^_3kz1s})&>wn`N0VocKKtTef7#5%(hDhHCBb>Ikz<#v1J^OjH*FfOT1Zl#T*tlkA3RwR7)dPlmnSfx-4(>l6e?B(zfp;ClzQTm)utBZiuS<|LW~Ln0 zsL1wo?5*?pkidv~3NvB$0L?d;lj7rfIJ$-$!u@|r0n{RB)4%0?HyHPh`S)vOLjeIX z{9E3CSLcK+AnTxx|AjHQZ-MszR~Fzs0A={Emf&~-+VyXN-@T4LYx<37@cl0GvHn@2 zMd3@F!#2Z=}Jmc{jssUxxtsOkGo zL-)_8h1M#dr9&-FP_Zez`X>#2Ekz%BQE+a`-%zuY9see$Bk>16ueYz`$8+wCVihua zFRA}(f$~BM>qBEs4^9syvUV7d& zaf*IHi!`8gq9x-9#V&D#mz#iitKnZ!P>M$>#)4ueaw5g|kKMZcr%BMy`)`d8ms6GO zIeWC=6AH63M3ntp($9xiLyt$5#y12Glf5k>f|bR-hpBSdunQKULZPHtiobzVomNpe z7NlY~eu*s+Z`!$}7e5{}FBx1B=NC}2UrJ&hMTuCRCav#kQ2`qX91;RxYLZ5ULsX*B zG9a@gTS+LUR8y#vVi1o{UBzBtVg#t68I8AlXFzs4fhHJO!-_TSMrEF+l%_!(yci%C zekEy9h8Yve5u%}=7ZyNay?EjA6iKecqu^i3!PcwNxf7e7dBo4Ih>c|L&eQaQ+uUNi zscv)-Qeo&6?9}Rg1&9I4CF+z&gb1N6^Y6bwnb^aSFoDDcrsBK2FJcZ$N|gO&bLid) z-;wVb4-~$zWd1C)in2;B!k#3L`Kt%Nx2M50H;n`BC31*x+ToQzoX6?{Z%-&iiUuXs z63pNu>iVxWiad|cJP%MgzGQ%2|JSaI#pU#v9sw zdU->>C)&NYzBd9^t{$7iK9&)I^dH@wm1J#4ptfyQv_tAuzf482#_A$^G#FM%o7pS% zQDQQ6oAwNl@Jm|26txvcOS86(eiX}qt|~#P#A z7f5%(LNzKyHJY$ElbpCZ$F`XusNe(DQ5rI6aB!X3Mco4Q7XA~MXfZ}ymhX-NL7Z;H zUgH0P+YT3xvIR0rDwZy&NE?l8H;y&>86CzL+zk&v3z~tp?t>HPGN_2X0Mu4FKEP$_ z86eJWNQD$TxWa%rS!d3H*Q(m|tX2{^1zLY7TQD8fSx8|IMz|<>mRRb}GlsR(NPMUq zzrBs~1GL7}TyF;{YK?#oQ^-(PhU^oE6{n+bn(dm2T>>$Vj)ZNzWEMw=ql@PaP_g4K`ss|074`MZXt)r3d?q=jVI&`d>}=L=*K-Lm>%64U!yCqt8^Nqc^HvheEq z+S+Y1BBFhwVqY;fkg5d7a5XALI=&|N$7uIWYSOAzv;(oozvf+p7=6imdu zZ9ttHzscgy1@Tn*lbiyE!}irIQO5LLa1H*P7*Z}nb6JW?ii88 zvhhP#(Q(q*2~M#=w!d@vgE`2~w$Z!H?#8&SZU zFi_VrlrMjt-lGXOkatsnZm)0oSNqR#dUyMdeqYP|9HJf-XTYlcHUG0Jgq-|WEdaD* z1$at-IUvMY>slxf#6Y&(qXqP@E>`K#={Uo z>P8ckrG2#S&4}!&H@E>OdP+m~Z-?$#w%YJ0S$F6;tM$Kofi4T3owa4WoSz6{S5L!> z_vbCi8FSqRpBVqwAqe`dY`3^Qfl@#s{72;WD3{FZqy+*B;s8z+BmhB6h35f-Z<+lB z9V7ZbCl%#s3Q&sb#*`k9xCVsFY8^b@x{>}maUSXVzEC2xP5osbQsqo>XcvMCBn-mS@K8OX1SZ_$# zpQ6YQ#Mid`De@GC8s!bWl8}h9fEf@;y6Acf%t79tgvIKv^$b5bNSPS+4e<(P5?(bO z8Vx&zXDE6g<4Q8(8+KX$3TP2#D~23{x40$M3t&HXCTMj#U=h+#!ZT--1m}Wlp_2ds zSqv38>2Ma9P;h8NI2K|H09i zhoa|TgQYLGWr)lLo>!&8A^(orJ`DaE{3(W!ew;*KfQ8av5&rFC+VF~n$<97e(oyVH z6Q%B7eFev$<|AAYua&2ku_X5Rg=Zy!{6hl3Fl%ch2GV$^iA_@8`$O)AMfi0ltaXSj-<0@tGhG zcU7F!Sb4cw6A-c;nL*K>l_Ycom2#&(G|MGTx>Tv?f}SUpN2O{{CqQgB*A#HlXXpy^ zeM*svkX$20^+J-Ay|>ueRImi42|)g;7S=gZt>w5$@FU42F~^in{^13qnD87@X{PQC zOgJRaM+{ST>a&84x&?=Zj=8vM8#18>XhbIbN5D;sH0o#&$e?NiZl<=te!rvk_=P)? zF-q@)F1iIio{lsMFksh-=vIf9D=bioD!0#0l>|!BO0Q0hdfZi%~=m@TG5QUI#dDQQ10DDvbgHYe1t zX>YN$gj1O!HW!gh__ZoGK`K?-bS6gNTajJubZol0ZmTha;{3h}=Z3~3sh?*ruXWsg zBQxSSINr1*J4qydEy-#8h@mAjD0re{^&t)`5FHRy-zG z>~1_T*83Nt8UfVwB-Ndk6}+wLysl{B=Co+<$Ia_U4jKVJ^Eh`kcCkZVH+ktv9(;~% zI?%zve#k)ic4_vtS#f(IOzx&Y3)kE+nTznY*IhE;H|KcVJu=-__-LyloXW9|q`m~A zauP@BDYrTMXHywG7j<&Y_A^3luc)gFMrCXVB{hu94gm65hOon6C?j}SRuM)9YDadW z#SRQ1TO!S6Wq8`ctWRE=V$U%|-hx7mIZo>OJl;b)p*U>sX*K__sMaoZ$^L z5dof>kRM(A%Csv%Xz7#{;cb(xO#_;|C}(eZX6&J=eP-{gc3ccIA*->Dfx|6fKZbN` zmWUkAHUZ#;grBz-C5{uz<;1thazSm|Fp1Z*`#k2e?6STwpTS!@#RVGy8@_pOi|C`i z74n7PVj6{w*G60A7ovy+1bTcLid1 zF9FMEUmkf!>skbY5NAJ?klgbq7(UG4 zccnPYZ>Krctq-O}E1bWN+YHC6@?^rk<8yf%-Nt|G077}HEcQeClFh}DHS~{*8GEcU z5SE{*HEV-qtP2Uht}Sz(S!OmiS+CNDb~tQ-pKkKZ*k*2b+Z00Zx>3jdQXOjsZq~f0 z!vK2`RvYRIA4@g1zSnEhzRsR?y4B%xw9Ss_cu_;Tda2?Yff_}H>a{lCbPG5rmJqNhE!j$fBg ziAAubZ-U)-~YG|Yw!5||$s~}-Q#gFyz zd?NX$5rPY(@%?u9-1gbCT*H7v3A2h#+Bk@2zR@?DT7U@Ia@;vH2Pm8-+{OYFUhuv> z)!$eY%#Cln-JdP}zVg)+kQCMt+IEkaIzE?FYH~0J0CDaby+eyW?ZQHhOvtz5%L5F|#{`NU{ z??qj&T5FEynS+!VJn=}SV#(i8B!n=xP9!&KCZoE8mN*CvrF1+~_Xk0ezJAYZ)Qh!f zIY5Yp4yIC5=)p%C7jjUIgrxdMPQ>y&&AdnyNQ1S72sD>}5{N7x@}`+>LcY`hnor$p zL~<#xQ_1|K(o~A!;wsECh8bUobN8GRTfkGPNCsUI>(=Kp!HzOyF$P&;Ot+u2^U-)_ z32Ozg6M=s|YX-2{P47a;FkeQiz8h`1eC8g0C)J@&}_(BZ8%vAt9?Us+Sm$XG5# zx$g@ul?1<}$>T+kF)-`B!DM)DsqheqALbw##SDq&p#;7PUoc_4%vmogBS~)Pip{$& zoE`xiPc+GgcmV#OFhNU63rZ^`HWXb#b+;ri1@ovaGW~}E=bV|<6Ky!RXTfVtEO!Ms zU+OIp&J&997q>D|V3s}u?t5CcD%*+^zFoi(W`d8JL#}O=mKf}IHx)#(qI6#?vJx(4 zuCytMA}%*Ods=SL5{cmPC)J9IG{Kn1Hsl_F>D&~3c(11}_>*d>Q{{n@x-|)&7Hx@9 zQe^+;Ms;5GmM-*srk;^h589DqkShbIyO)RMP2z?-T!%=WX&7_sy1qmjp!5ZqkZ+n{ z9K;%e*)z1-C_n56R(pF;+2&pm9+o@)h(oJGE2rJ*Dm~z3i+7oHnhV-G&THXbW?wkA z?i&yGM}h2kr^|SW56%PA^vN(;94}dPlD-pWgB@zoZ6KH5LZw^K0Eciv=5!tCl6#1_ zl+A-uY8HBhaGA)1V#hwE$Fxf?eX>b`-)Jd3vx*y56s&E#h~s`lKArws20N)lqU~q~ z=fG*<0}thtz?5FE4{uFd=L3a>IBT`no@?&8i4CFA%F;{ouQ~KP2L7>4jwK64ktuzu zVt_D;;S@SS>@=*Z!_r#x~}h)3oZQbxdhGieLh0kM%Y{a~^4hI$JnEVJ$rVk+8{>$r!-zF@aW_T)9iT~Q=iF4j$88)w0js}c6&09hUaU?>`kA)Yl!|Ed84X4I{(0kE$CN z6H&jfC{-tfa;9{75Lm^>mH6f3aW5|$Y`ryHd^0;@A4*SfnjQB z%m9b~napo^0NnqZexVftIQ}oDDi8y3|L=283SjfUNK0N0u>F5BlnyiformwymQLaS zxKl9N6jT685WxS>ZHfe=Me0vC;&x!NBXVZcMdy}KC=huq-N0yJY12?35m0;pxc>AG z>lAe(EmKo2a(GKzM>gklKAFang7~72CyMu1uBppO-xbV6`h68Z6^1DqaI`5VLe&mr5^b#_XkUa{r)q20xd; zB2mySJi58DyS=OVl>Vs&&-tW3v)TEAgI$KQybD~}ie=@tmV6;A+PP%4V`_A@qFv6# z0bX=V3(G#OC^%!!cCxJ$OE=8wH&xKLvY}Cut^C?mPs@3Qso8}T>%1kVPR%!TSJHK} ztec`o>ng|cbPL|^6A9QOL8rSSA{SNbx-h=4_=_L-y>>MGsvDR=SDZLD!d>& zezS&wNSseZIc695OcjAjTbolP`v6Zf3mnFqRWjq8UBLOTyVxj`Rw-db#X}^ox>=RB z31|3w7ivLpco1p0#aJbB(c!pZQiP&eN~$UV^?l(Ur%jO6!S-j5cogD7eWRJQpls#u zjU)Z3yY~2OEZ8?d-yzT@*CM^3HB`n#rm!rx&4gU))Q-^jwhS=*Ja}k>g*5;iUd(dl zdN!K8YVo}G>WF}Fodb2)-#;^RVXx>n4L}9U91=%$*oOB|+JA=)Li?v#O6?Exy;V+U z3E}0hc`S|lY-On|MmB4GFBv&DeFe!i&vcY{kNk2VORf-GNDz#e=8-UY| zOCHC8!Y2lE8_{<-0rK!neGdfw2ucAp0$B@1MMBb{0SjFd(w34+y}}PxMM%E73G7L{ zup2n8Wx#UAEL$Q#1;rR}c}9ciKi0O6e)v#O4c|C@x)y|^L$nl4J zV<}|ibcv#~tFd*w(`>5t==&kp@1Ini4*pwJuxI(xi)+(R3-t^b`Y`5!*ug!8?9YjR zVLA@wOS^w}+HY0$*fah0LvtT(e7|s(6o(v>yrHP%vyn_$iPd{54MZQH4x*K&sSm&f zb$o7f*9V+|{5LmWw+X{K`3}_;G5`PETwVnLnbxiH-JhIf2)O&-p%!KasQSOpGInbK z!ha1;$8sp)Y5bl5R-p5L3{JE90gMF1bLoZEI!PKN&YPQ(r=>R~F*~y^wH(;tIuYfP z{*eBuu6>{A48@cgR)w5JyMK9;A^-*$U2`}58Ujn^>ffz2z>j;n30F$o zD8{wU%;9^h7VKI*gbzh$RYstye0z)E5+_5p z81bUDhH{nk{C-x*Ofoemt-w;!nF^JBhCwZcLJp#ck7Bb<|D#Mg=NT-0guV%jl92Zw z(xuHhN#V`TM}Dk6e7Lg46ibS;me3gc$ea?9gG7?xYyFOGZHVf`Y=MUl`x|C!f%iOx zYiLrI9`lZsKotp6RE&bfOHdVre4IXK=%^@0VP~U59{{+w0dr@!FI(^H`CDG7TB~BU z(gd|^vP>6oUEiz2cc1^q)c$SwEUQ~oA|a!|2)xo#$M`BKUHPDZCnWLc0)@!3<=h<9 z)TSivZd?{h_ybE@3=4wN44{o$Dq#iOgcv@H?w&;oSV+IjQC`6g&q*09^=zW}+E-_tvO3bg5lR;n|dqj9Z}^dP$86 zO!V{q5K+v3zL0R1Thsw82P~tMIA~OEl*tQ;Tt5otB~ad&B?C!DaB>4!{)i+Chh9DJ ze-f?*Yuxd#uew(>dIf|Opix#SY4;;X#`JjPQs2tpbf}wMsdGu1bSVKG3 z0&diXo|eY94*}YR(x7`xAVT;dLOA_76M-zV-Qr{7_o!WaQvF1mb>cwaYDg;;h2Xxp zDJ+pF^tcQfq3%x3=7Anlu&3SX3e4&6U6mf$j8fju*@j)_c)eM4o>Y3|r9e9Cy47G{ ztx#JyXnH)=qs*&Mk`Gg8M_X%`w3fc*q7AUC0w<8X(l0O_GuNa$G?aK=1G+O6H%GV> z1XfI1o2v&h`^OCy6%Irjmoh0zyC8Hj&l4jvEIAW_a#Borlky1Ncf{@zv%X>>`19(x z9(SMKU=pT^h;$K|qRuo;HFK@v_#;~oa(M`BX$3F7lR4lik zq0>)mlTosD>%ZqejHC@jP2I!;WEt$Q?^14b$a&cOnHPdSvMtt&eXwqv97ymO8X&(s zF-R;L*iZsWdHVP7ThbN)P<0eabjMZ56qvL*i!HsvvTc0sI%ul>$RmufvJ-)xH9^ZF zu$EW_5)FwmZuE@sKTl<_L_=S4n!($+2#tY4H(V^Lx~VF)92DeF_|*ythGq4N2eFoY zgVN$B(e4y9mS0H5ebg9GK$Av38|z{F6*PgLtKkx6YvQKi9?aOsv{U}Z93<9^bP!uV zRnUkRG08{AG8>z-o7y(So-Ac^yV?aoTJ}TaMmr(QQXsC^}am zySn3Mo*3`Zw^**y!zH=Ti1*Q05pcjO%%Lk)YbC#)plLfriX!enT^B#!WW*GK9Y-Ft z?Ty&fCXSv*|77iq-*)|xVEkWo3HO9BoU{w2{S6-mhu7|NnoU^d__+Gnl=3D zPs0neijuzx5_*ghNsUCl!(0TYhu#s=0W96bw z{%Lw@fgHTK?B&y3P@bgU;Dn{*76{(I{DjDw*iHQPD43nWbFy?v;Mn-~b6X-KxPb$| z(d1c!&#sK*6av4lustlJEI0RHLzC(5>lXZw^XZkh$?+dpQP&g@VuwLK>KJWQVYQ+G zS<-(CVU!=Q`6FUu=&xX*AqbB_`cl}Ze55l*ka-@|b`wMRc1(z%FK^)opZV8;9SRI0 zmC7bw7Pt7kjL&wy?a^JbZ6NA;xNAV-a z(7PsRy;nv*coBB#F0+4{h2T)6G?LN@uKoAbkg~)X;2@7Xi(| zh+qhw*5EmCl2_`wm%x3x5F2gQ+BUZ@rkf+eMTjGV7eNsLaWvgfb@V2M+fLs%^rS5B z)a#&FR@SC;JN@3p!Rb##> znl4)%8Dy%;+xVYTwI$$xTX(NCwdD^vQT=Y~7zMG>B3P3LW0#>9s#!5&hH0xryBJuC zCS~PNVAp2X5l|c6pZQvob}3<#2dmf5rBTNC*KU2rK0?g!;(^GqwP^I7%AJ>~>R#V(kFI!OWadXn$tM2#Y_PQ%Om*(jD{DrY$A<1N7^EzE&QYOJNT zx5=}205y}5q4x&FN}GVJicuJ3au7OBjJDXkDU%xHI5 zQIyhpNSi7E6UIV>nu-O*fw6_49Llsi#oi z3`lTh7u}h>$ULwI@pIGV?1go%B5K3mK%#oT@ou zNO*>sTeIBNlULHKSFtBUf&Rf~^+^)0%(1uDeg3wxpcg-l|Bp%k3%0+#EKwOFSS+Yv z@RH~-L@Uy&Mj&k|2tW=*D12&cV*f=opO`n457Ajt5OzN&3&Ic=oa$q<_BA)c(Q^a4 ztHle`;(F^Q+WQ??7I*8s&+=^M_Ib9>{(HSUCjMv1bDG1|EQ_EWu_dOzKT_M%rFTUu zZ$ALBb+vN%b`=j@Nq@?7eFM??EL*YG#(w9%-maRm*|-cae%}J#=0&il;buBRIwlx{ zR_#1H{Spve{p-E9t*8>6SUjHejKWRdF;dz09H;1OLsK|$ke(~3ihR2OFBZ-=X;FGY zRuK#lyIv|BYiZ8tnUckkn(nxHVA7B&rzSgMG+SO^SZkshn%V|DfAoTU^A7W_6E?Yg zsuacCQ{SVp++7Ba)rTu5sF#y3CkKu8$ivuw+m7Yt>?zA-T{H)tr-8+7L6h3$nR#^20Os)>A+b6wnzsMg>F-P$9*gz>kI>$%S zW{|Pf*;&Q4w`9m3tbUoLwLc2CM)dSx`26<#v^`i&x?Tjb_lSGn>}*tQS)F&y1xI<) zkmJUxsmUzoC}GG#$tY36cNk5W!=CVksw~$JcyW ziQ`mG_an;;c|FwSp61@v{JZcJ9GE&B%J8<;#J1Jv1y-=aDMfmoqgXGuof-|_?8}Sk zB>hZVvgiXc&y+ZV1(T3LQkO2%+9X(j}cqv+oRGL0L?}E$CT7n@zyslCg8Tw2t%`NO>2d+ zh-ht}ES6GojHdna4dd8|Br9=iN~ab2t^NNG5*6)8V&FXA@W1m99UkDi;@G?hs-zrxI3* zg&Ol=E()Nq{1k2ZE+p!Qp`c^>0xQ`SuRTsMDk32Z-;2of_ zo08)a*H9%Ypc*ZG0+*&f{hn|&7B7|fwunkz{cHy3@5ozXmHqV|f8%|8=H06;!M#wG z^RGB^MYn^4?w{2%6+*E^7ILa;a*)vaB7LaAzH^1yvn(T&S>j9C-(afL%eJ#2R#}IV zo*q0yxnkH#pcXMVkG}#`9%WX@Mi>|cm=MI7h@yt1CK3@a z<3>E{rEM-Y6kamBv;8Y;E2^N=I{B?8lJ94`W7in7D?&0AMD8HRav~bv8D5zlgaOD& zgO>u!{^dxHo4C_3VRfvs!$pdUIC;s67OR0Xc|m4q9YHQ7o=_LlvR(@EmH^?RQrxcZ z%KLK8-i{3=Q5fKwY$YMBu3+f65saSOgL?DEttLdRS}GbAmEJQOonq%bwynz-;SIea zlqWP@m|wgYqy5_3Jq-O2^#VB*o{CHJ1Fl>Ga(_K7|5d5gaEj}E&5c|8$D$`YA)TlM z60%v?)H`+#2*xnc9j-l~5Cy&$C6gsU6=C%0^)vY*{z?X?3q1K7$!J0-Za}?{OGs(r zl!FvRH!q|RNG9mBngucrYLOvVSTY;^gMr~VKhQDcZ)-2fMm}M`nBb2)Oi8|H8XrtS59UIpk!p_&+ z&dJ`ct-*7Yt{v018u894cml@>;sJHptP?Cldln*G~p_1tBQBH{(05SYLNSV z%nr*MgHRhX`@>-a)dN_IU*pbGubnC>RGgjSS%pS#7-$gQh!MgGg)B|2OEDArSc>Ac zV;HKLX%StHn}QPT2v#k;y%dlYS^!RHk1aG1dUVDGMo3M4p$fMCEWw!V#BPz;kI_KllBdq z4FP6opnBP=jPo@4x5#4PrUH%zupb=0{AaPrlgLv1huGHWP*o%!&u&QTUsmx=`TP_P z+4Ugr6=e(wUOS-RRo2!;@=!f!Q69dH%x|qyttmI?hqlutmg~s0cuYpJzaCA&o>na0 z1R4W$P~T{~aD4yKUfp2B$VY?s{@{}F&vcF28OTn>$qQ&7iD|&q*6HeR?u7oEwnnS? z5GLr=%VkrfqEgp*r`{<)Oey7R2TqJ`S|Pl*rTvZjjS1lWd6K=5N0rC+@NlAcd+TVO zugYHYi=lpUDXh03dUJ3zHj?w znLT}lKArle{CkHVX}9;E-x1q)hu8o0R4e^QcK*#*f25`iU!#BXRRoI$Lp<*0k=|xe z0crE77}`qGmErL}d+t6-B^r$nt%a1C3=NXIZ}%NMUd5$9W1gStN)@RH@!`V%Wrlq+qcH%#)Ef&udk4Sbk=2T2 z1PbroY(z_wO+Zg528zJTEpUgIg3AOFv!y(d>xs?50Sy~jVqP?>ejXVOph1g-4KM*^#0R?r>9&0EZPF%*A5QUbPd`ilv61iA&W-J z$DHjM(%NdjZ{J~W9u?3y*q_i<&MFg}5qDe_h!zM5vjfbZ@OVr_$an>6WYtN$%9Y7U zl#{=Sk0WqXrX&~~L#;%^iVhbPi}4J4CWF_eb#@*A7cmZ($co;l)5z z$Y3_`7N?+p;T-(bEPQon`8Zi0QmKtRa!CR_c)_kD)&kA^RnVdMR{Uq+b6>Z`dQ3w$ z$d0=wy<2_|wa;rQy9Akwaxx>I$^?!aU4_^GC3VOyq9{niGi#PeSRua4Cc~5uc@;Z* z#EcroX0{(BI?NLYZg&Y)H1<OcIEIxlR|GhN8A+6B&>4K^w+# zY8HH@48_3w>T%i{wnr>gbVJS3GTX7VmqvV_@l3m_0}#MJ1fZYX>4qT)BElZgr(xCu zaDgws9lQhM5Mx)l-PS_y2j*_V89ruDIb4LR_=$84!9pGT8a%0sBJ+6`NNU>QY;*{B z4*c&+|1@iCbREi`!OFLl(RB-GX)lHFxhMcPA&~d#2Cf-mF!+}k>Mpc%?}QZ9&V(xQ z)+VM;Z!C+b=N2l_>i)yb+ZT?wmh969fP7G37%rM8g!b^XIKhZf6uG`QV!0hF*nIVT zc38c|<_akQC$|9+f#m#<>59Ol{NI#v#2vr1EXum9ir2QfOa+uGE!vT~sfDkbC=QE) z4iAU!?;SlbhrM-0lvdkHqKtX3w0yhh><+h(YUb_CTV!05>|K z487JE;37`5_hKA`M%59`M_0q~3@JRUiE*7fbr0}XZKw%Yk#};=b>Z4OqpEiIj>WFL ziTAcLHdOU>A*np{aR_!O=J>#azqxkdlKVnZKYX9ErcvMbn{4r{Anh$K+MU32o11wA zq^m~3f&w8Z5RbE5wOh>KM?7UI&I7Xx#cPSmwkaI_hGX|O>xx@DS&Q2rK= zx4LYjXyAxi50bRo&Zlh{>P3y4i51)JeQHMWOY&a?9LH`i!E45t-$?X0!rC0hT3Hl0 zdZSfo1Fe^)(0zBmpk)G;E=7^%toZ_R6nX6o_9bgc$pI+O0Jxl{5nEssxHtp;mr5FY zBY+Y_JWapx`xtqvO-OGH7O8EUX4h+((OOf8qBh~ILaqF)Ds`t$;9k6(RztygJTv%U zFC-`m)gBYpb|B#s!~U3NeX&q$&#=iOY$%2gXv!ny7E4*1JS8$zLr`Kr`O#nP#DIzV+rOGuNSLNIF7P@VA zb8y7tHB1G?+iPWcgUzird^Zq&xayx4k8&cm__Q+YPzvh2ZMo{;Z@mc`e|mY?c7=ug zYvp#ycQ$e-+r8Y;9DAydk~Y_waNm=sffn&6+yB7j!Pyc&%f75D^UQYf(w(h;Th(t7 z!kK0wVL#B{P^iv9|E6RIz#L;LHg| z?TE-qcfB&;*{MPL%4x_cwp_aoY;7i}B`6%liZjQbJ6Pl)j{S^Z_}YS0U{{4P}b?jvfC)JLv~r-x#`SU_6|@tr!)b}*@?z-pX|9#uqpUa5P3 zrX?9sJ#Ai#CI}5pPH)s!XkZq8+Xf+AA7IiYth%4?rARlI1jnuoo-Rz<3ed>-t-vL8 z{dRLNSg%v|>2Dxe_`8Pf2pHBl!D0IB2`;b^!X1y0s~kxw3jReDt!DCtQCt*K+Ztfh zi#XGlbCg)fYYy=J;frBUze~&=6dbtRSt?ZpCYawp24Ep}WkzAHD1MPq>H$aj3O0$!UZDP4fC#N0WH zVQ9|Ci$i^nTU7k4(w~JnWF};|f#UR|4zYJx=&J|=&GBn~BM3M5rlitTE-{*MynY(q zHz)!Lmjy;ud#Dn;7(28$31AL8Rb=29!L^UQ(c$pfMLk!}0`yON8(n`joPHeG_QhSC zdAzRs?*GRXLW&HE#pMg#Ma4F%De1Jv=d5W>w?Mn~1ai;WJ-n-P`XT{Uwc+i~r+Jl9 zf5*+sAt}&={0-n{OhG5D%9!daHibpR^c#SP(`~+K1zLD13#(z41kO<+WgOR)BCeL| zrLXX|1XWfH0CBb&V{ag^Fk?&CRVW8laqWL~dvi(V%Fj|@B%Mw)VPGV=Pdu1^^@s=0 zRiJjWx^Rsp9#D~x{Iv>g7f4zoC`iTX=bl+qvllaKH`_#SuG@Q$Ds5?%{;dUTY`1As zQL=^VG6a2Kk(I=i0H*oy6T25Zlk<<-qe=op3zL5YE-(_0f)^YyU4mf%QureFnr4v2 zf{R;T9Pt^4(bbgDqDWw6;%I^ zx^_*I2;i7KOa@QbM#vHhNTe+8de7mIzfz4ROlsZmP!K#gIL!&&-H;f zLcFK3dLnxIX`8~7**<+3DLsd{>>CiR$FVq^57n?3&q@qwW;XEY8=GBf4NVnT_P*+& zjpaPp`Dq9kgB&fRG|qA;h4{0;H{VU=j~I7o0D&+vh1aB#*U|NG@mx+rfo53y<^ic} z0=%(jn_|KPecDzi}rOlx&NN8H>7m}#Q^K5HNiqw;5@X1099>W%T76|)T zf(@)O0!>(1Pos{Rk(t(P#B0y9^1xp=^(6|1q3Z>ZJ1jw_P*vRNzm%LpcB8aPd#d#X z*Bs=^g&dU7Agr9MS>FVzab59OfGfxOxjh!iC6wT%|X2ZXlf#b&Q&hj3=u36gSW<+FPCdlzLAZ z;H|YY*sA34gX>JsVJ+$}9Dy!&CzIrD6_;S?8%`^(MKb)GIf>K2tH#)aHbbK_L_aaz z@3_PRU?N+4LWX=o9##waHzRTvA&8oZ2jxt4LCw1U%kTZHuJ4P})92MZ&3ynsj(Wic zd$k#;`F>3pmI52S0iV&$Yxbu z0QGU8LLek0n6;_i7L|rNd{3VQ(Hc?%DH=qBu<#My+z%OM=kYWVp7hFENvqbgGV{jc z!joBWjZu+S2wD4gWpn0Hp6I!0oEx5%C`iek$lnX}@}4^e_%CdL;XPH{L?^)8_*dyb z&S5Rv=C0sQiV0gJ=5`o-L9qHH`uA_GwqUb*j;s#%rlzeO0 zhU&!+IK6UlXq+{EV!MVOqI<8wcV_wAYMEh2%YTvT_?BEHh_DJY@cSJQkIMnQ$bbte z+v`$v<*rj55Y=m!cn3Pq-{LI3<9w_ZcxpXI=86IEpH;H-mQxGew4tXkrb^_Oov&}e;@^t1{&a> z1sLMDK0VvXLix|sBS+xdDQBS*Gn0$fD`{kl&aGFm6)7&xog}Q2 z_CZ00u<9}wB7p6iz8HO{P8(9{^#s5mB65DRWTvPN9_>Aup$HYu_Mf9=Vz%AaIe9ELo67zz2cK z9zwuBA}tPQe;=gyG=grnjk4-MUCSfjpRh{&y3xE~$qdmlGsjsKHs_tu3*1Z_>d>E9n$*UzOh~q~Pwx=@A@I5j@=IS__^{dtt zpql+T-nmaACwIAY4K>t9afFSlcid_pth<50qFoU;MbziEca^y0`Guw7s~MPLnm<1z zqX`SGsaqpIDS>`FP@}nU`M;b(jm>$lveJfbM7nhM8gTDqGnNAURf`db4bb3YwwEMw z0@K^Lf8(RJZ=KV%5w|hyRYP4*anSVV#@nCB)_WG@z-lP*T}fm(uU?;f1$-uSbxy=Q z?5wr?)c;D`(pp>jXLq?RR)J&+*Xz@jvb%cS)T1S3mjQH{7cFeQ?8E%GO}3mnT%!_F z&p!Kjkdi~ks?9eR&%F%0Lfxr@B+-@%Pw(Tk3Z$a3h^0a$OX>MrO}wW#rpT7;N8sL^$R0b8Ft(+%bXW8 z)_&|J=MtD5daCk>H{mQb$$ghyxlF`cR=E%6AvY0#z*|P$@@uYQ@_NF#Qpo#HHg!G4 zDVHUp$!X&;jmU8WN#Cz-7SC3h((frFALsey7az^(1+u(P8WZ&abHtp&ZH|-KuJt9# zUk4AHzoVr;T=LdFze~Qp1CBIX`vXp-vAqYX>T00Zg=ALf4{<}pKV}5QNorb_^6qtN zl`_nv=mQZpWu~f+U3=@$tfdNrLvF4v`XIB@>krS*+gMqUTsKvsA$x>Wyq`{G<+p_**(PMJ+#3mw=#0K1MyR0w?CT)b< z$@jnvl<0M*?`QMTEZ+CC*>129>SbNA@JI7Q`UaJLj&M7cd>_vMbN8%>wucJPr&>*$ zgy9@eqexGP8KMc0{^H6JOhA#?2!sx!6zq_x2y7T)*_JDOTV$P;93U%B=gKMez=Vj} zGkT=BDVeb`z5&RKted3qkQPREHjGDZuK@qJclGS&uid`ruDJ(J(G9pR3m<&+PMjmu z+20!AQM9jnv^#{YjVYL_-^OqvFB84(+B+pKLN~9S_z!f^Mc8-um?O^fQ9Ay8J&otN zw3BpCTN)x7tj*@#A$ciy8*x!yxh<7b)336hI{Qgk?Q}YUG;x2fOHPdFZehq8O#yVC zY}@+RxzG~9L?C*#x;rR?hh%nUgTvDzyOupOOcYsx+V#%zNk~mXZI~znYlmD3YJB!F zx@c^5xnz0BK7|{j%~f|NeNfk}IbA^w;LVWELhf1a+eNNSjz8MsaB}?epUHhmIbzE} zE^+!d`(RX>Xyk-u=6oxS@Q(7Vgbt*LJmh^Qw`9L}usCm#w9N@MRC6G!Gq^u-w!fpK zhfRQJPL6|;aKw>#NSXa&ZYKenhd>#Y)Y8?}a;kTxm&i1))=Qt$DXqCx!s<}lTuclF zQqx%X_%|s~D;~gU{`3J?$&(K3(wH~c&Sq$@7vK00Ug0pwgf_Z?GTzToF9+)GE~{^<|84buObZ zzS{JIZ?NWJ8?L3hiW{i)Pv7S(+EgQX=p4s0uAm~EXf zr;YdvG;S@P0V2Yd-VB$`8{0QwUydJYt*1D07%h{TmPMi6=z^56y=4hDryGz?neqR! zP||bvGMAlvE-4J3M0QWQJ?UXFYwM3tY$w&ne`jNppT^EmMk6QO;|KN@t5yE&KIKAV z35QB4?qmqpN)~5xXN^RHgPC;S4ALrra}b72*-kHid!8yqI$dLxZ}iESJ%9Vubf1MJ zsbW>F85xUYh%-w?pIgZ>wi=x-zF0c-MAulRNp*(L#EB^XjB72 z{T!P>t(L8_gjOr}H*Buo(5px|8)ocP*PqXNUpSd1>{xw_*G3X>x z;@-iia7=u%%Qopp1XnX~fHVM4#VKG9tL|rI4L%yk5m-iPmTx<@7*B2p4AwW7u1;Wbp5Ut4&`NuWwv_cb{81@XKg)%cUAz;7D#;Vz*}It8L*THUS#=Dy8*ND)J9EQ!Mt zBbj_ATX7m(0nk?+Jge_=KoLxCaNaRW)FVeT9rd6#BjU+ixZ?82@o5i#0tv@rA?gL= z!9_18UV-8)4Z$T&XH!C2N_JsvK=2G48L~xng=`cO7VywWNGUyAR3EO69i9=y_+isF zTR&Ltr)(Nrl?YO~ngKaUb-vMPl-$Zg*8;&jWkHgY%07__bt=l1o)Q{4?9?gEo3PCK zw`A^&StYu$q$m7kP|vXPmQL!q;r1bz$}-{camc_$zh1(3RwnyC7!W}SA8`;k52XAT z{dp2`2bBCq7~(Z>%9)IKSXN3CL{d1Mbf(lS8HyPfBM<{7lN;#R&n3u=GcK3WJ$uUh zWP=h-mH{?BmS;B?^Q6`a?jj}wVhGtVOheYo`*sfFc@g7(&svf#@wIX-=gKzyD@s@ zqzv{(PLuz%-Epmy%j=f&>>j8D#rg!Pc>1D6ElCP`<9`2aiRzCxR<;r}9h@9nWz;|g zadX<~Jw%z*A5Ay}4Lh{H@nah#`pt{tr0xZR(s86T!lN%a(V#~>Lq&hp}%ysYawq~vF&P27-WQ}+h?LUY{ z&14^N=~R^v0Npu@VFlZVB(biE;XStvklUZ^L=vtGHv`2Y*g_5~oJ7*Nc;~hr7B9Ga z+}84|ddXQ6@m~Z|aD|NkqG~7wai%bt`LFIB-8-{X*GRp{YE|YyMJYz=I`P)0W4%rO z6Feoyupxc#af06x?Rm+EG8sJhPdWP*^LuY`9!+%i=H4Kci(_T|0OGyDId}M2;Kp0j zP*+PTe0cMdj7M+JShLx(tr==-Tqw|KYlfdE70^6z-KK+|>3r^zVReml6@Q@T;Xo6> zhF3fHsxCzJD_dxf6+PcW_)~J~si9hH!YyGY5K@0ucg3$amX}iv<6kl4m<{a833-v`8_{YgMy{{&vVqd8Ao&>O-3ip*`} z4!#xOy?uj!D#~h_iC`gV+aMFDAGK7uD^#@~txXb%8rZ+JO52UxkA3b}!L%Fj$v?bp zH~az&Mj!r6vm`Zi%!{EV>?>ISe-A{`J;1S&DULShE~xlH9GmKF0~ZV?*Qz4B$m}Jd zm9ssm;*614A}fC!`{3&!G0&_WXn87`AhK#y<-K+!-DP~dYSie-e7GR#$T))!uqHB9 zKQnCLp>hR8*R>Z8Qo3&bYc6DYNHmXn-@dq5_O+CIFGmK#01-2djs9Z#>U?b}8RT2^Zq zIxRJOOZh9F(eHGh@AFpVg#&esjPn;ST=BUFWJnGoM4*v>Gf zf$ADhh|rH)!s`p2&>vlA)mM1%y*8o}yE&kFIZt7%FJ@I>Kry=p;1<;{iFG|1Q^bN9 zol!)F>lM51PhQfAnk-slEvg5G^Pr-foPO*8P9JbQ$9FI8gO~VNPHt22?v0h7f>fT! ztz5RW%KZmn_2dskv#TlBTLqAPS3izMN#JX5a3LOzADdV4gA93#hA(#NB0t<)t@A(A zfp%8+O3@Igz?Ri}pqLVP6qavO+b+e$_O=8Z2iB}r-W?=UVtTT5BW3WGN`{c}PN;O7 z=xKFmK`xW_451&Tc8IE6&P)8$UzL#&%9F|PCqE6>u*+DOYFIe(QB<&LKKu_}1&qT*A_B{=AMN(Kp7eqCC z{+PYKI_3Takhl~=6mYD}Z(DBDJ!F{opDbDxfGa|#WWCAX<}i)3Wi!AhJ%KiIqoE}W zyeU2|qA#h@#CX|u>QAF+!^M^(MYIBeAZTaQ&NON}CwB)G*{fBb6V(Qim#IedTnV2N zt9C4S;9MJaspp+<1|UW<1lg!DH2p_lQVKL=Vt%@VZH2k~?vl%V1AFcstV(Ui0VN-I zHM`!wm>~q4r!3OYj|m6H?CogruWO#LQ@ldw_fx;ic{tZOtgeYT67wr~)35QGq0o;n zdpz#LmeU$xmWeW+tbS^#3<*M$5J_Q@TY;AkV6CuvSK>ofoz_9OkB`s!fF&!}p)mkK zj)2C6O^^p&?ZAKxwgQR-ySX)MM~jQ`c;J>(%vQoE5slVSDPlrsq9m))25%+i@o9L6s@#8-1&ieGB=o!tL!((H%NS15z~r&!0V`Vhd~V5+P6FXZqQ?fo zE*O5n&3nk1`}7G#W_L$NA{2~Iil+n)+JL5vGs5|#0XX{od@_h3gd&Tx1*mXm&q&~U zt+eww6`&J-^BWfsXaB3s$ijK0r^Zk8GnF+8RUu~tND-1#DS3cmg$YJup}*Q@O461h zOIPg3qU9N{G6~!Ja-2hQm`XsAg6_#16sl1F>adehNu22<(r@2Wn)WlLCvyZAVj4E^ z6#1OcBQY5_ZF0g0=Q5FXc|5*eI;@A8CX*gb3$hbI0ZP5FWR&rk6;;CCI0mHeAm>EN z9jQS<`KfBa9fj}>#kslpGB}eN-*Jbv8Eb1)#M#4E4feZ!de)b}AaD6g6Z}LhLFSgl zJBZ1H=tFyv@)ro8uRnA1<$woPR}Bg~Ci&}l>{0i>Dzm{r{Zb=&;YLGuf&=G@R? zKheSER5$cEZ1U1A`cKf-aV*sab_H^JN!ioRT_s$sE}p^;VasY8t5--+*CeWdud(6e z58S2DdRPoQF8fw{+juT-J+-I1=chNf?>wnks(<95=sP2{6msGxxObnjRXn33+FI3dyM>~XjWktje)QX=^4{8do9?jh zBWfIZldn7Tyqs%oNIkC<7-$03qc4((?*A6V!)%M#1e|b6gOr+x4sY7sm9R6EjZ%%L z537_#o2ftw-(AR7^Jf9UB;>8-%pPl+Edxv4f>E7}2AY5TGBP0KrAbnN-s^JYG&yLf z%c}Oskcsbm@5}*1P5^#>&i#yF|_BsiQ$iKePTw2u>YKBfAyLBmqzS3^x{)L z!wlKV$!6}|%qd&7qn%y46H6zzbjC|3_cm#7?OHi$ZZivfNyIC7eRC5u)2}C>!fV-Y zFp&KjRa%kTc0}yd6Et#P(MO{&DK^<*(+he0HUYlO^B(*1e~yzQjsMEl{GRm%88s3> zl8y;#l1gJY(`*?)xbrvkz_s|Lbu1D|DuzevzkJ2D)UGt-jJhOYIS^xI?qj=e`FGu$ z4OW4AN<&35F+`0CXw&EkzmX{oLz~KfA2N+gU&uhSVvX`zCXDQT|KCcI2SQpClZ|AP z10CB*^i?dGF-QHx*npT##Vc0-!kGGG2a`gk9aY~jtERb!Oy%w$6)5_Gnt~G+1Br^b z0;moH8M(-@%l4>P>=2D};G*#N52Z7Mq(6YmF0ArH%cy0gQx%R3fmm5qAgQa~db9*+ z{i9K;j`Go{5aelpP27PYo#2NJU4r>9PN)r!!V7&xVD|WiTOb zrKM2=C#H$|nWMKBiAXneWT#{#VddznIwGs-JH8 z#lCE#+L(UZPd9(InPaE5$=ux+$Xo(%$60|VWEfNG4Rxq5bLue)%9Y5N(U)UkrdYUU z{&uLEVCXk-au>_+enfIV)g7eC zOW=6Qq+>Q4VC>afMorZoYpCa56kM#WbEQwDz;H}oD)QxrD9k#~AUM|uX|jP!PsFj# zH~+7#D*>mn`Qq;7jjY-CowD!QvlX)MQe;hw$R0{vl`WColu+3s3S}$VlOkIxWh*2~ zl1h>ApL<{K*RB8iJm2%_%x}&)bIzGFXI|c!xPW&{T)3A4xYUZbVPhi3do`K7SSFh1 zJ@V51GdypmbyvF`4+uU!T2e8|NiFDj&5)MLX;b8FH)pHh6K}{$bC>IXFx-x~e~!v> zV1#_LfbXne=9AFf?DI~E+SdGikxvv$+;9AB2pm<@n9&|FiX7w)pR)}auvfLkjm>4x z=6*S9uOaq3b0m=Ubl3C(%W8?=TcZSMk1cHB79Wo`@5}nJ{lL%bB@-`)st&st9)3fo zuAS67D(E!k*zK#!bd}YCb*QyyF0G%kw_ZWpJ#LxGVx`~Y?tP8WlfO$3K6bfGBj9|p zOxtBmcQH~Bgw_xx3m*c8^d zOwNN5)N`xR3dS7F-AdtDb>+Ne$bqxoEuXeT@>!T5tE26?uF4AnqX8-w_nfx&pB`S> zrTPBjB7N&is3uK!;!mDWT^^!>RRxv62k6QV)i471aHElaa-HL6zWS$K2-}!&c9<>>h49C zryn0H96w+iFeW6ECug3TY>mD2ee47_(($XpC5eDEkx#9^J$afC^eJOqi zf9F#6mD%(t>@1sDxy}W)4<=R|l}gpGdee-p?+J7@hKN0hV;S0uGci9-qjEadbe=Zm z(80%Q(K>e>7I`gnf9%4Vr)N3{9`2~scTlZRwKA}_wIQf~tHZnoNtldj3j=cQS7UE&7wPKq^tBL!_cpUd`n3l^D+UaLu=ZYwrO z)c0QM8JfunJ@&lkRk8e*BiTh2GcuPhKD*5P(B$V7#lyswNpHp@%@13$=R@Dih%0*B z?NFsVtIE1h%@6lI?78fchpv=&rs6IUKPM#A+ImUjoW@tjtPV0`>xLEfjOiy56KM zf6uhJ;gEYpi|hxw{LPqd+~>ufsglMyzoLOb<8C^bI3-5JX`C`)jPW;{lvTat~f+8WixztYo{u079llV*U%D;uwN9uGx$vQ9B=pi?3JHdu|ojXR}oK6*e%m|?}Qz-RM82|k= z>s~>Id04HAO!`yNy8#!xigE4?<*psG!Fv}+MFWpFiQ1G}J&{b;xt8PNSvFtkGN&MM z%qn5ip<;u9M#aSzQEM7z&cuOp%+2-l?M4|gCph7G$cwk@E7VoGgwj6N`nliS~8k)uT73MWo6RteTy8#__uGp~Dkj;`~nGuIx{x`dB~X zFITHReD?+$rmr$T!Z|zysw><5cpJ_ zs6vg2Iy9GsVW?61JrPZ}COs)$>F;}CJTyCYN0n3CE9L747N3~^UYYDInYlVALo-t} z;_1d)lP6VrE|c=em5c|S^jGO{vJJZW`zwo!I?L>=6uHcY>|S_YiP-2*skf+?5znqPdBvd(@pH)^3RkzSwFn>85(I;;^?L)r~SBl zf7(SZ9KPNr&m4N`&KCYP!S+Odcb>cl8ad{DH?#?TpP}3TZNeP){@88q zL$(!7-J7Ic9%2JthZ=Wy{-`txe(=;M_ouXXLHyAx=hN?TGTUmER?c4elo{7DYSUgc z*&Ti2@Dabe)2FYyRGWYucAaa^Es%sI7J>0SYWDa~j6g2Srmn+Kl3sE=XHF5T3g?^*J;sdgX_o>ZjmA-{IR6D;!ly z+3ll`SDCKb+_K1eC5YMnoG#te9U$Axt)>t8TsM#Et(w@*<;mXy&&VdHe1q#1r)>A| z`zE0@xBK;hu~2HgLb@WOT-_I=&OP(=UNj55bDrYi`!q)5F$^7cC)!ze+rCyVXg;7D zhGlr|e#%9VqpT#{&hhmSb==34J2$mwXO3(dR{-u@%OdzHq>N=$fLvo zYvfinosBJ9xxUrIgosVocWN)K;}Qrk?xwoQRN_$>ygP_lrD*^wa*&^-2*tk&(EUS z_G|4jyr)EiY(I}3zLl#N=;hb0H&_|i-c7!Ie0Pz&u?LU7p?DAyD&*mQYS=64>8>Z0 zA3DQk0%^zd4*Rni7|1bJ7dn|c26lXU737oktHCs~gUp=(HOV_i3MW)MDm|9ha zT(f(ArQ>a8c=GwLf`(O8C88HgaQiFG4cRo!tUzVzp%`+`@(d3lb6L*~?JiV)1>Lcq*L{exhIsNN1 zvv}GC5Xcyl4v>2yd)DebQ$CsJC7Hv-pqBr!4Zrwk3UWij$i^llVC8I;cP~;?%j#C#^2CDgE!?XTtC?iX6SI9| z)Z-WNX2&;M-ifP{-@HFj<%oQ9b;LET;$9nMR_0F_hPvjyZm1A{HZjlCRp#aWvf47Q zLfm*qeucPK(beE;U+*{pZD)_X6{@q5J2MVoP0fxUTp2C2?pdu;nJkpi;A#3CsT~~C ze>CLB{nn+2YywnWgWM9>aHpO+&c`wOxRgrq+yj%@5v7{_O_o@N0I&+l}7Om{{G*&fZ_kunFi=-G(jq@haM-^rZRnF8KXw z^z?z7V<+8Jnmivzc5k)l!^*5kR^r^Y+TFBa7D=2P3^2J;_N}n^7~gL9qg`>&jysj5 zG2~Y=A1P7?9w1s5xAlr8NO`byt=sbTrF=0&nMsRSC_vYF(`If=&=2y za>hmyeBb_+`k~FY>i#!#Lk~JxhO1^Cu(*X+o*$u)J|b^g(S2;uona<-RAqy6Im}{dM>0tudFUce2q^WhT(@?9Z^L+UjunS!T4N*nk4CD2`-iF7V#3;7#9U zSgf>k%l!~|?AU+jO>@YM@~MSirawQ9oVqRGne1?)!rA=6eU+IZuh&0I78H1@kH%8g zyWJSj<_a20^s8BwCA>zO-Qx&zACBWxPniI`(j#Jb{Os`iM}yJ1~j~% zReF+aJAMrwI&HWwMK*bSuSu`as|!+Azm84u@bz*PoZY9iCPQXn-F@gx9`gHYU;C76a1cS?sYg777%S7aot|0U)qXa zTJhA)D)Tz&!yzdKyt+C0_89>Q#z!u@7U$VC`?b9Ot^e8i_>)`DEwfI^V4`~%B$v*} zSojC@rZ+Ob8F}SP8_~({jya@#50TDVup0Q_v9D=qCtFjzRbuWrO2^viN$i#d^@+yTWcB*u{QHu$3as zO;6?R>G4l&KxB$S-wC(4cvCbeu{t`tgllH8hwCVgcJa%P*1AyV+RLv(JPa?^C*9W$ zIz4!Yz1`hx0ZtNO=5v5!11ElctM`xLE% zWpDaaK-&}rKQ8#IeAad;>rM5^IXpCuX@WhyWo%34f|Xa(qZFG{X5NI<+{vKr>FH38 z(fm?l_?WxZM&q@g-|xoMg}^%}OML=kaFeteYOj{%J`R2^%NTxEpA^HZ`971nQ+@y2 z)T>(;$r|aGwWm}}qi6XeV|6~bjW0IdJG17Px+!pHE}=O|oWk zsh|8B*z@GeyQTa3HwwA$7BRhG=wx84ut^_T(rM&Pt;do5alGvbc)IQNX^JdH^uR3L zUlgy=Lt<+9)+kCSQEJyFgprPL8=RST`eqYKGBQ?Wpu-PuaMN0&psQhEK|E0;2<1jb z34Opm9;6BViH{diK!393MKm_!PtjHBtCZZK*T2>(z9Dfj=LjKu|+SGhuB4Hpap)-Qih^#3vu) z30n9N33L!&_z*7anO&(lyEwq01VT&bO8xRo@CkUTy;}kq86&YHp8qjm!H)meC$snbZ~@ppPHfhAQ!!A1Xl`l}aDQ4rB!o zT9m3SfTzZ#(#P?DT2X{n06%o)tz6;wbNe4d+p&>Ijk}N_g;UqBfYCpm77&5}qk&!`*6Q6vH#DfjE00DV1X-Tb^XO;iI4&0pnWOW$+g_0n2j0y?M`%i$IFv12J zgrKsnLWme@i+CZ(ai)jZCQ8uE(o>h`8yrJ8j383Yd|ncvv_MoC;URdVfIr(8f2ScQ zj2%f5vA2P%Frtpi$`OXLVn(6%oSMcHIGuPHHZeaXd0rcAOo!s;_%# zmZ|1?8bmiS!TS$M1c|-}eIf`qh!a7!5yoI`{*-Pvgi^!9=oCqyG#~L$U4rJ|mBz>; zh?;@kf_~ zF>Q+nCR8C_5Y7lu5;32d{=n0TA#4Qv?5Jz01;l$nJgFy-uo3ax|EP6qRO_7-91S^~ z>+ow4;**b|;39V04C=+!M_->I9P<>4>4P34HG2df5l;oK@?mH|gwndNC@%XU{7?ul z7a1A;m*N{wJ|;;B?Q~ci;UElS(!ng%7xIsXW+EN-r>!WoN*v)MK!v4cZ>dABP4Jg$ z(i<${|6v43pipx~3@u=hK)49_^QlSoc-0YlGBODgys+#acwY&G9gU1Q*b}RwNNnRL zL?7Dr5#Khhg0^u>A`%1!%pHP zdi87n(doS;!bXstAq7R3?nIkfcVgK7acWEtUo02o)}{ORLUKzV?G`JA@DMb9q+o3s z0gY(D3`31XIF%&|<*~%jgG|+RSN^`8cdivGJN3_uZMQ}7L$>f^BYesrq-#ovuxr3l z8m8+W2ekc^^m>Y5{NS^C0p6uD0%ws3iRc4vDD<^7!i$>wmoyx+{9%+VDYHQ~m4W2< z-YD5Z7NG*~We{fYYdeCL3g8wVu;fPyP(pd~$-=gnAYz*gN;@eU#{Lyy%0*r$=xpmVnY3Uw208Q%- zXE3Qf=#$Y-+vE{mg4Z(lTbR*9uVvGak-?j2h)+JIG7W`>$s_CpXkF6-%PWwp0E~K4 zuB(?&XdC28$i^=Y`S=e(7^{VJbwl}lP*r&nrgh-D0-{5ZeB)Gj%Nk_b4}qjB*sBc1jRj0g5!z3n z5>?t)5#dJ7dJ#{as71+*@E<671W(@e1SQieA?VVYvx8L7ji)H|%5w}I z2v*w20ZB@Dr}h5P)~y6{8NOx#^Z2&d=fpO4!g@on($!=paXUDmUC>ITiQvF1(5!-p zfb-keXW&uK__QXd(g$cDRT2Yj9z>y%gBZ3yVvIo?-S%(s9AqdNQcr9bg~*+A1z46EedD_euHz}OEdMQDQPi@*+7Lep{Ubqw7p1}DIf0q_Mnw3j;|EKRk_*%~wCTf> zcQT>meN0%iq(TQEtpVHmSkbm=RxC3icM=R-d7|+^0hBBTYj)D9wijH}KokY>Q8>D+ zGt&vbJoEYRkK`yXn}jf#d#6ACln2g#byTBe$aC_#=&Vq*GHSkfgP`+Ip7 z0ZE#O48d&NK7ndN&`UzlOQdFds*LBy1Aal~1n6_Ukvs|rP2l-^4Oh}cp_V%~=GW03 zP@|K&C^>J(dNHHVcZHP}ra?zoMU#faJwsIaxg8r3z^DZ|Y_=kj5g=fN-2}|FHU{ks z0hl@vriDllEOuYOX7U{j@picL5+HGC$6hRG--{Ils`BgEYnz&U)K<7g*~7|(6#EHm z(MDtm_AB%j^~r;v3b>Gwf-HeE1QDeAPiMw5L(lguZKtxJi~rfmZK03y5aplMhEgc5otoW zrgN_q&_d6s@cuo)na6;YKEemGc5bXH_u)_2LsNLN5U|{ZARuW9i-5};yOQXo(ru2= z(rL6aZrgYpDkYu&Qd( z-q9BGx$7X= z0M>_|tV9r+mhy88n!MnERUkjo8M7_JaNT!a4e~uw(04pAeUa_~x4*DPutf3arF@kZ!V1zId!gJqr#WyE7!u)?MRtB_<5Oso! zlKD<+Yrr=FJDvU(XKM(?j1YEEzQIq4C63hrdcXy$1#=Pc$;Y&Vw?>FOL8!oZUNILG zTKZ2x=mO-75k-Q~tl?NcYX~xdn_K}RAABiT=ZXsLbtR|yXDLrw(~&cU!nnqa?#r26D5P|KgN#Mnl{yWf`_^!bvD{g;0&KfEYB2eI^!+g1|nUEptmY$v>^ z+gy8m%@dTFAgtIEC&3mogcqbrA&i89tVD-$;$O)OKw%;z40tn9sJtn{iUxtPDRh#? zHI#hLbR*<*@Z_BZDA}oiobI1tUv%5y&IqUSHG#i(a4Uep3{gedo0!2kvI7Ur5GjHk z&#c&lT^;sR5x7zTuv^7*XP61fi zLPIlXQqX}T)(9s->y(gU+z-egO7!o=W))nwM$}N}j^Y_;>ixlh%VuNGt6>A%Dh&Q; z>oK6%^3OW}1s#nFC1_d}xK5K!#W`a_-wYF|7rrzkIMl=-j4uc3XPYefJ-wp!;X^_%q`hVA7P(|GPpl3Bz|20_Aq=Nf>{|;WZe1 zKp69vOHILBJ4BTb6ok`MfS^50F^biwJ+#*-sDPsU`UJY0(=@dYCd)R5zsuuW(?mQi z=$@va0k`cpW_5!-9EjPkKLE#mQBeOg8sD6@&Vw+&SirPOy0S{F5&Lohr#%~dz4yS- ze!U3+^UMf8DB82}I;0*?|IAFF%lubdu~pq2z71{t8~z8A5%I~#L;^(zL=?5UlLJga R2SJDfqO(bi7sd+N{{Z-|Hr4as1$Yz4W{})uz_SMrjoq@9;1Kifzu|Mm| z`q^@;g(+3NE~pG;_NK3B*5%5&CWcQ-GcV&|Dag^W;a$N0Hd?O0~@X zmnGNwX{t}-%6c`{acm?OW^kx1*U~vYeCUvMrbuPp`Y$|+(MT&S(t430Q*+{+x@-AS z8=rIIaIhYP@T68|AK;Ot3)s~$WS`x#JJ*uddAeI?SBTqD-l@PK^s0#I92R9DoJg)4 z*8S-q$x&>mtLI{WtU9p*k>xtOZ%BCWp-sw#;ZrozIroA!S-@ z+|E154|;1p2sv@_GJJ0}MK?@H-Zt9L^>6@d)6BJ2Xt3`|{Kw+60ckGxr_{WQ5-g>c z>&7wkJ=ungbB{|s)u^fa2 zzP>@iILbn$5{^7W7fdIWB3&&$cG&iu7Dbu@Jft2_kyRHHfSHQpZ;4Q=MHw_Sku+TEDZ^~XNN7VjNar-}loNv8Ty7#<%xplVihfUjx%~7G z%1MoI5V1}m%?3t( zGy}wjk%a;#BGmc+UMM(<%M1Epg;XPj@nba?n7^I^Y&T1B&!1-Ki~dc&2>`Fan8sT` z#b!Eoslg{-00|g+=?l;Zimpi8z;LP8J1dfYPk=o|E3vHY8{1zR2_XP(J`jXg8-MVO zSMV}8W(bs!AM+f@S(yL9T=@-hPQq=UT5*r>6ciK|5Qt8mHFEQx_cC^&XWMJ)V)Og$ z2swWVHVc35fY{*-{X&t^4oDq^5G{{o)@Yp?6Jam3eWB$gGOgGSV#)rxd2`0C3;M9V z(%d$!=v@th+=>W}z6BPO$Pxd$c#B;Ah87E$<9hpRjd_G@q(e-$k;$7J&$|KiUH>NB z6SLv%P;~!e_8gY4FC-A#eu?!GPZO@1RlEqdUh)jjQL)f{V8vWP4%inrk4Gm*dG| zelsFAO-49i4*vaoxPs>EwXy+eU5$}fFNp7Q-f||hp+oKaWgJ3`9X|-S!cTkVxszbM zJlAClak7qsi3Lwf0nGRh?6hmJVL1DmgbtuRk&!lOD%$zR2Q`m{w59qLlb`VPOUrE9k zB=FyS6&X<+z6pE(Dc@k6mm>Sv$<^Q~pk6+^Ug>}#FOp2A2 zl8otnlhf4SL9b~7Z`R7#S+BSiVYWwZ@20DP+JP74z!|WHE>Ai`o7EjICZV^AJ3h`XX!WH~tkiXQ(`0c8T(xrzy*+kj<)plc>=sZgYSgOdNw6 z!S|5j$BCu|0H_>@b0VL08I;fy$iMy7u2BNZ5@Zbj5(`mUwf+#RD5FehV%t1#p-^^Q zx-(TNtt&7LLusLRf9j*TgzL(MVQQ2)3(BKkujA)njH{+P$4M8a@nAU%Hncv13^d6I zO)QP8_2?hc*{s*}Hj)U3h?8o21_TXR9uKS1DH9H^0Qbaht5)C+45>BQTW@#zTG=5t zNp|K1h-G{Tbgs)isu!Z)Z@lURaje5QkabT`v-;efULLcH9`_tpA=wfkP#oayNV(9? zFeQpZf+uE#H|KEp%&nURQ{bZe8^i5qkyi>0i3Xbq%xH%aS-T}{F!AIDx=ns6uujSz z{#|Tefc&{GY3FXAZa3CWBS#7ig)#TAOw?l(u6V?LbV#)Xm*euf1-jSSe6fIQpO;!- zZ1R4vm0C6_g_;yeT?%%VI(O2Vkl)+G8fkt`P~gXhg&PqytT1}4|5G2&NST8>A63C> z5U458QBb`}rYJEc7^A48?$0@mRz$^tgNY}d@X!8bF8A+R?DAWQXLR{(n#_{$f|*A$ilW^bmd3*6LR=)S zx(*lov($2^S!9OG`%3jhG`FYq3#8^)*b1rio3WQ%d*F=;9mz+B8QhV7T47SazsJ7@ zfS;_yRNyv&YG37X-zsXo@fcZ*BK)~uLCrl_-N?CLSg6owp2Y}C&Nf(w1!nIQg>YUX zd>F>56AKbZFel&Sl3UX~=$-CW_YFPogFm;8?!zZ`ms?)1khb}a)&&;%KH3H4X?V9= zu0AQK6elq;GNY?v6w4M}f-*_+^GMWd0IWmaQb91!bth&e*748A?P2^PqDn!aAo?)C z4#y@llbyg)+J~QiNI{YjZd;$DUGF0>Ci=*nuOTv{$7_9mKXP`*!@qs+;Nwf7BZ3G% zQcRx|2E4W*@b8WC?^V@+04^Ua|F5vsu`;t8-~IAM6NZ1sGMf=e-w#! zk&Myv%-qaCt0R2D379f-ZN%K%J8TR39RECyzA%Ud5C@z{`l@0!j2XJ{wlAgDLeZMUb(yTvmjfyQ*(a*Y zOONCeEtJnM0)q)0idz=$ohsxFLrTiaF=V<3`fXksY867;Dm9XWHgVSlol!=eX@*Dw zB@px-Q;}eE=ip2&Zx86OtC?rQ4C#B4e841*f9YjO`!_qZvZ7g)GR;H-0EI21c|T3Y zh2v*xzvP+fE5H>zD;yxU5H96+lj465eyO$}i%gJ+guWj)$5r0YF28}z@`EK3dKARL zYL|B+zi8N+VsGZ*Bib+OaJW`f&tuzd`{qH#4uG8)qYCh_p9IT-Xh+avQf>DIqontD zVY!0H_uL=ogpYb|PZXvA+8|%mKS}H8Y-ac0-L4{T=}KJ`y}uTxk(sC?O`#gYdfr?nt@A5 zMHcmoxdIc)0t%t>jb&r(81ESOah*gx`zNupJToUB1SAe-p0qT8;R_k#!mg32nZ3i? zf#)z>GwFt(rw=Bi8fQ>$yHWyer4{OFqAKyIB*J+@fqnr2sU%cupM1>^&|!j411ALu zKRvvC@5)ke0%nh3lYnpfhm+c{H}eNpXo{Ccggv#2i9jX>;jyOF8CJ#MhOC(J0N(!7g5E(L)V<*mKD7sy6|l78`KFEKD1=v z<{4iN!zFKwG^<)3N4AcEh|-fVZY?cIQ7_|)G*!cI=bx_JbM z`|7l168`xET*rl*S#x+(=)u_qCTF%znZ2B!AXrtmtUeq^0XV>a$t_WSN$p|U`+~cH z`kr3yg%MD`T{ILWQcLgblw+W{A@Kx!ipzXk>Vv1^a9;Ef!oqpjkYZKzwR#h)V2{Yo zR_SpF;>Zf}ttJjH_RvnrU9{?-9@JfF?#Qir(Zv^l8r{wP25X1C z=G$@r9-)2{t&D){tKC~pioW!sA~0{SBC2f0zFX?t!*rK~yg(clvhxD_<|I^6ztoK| z=DiJxrQH&IYBs5&i~_y9E(}nUDdKP^cN%OXJ#j$7^3w@K8+T~TM)ml3DjqLezVafF zJ{!R>;US$KnLE$JHbs7vy4t0aFtpF_A6Ni*QOYS1AwU&))M|WB%aG;wSC=vx9zgb) zjWw7#&&lm9TgE}~Ws_X)?B|d zvl^gs%124m9=x$>3y0;@0k-G^GrglBX6pJRMBiPvIpL$2*PcW^!zs++HQPt{<%3r; zw!1TW2=E$vAZ-;yF1>92AokulmOhkFsGOxBOe`ZTHooC;tRVN!ZIhnkjE-uv13l$E z;r!r5O7T2j*YrvC43brlUu!&#$ZjTBn)u>v+*_w z)5QKCnqkR@|KB?ozHfy8MeaZF!2cH|NS?yN{dYCK<{{2c`K3q-(09+~!wbK@XT)>z z2FvGh5^5F9by)OjlM{Abn7+sr%a7xcSU^A@`wkiz*a5k1tW_K^e)O<*qe{?b^a5te zpCLGCb7nq*R_}#P;KxF1tiSe$lx}7LqJ0?%sY1ovKA@Sq%zOrF8r9}C2Z@joh66WZ zWWOHQ;gQL{8?QbhJ0=B=&O(4Z;muUEQu$xmo<&h)5u0;LcmH|Vem!rTUQLTvB*_5Hq%Zke z2_0M>lGd%yuN{WSeOHWFFc|Q`p(zs|U;h3ZC$6(k6irEh_Y6w}MJtQPkJ(_&yr*v| zM|$Vg#Z!n9&}={sU81hY!~^6!?%-FzUqYncTYPv=q;JJ?*LED8Z~g>nWt!>bs8cjm z>!|FFAqL!FlHyvMq(4}D)8%zEBX;d8InMm?O+p}Au-WdD&{C;nPmj?x@0{y^Kyk`M zZEHVYim0Ce0w#f2+4i8EuEffgZ|muxVX1Z9PBM5n>w1OC@6YJ+eJZ4oJIA#pFI&SA zWwYurvn=vg1Fr!Z_dY9HO?_ioevD_eXL~QT6paMvg6=p6#J-aK42rJZP03bh!M0Vy zFGy(#S6@{7WX2BRSy48=Blv4cuROVIB3EYd-b5&XG6i?1Cjro20-w!pK=mtdK$*g3 z27Tnz`tG+=ZfUc@Q&Mx6S<(zDj>oC8w^!_8-21(}W0R`048(dk3erKYmGk>o#c<%qy5b-*EQ{ z7ek!@JfNelueZYtDO}=%@aII_fA!s7S&q@7Wcy>UmP=lDW7-2iKaEfVt4rZ|o-t&K z=0umVbM2LDD4sDPm=wp_Ik7NTDSN2(*R~n|#|g)|-~paF;z=a5|GsU*YtSG53*uV2 zj%ww{F))_T$6yh&qGB~xM&2AcaFfvj$N>DNR-P?wR2S0^9&z-2a_koAZs?)LO z3*bLI+4uKw#=6rU8=BbmJ9gyCQd9cRXe8wSb_8JqB=RpxFfjEL7DEz%z59O&vs&Ze zO3urxSfw*&{&SnrxF%iRE;$}$ud>C73o$AxGbsoIu;yyOwmZ1mABQ#1Rpo3~8wIG~ zccJHvse4hwUcloiDXCG@H1=z^yK>*EV-n+gV>Z8bZ{+@r6{cf&5>Hh-z{?9~dm)R3 z*)Pex#cvI}a>@(fnF07dZu23Zv~#UbP@Dp8+x|UQGB*TW5uxT-hm^U0ANZ0JS<8*a zVP9FcxlJwVrjI--(HgQG)o~Efxb>KoUC4Ay<|NyVIQh+Uo#l?1&TLslU_R z*e+T|X(`iPe7!xX(LU_#Fr97ZCM&d9IqC1w(>fXE=&V~x9{^apmL1(qImr!*t*wUd zEP2zGM@lX##t&<_%IZXvVA4hVKKbwFcnH^r#~0Y`lRsrj_Y>+wK3}}>K2hv*#!TnC z7s-c)<1A-!1vVS0j&b5Eu=&yu1O`G(pR4CAI=iCkyuA>f_p4E1)9+)1tmOrJ$D#2F z6qwwzIu!M^2mqg5vPA2$OnPvC&Po{Svt=_iOefTiCsn;amTe?!@oThUix> zM(5sxSPpy0}ysXHUXqTf|n&qg+1^*1bYE{BB^UO!5PH2vr>RX9z%t8-yt zbyPRt_9>_SP((SyGuicju-pr-d`gl^29oT}QAoWaITotupyl;yi z{_JrJJ??WeSu@VN;>Wh{{#i3{J_)MiofM_|C(i&rPtFz(EEY|t_YN_P4jcYs;Vn=) zSQoZ@jtmeQ?5%Kqg56Gz`B090YGG5vArjek#PKkfchE7DL~vHfYmz8yS2MV2G0~hr zdqG2S!a6X6ougC>`-z&{kjzqrvo&?CP}r+|%c5XjR~9JaidJ zUOwRa2afu`5y6p2%5fxB|9nJ!Jn)c^P^Nv3i^qaH@zo<>|ANpVet;QT+(OGby3Q1B^WdX{n zkO2U_kACcu$9maVw@9h3`1?^QybF%-CScC-k1|Y$`s=f(Dfkt9nZo$m5eNw=Sb0HV zgXwyoX)g4Z@bJA{0a7NmnFird=o?gP&dOw@;1=>n-<-g3Sb3$q;(z8H5hsxG|6CPG ze((%rir~7cl=S{W3)tPr{>cQ_T|E>X;s&S$vX(7g5v9`>d%7sjxqT@MqFdZiLy6{; z2-djv#tX3pS4v|4Q)tzKYKyVctQv&pM|$p{)r{zVRWAGa53*akBTjj-$<&)gKJx;* z0J7|#)$=x6Zz8`X%q}mj4=>cX#OuQ6;xjI{L*dg6R3+UrF%Eh!l1oB5(6YlIW)>iI z*mw!=U;KzLH%}{(7dnoH$zpq$v!;v69(4}B^J!Ot*-!yy3OJ}P6-2|&fjYzo5OnB~ z->(z%6E7w$-VP+yIV)7wa?9P6`{wnlBWoukcqn@u&;P-F^;K*;x{!8zdaX>u`aO?d z(*J16LGnA2(bFI&yavXX2O0CeUKK!JV16YB5qyXd3YJ8F|1TY7IKLfR?y%oNKn>g@ z(MEHRVP6hx?ogil){*#+drD*9hpO#!E=a0g1$09_KMOSgcc(yC#zI(fs< zM23m$e9?yf9@YMB98PJ6}39~Ou z^*4s?SNFCHRv2yHER|*2@vZJ0X5|*7#AD9nTTpN%gA=Bznn7>*Vc)K=6wf@ay9lhU zh{OrufL5W)TW?>~WNNys6+=B)PsZTJ0VD6&0@0b(ke_k!=a!hF zY@l21Dq(I>PQ3d!;P7&=%_Mz)l5PEk15q8!^e{P|_G$>naa+ zSWb=5Z>HwkzCyFz6~BcS$myDyKZR!Sq}zh2rCsEGq6>1Kl@S1xdP7VBe07@rx180W zz$wJNUOrGbNX00XVLqHstc!!Y^kDE2T@>m0^DpW3hfDS7yK+dO-D@&J^Co0JjC%V% z*E^r-RF}N$a0HQdnYf@ZjZwmcxgXNg<4LY?A*^1~U-3=Z25_Tv8)#R!Bp~R74B4A# zfEP;1?lMnG8w2nFQjLPjsils3V&||oD-kL+`WEY1bN{yr6=+b^9ahHK;~@!I%^%4> zx%T_=X$HSRwEI#yem%{N$S9twgEtZ_o?b1r{#B_E^aGI6bC{wQ7UdBT6)kQpLap9p zWz3Y}<=f9Gh+MBA!x0JgI#1c~?|mV3;iUN+>#p~1AByUOnBg_qc<%DEfMw%7f$zdW z?6-gcYNgJmlRB<1morOj~=9%U6F5~Jalk3 zWgcg*o0$gK-xW@N6W5-)mli3IjsSn85rccd<-8Fh`(p(Ew)*Co-dGssq?XG^j2snd z70_!ZGiLQWGa_p(m`zj+0;-ISrn_*Jkm#k>N?Y1-O1)>%8a4L{+!yKU&RL+ncw$*> z-5C(%(mcf@8Y$WShenbyc`7?iul?N}Rc=R#NR-Kb1vf6h*n=;k!>b%|4}IWTuVC8( zG+v{rgl;b(PJ!q|%?DjaR+I3HZEy|(aQ({wuIV5ua#c!CR)h8zg8I)n1O)TUi?*sS zWcB}igt$QTVhiwxf6&vSV9w`<5+5AaU^xI~i|vj11>y6#Q7vf0()dCe@R(cU;#nmg zc5eJXV{EnVy4yM*M&OEe0f%O02~Vevw)MVi+?W@Q@}XA4CtFKP8ibteARH7Q7@gF8 zGiiOod;K=Aabp>{TDp#x+O$YUD>a6V$1{Im=j%(t!VB)HK}n33ofJ)Tx`8kXAioiF z_DD++;K%jc>jJp;vR;^*DoM%ee(yzl177teHFd10&2=txsI_0+s(#O(qPFLGyq?Nw z%NfXf`f-rvp9gc_)TWEI3d`B0>vF> z)}*yAV-hD;GJFKeE>H$-GLbMuPh4O3mkUXrgrY;e*IZ@`WkUU)bCFd5B5*^qstnV3 zR;e_5b@R1r8dEuXO_~ctn3!sF`h*6SYL~0}&*LqICtfeZ_4qK9t>=}#?=JsrUS?39 zZ9|dFBn9FrB%uLY$&MafYw~z4W>UT#D3y?`S?%`btelnZW3*zsV0{sm8D^ijE>OwAWBjh(h#>6kC}u%q~M zJg>uJYn59h7vAgrR;hSqY@dW@>~TH#3MzSvGabYs7m6I5#_5Qr{|B!a$~nVf|F|AH z9hUta`$u91Sb-iVUOv4JRSPa>0KT)AgPqcBE79wBBbI2J=9MJ?0%roMdTbGTga%Uw z<{fx$4R>l?JwznY5qZR(cMB2hU`^v_GXtj>Je9{y-jFmvp1(G^I-)Tks%ch7w*n3W z^D{}<5?Azf$ zz3^{k1L+roWIHteWkVec@MVQuUsr97&N{Vhq+(tP-*o~cJ1Db`FZPy}Jo_qtF^3SN zN(m=As_tRnz^oT#q#H=>X__!wl~(Jmk923J^UFhNP5Kl;sSANqw~b?~>oMmUg5K!= zj1n=&It>I4dgbvz%!Zl>twr4GN=J5$8W4H5d zX;Fx$2qWZ)7A#*}0yK(tNkk|V+>B{2=f>aSc);IR`nN6SZM?@bNZCYja)m^3&)D*Z zo{9D-;qD|IwqBuddz@)a7#7mKMFghlIdtQ7P>BHaodI2HeSIrn81fxNNhH%JBq62y zZ#Q6UnGkY<3Hw19c11%bqJeHI&TLe-*kA|;)h?BAtq@Meo!aZqdD=Ny<$4RHE*4Aa z{(}h9*WIW6y7YV&mTiMA1ET-|e_!|4+r^PJ=N1|Wia!{=v%yHM2U_8Zh3>OVbibe~ zxpV<8QWBklvau-s*4>fAt%uO3dh5{BruKe>%H$L+ATL)ynxrp-rAb=E@!s4h* z7Mh!PyLZWt&t>${EgERY8HP1YT?XbFSkkgsJ2WhGA~yS#|M^!ENv!M!FYZoh=%A7{ z_Y)dGatc1=rjt>~LJY#Qa#|E~H0dO+(M$n!khjApQaresPX#sd(Zpy&fBtt>DL2;&C;*1xTV1Y=e83rwq;Bwko55K97x^6Ee#7xR13` zHfS%7VX7UOe6c`qnc(%}5agwF(5YG7bZB~SUoZ`@=^)P^%v@B$f}QI3sF+JFaHarj zgpP`<$F_g!O{a#g1p0P=U-dB=*Ln^)G3EHH@`qT!;W=chZ1FQXrRFI1(BDhOY+h#y z#V{I9R1}QemzF_bxlldwBbMMm*$V;7wf!|5p<7Z0+*?zVeWNZz_-L}6OL>gil^VaA z-zw0hHTN!M8n$_qv;~y&zY!RY(XIeA5^Q%+a#eV`G_E*ziACLo?qR7f1a3hM|8|m7 z4WT}i2apa}YJUaM{Djx!G(eKL7d!!|23R-bp-I$DQH^ar&@KfRok-eviSbR(kYk#; z7W#q_A((KQGMgbOM>7oaq(F~pJi2W1*2JEM+?mrWkE>Vqvr|Tz9$FuD=qCbFP)2V; z_{P;@L76(sk0q+=f>g=_o_$gX2THzqax$-aOEY@d7UbXOouHqnp%ZiSxf3c@j~qqWpNZ ze%WJ_eFgd2*LBV!yIz9=pB?~9-a|7kqC2RvnT01drA3}=VU_ho3%bS@J%1?u>{*@$%X_gk? zIgkXI`{0}-3TwQ>eR`Vy;zB?ZnTz_)3NvRDLPN2k`E4O(&H}z)fKW4q;(^9fat~>M*X0um;P_ z;;H%;F8CZ}I{F!5J(UJ`eo{e#FRpU$9aIS5dlNJR2&EFGRZcvR`IAway6QF*~wMRktoF4c%;>RRs&P z!+R?kiiQB6v%}Ba2k|HReAXDhUw+*beo-WoS{e~38zzKly$O04iFVtbZU-)6_+|bc z>l6WazCK;NEk*#Evb7&?QQ9`cg9&TWbW^W2r%*8pLT%_%q(w);<%E%{O}1Q!HV~&* z2}$^rZCbpxj2uVF`Up6GcdWRJ|G2x(O-O*z7;)Le<$kmmy@<6#4S$(6<1c!>LM`rs z-0~MwUBrR^NoXOIET>GlHalcb^UQr`-(!@0@*-ao{0jf`P0>3V2RU)0 zhm`69jCv!#_X@2+82Z%HP-7&{QoUXCPB%y^BvIQ$Sy)j4cs&@ zwljjtMTa$V`iGjiuz1hqt$?lS=CS3~*ef-TgNx?Dwktyv!&DLBZ7Q5)>`W?dI~!xi z@t72kVJx<=$$w`<&*T1nbGbEiOnBq?6OA^i#d$9^cgtaUg1~4gbTpuBfB)k78$89{ zjTHb^JQehr=QN#Gl)70KSg&Xc6DPg@aJs0}hQ>=%8;0eJU9=8oKE*(yg6QrR3#dz5 zkWUS(4$;NZlVyzQvbp=)ql$GUts(A@o;!}1VhnFR;_TPpbtP}$*QB?*Mq9?!w=*f| zsxJERI0ftXJo;KiT^*a6#K_94-wN!pzXUKEo|I4M*QK@2k4p@<#J&_>hjukwg4FA% zd!j~ud-qgnd29gh2@9Rw4W-|hW{tI7H*3F&(rFNMv5KDLbco0j+Yqmv#tCh)mCcK} zaQb|Sr~5p+zNY07(mBZen#w6ce2x%pwaq!soMH_d>{~%h^&n>~x$nJS`Y(L3lD4b}Qy z(@MGJWNPXEqrw6{f@boN+_5#k$LGdTXUiO$`;tD-i&RE~5|Hu40jJ&E{;hGUA8N8>dRE-Ej5>z7k}~Om0nqF( zhL5DWQ%pC)%+Of=@ezK(>lKy;hLUe*D^#23WO9PACT+yfTQ0`ysi*@E3&t!e#NH}q zCB$zlwCokOoly2LJE80?XdMK&KfHJp-3GkL<$Qg-Evi@pBIjlo11~pB&7Gt`RGd7L zq5Su0)CZ_ugitW?2_#%vT==W2fOmJBYj&CK^%L}q4|3~lmtXXR*A_dnI$|YT5$f2o z2tDV!h)J8!+#1pf@U&(%z>vAiOL4wFSh;zPFEGY{|q7E8@x$))VL zCQ+PzxyW@saW$A>_*M_Y_ z&$p+@VeE3BwqVo3S`SV|df9cUkP&i@GT+R#ZRVQ@IAVyo^mGdi*XhEjdSY+*w|10e zbB79!DtbfjSR2t>!MUGYNRrSWHC{4@wVP2ao{0F87e?dK0q^cz(7kKq$~kv4Mx#HZ z5VOJqouMP3X&EQ2WSntBWYeS^NBmopJSE8s51=)#B-R#xlE{-R?<#zN1hbRtegkLup(iHKRCszkjvh$(}xjoBRky1IFPjF~}jo z+c?XcksCzxF7Z!xzP#S;!-}^Q$98*W+v{JoYV~)oz)^nv8>FYo8^IQ{*?h#H31%*w z1NeYbL{WwUn92=o6;!)qc06m+6$TOD!S^t=3x^9*aam4?$=66jEtbPWm$u2Mpk@ko zg>q9WBu(|v75_aWSkBNH+C?zGF4eer*=nWHlJqN`=|j;Mz7%2`sK7^=g&k8vztocc zln;OF^&^T_OtT<~l2oP1pt?n*UN*=n2QUI z^I0j>BUOLS$n8awbWFr2$bUB#LrMO8ct##;H2r(PfL#O4CumITTjhe+S2R~s3+SNj zpG9!DPtK5@PQrZw^9}iKo{2+Akb_XStE{-74k5`{o5Ke)pXJ;ypuw+T3ZOVW!9W)a zXcS?%@xzsJJM=^*=22>7_7-6ma+LgbQ~Rhb$9N8{oS#Qr2Qwg zsW|Vzal4}U7q8XYg)fge?JTQk42dtTuKuAy;=d@iW)N3 zlwKbqK({aBgPneX26dzINQhb_CNEFBcy%*5hu!1wg8S=u7ly8PBwV|S_6fIOVwnYv z&eBt8VJ+l{N@Xc2oABXVe3g$GBvVPUVOT^Sk8HpRf1#T~%YArn-RW>bZ2KXAjbZR783nlx#v~ie}{@^9{$8Er%_~} z%q4XsQOwZYGir$3S{mcE`Vc#pN@cG?OBQ^=f)ciew(VtYh0IFE9Gz%jU_fB05j7oG zmsJ^D2{X&GpK)&Bei6te6+yg+43%{xNQhy!Z?y?Qqep3)#MuG#_hTIL+bz)!{eZd! zSP_37y2iY+@ggv_k3Rl~GQsYfub2;K+Urfq_K<|@z#UdkIYmNlmj_j7B`6h-Qj_sv zS^m_Hw%ggJBtO#i*7Cq0VEidh>PXcpI6zwAENthGh$g4GC~A2D%{I0B%PfW8{i*!% z->|=%+3~WY&(}tKbhBa;p+Yv+_hq7*X|4MD2+;JQA=)5nmYng^Q&=f!FAnLg>-g`p zg>sZJ%dBYd-zx-K*wJxMtghTPM44IVIsWE_sJ;7?? zLQz!Cux0qaJ)Xh)kI!{=4BZtPcsGJH&2|L16)p}mm0*IIHB{{32fAJ@@-6=~tAKxm zTc;09wEWYHLJTy%%J`VWElJ(SP2wxh^SYT`rJcM6gr5AsL3XCg%^mIsn&KKU`?kqp z=wH+O`49DWW$C#bq>m3m7k>f1f)$kptu6My4xZNar$BBX?{;QU5aMsEUs*Orx7$~r zL2DPms}Z11f@&h(IETPxP^O5^o=-;cB^kiGYh0{r$eqt{ zQ*sS>eUk|-Px~Tw6E5wB{>t3!-0@1|PkSm1@VM2o%EPhktn|3~`}Z2tlTwHYt&eSrt4JO}kyERSCIG z*aEt|mB5G=JPVW9FIqs+5d}`^wlLuK;DGRhs`K1X-uLd3+L#iWCNW1E`Upg>?yD$Y zU0C~2Qls;ar6_zSmY>1ihkId-f$*B;?BP-p`k59XB9j-prB0hQkF&ht{G zylbkD!K2?UbW||#G|9)~*hxrdir~vKyrzvP<#Z4Ha9$vDR21;sy`jP7%J5JQJAUA@ zNk(1HM{z;GeZv;0H*VrK@4p~gghyiceDSjTgIzYc2iZ^<-aIdNzY`(7^wF* zhr=@ns1>s>t0w?WBHkB$FYBjzG6P&)1Bj~O z`N5D6;HOo^_hHRl23vss!*`FBC9wWb4>M=bS1CtW)!i`c;da_IH@}p79HEf+-irRCSX@%~PuEX~7 zvUjtUw+qc{o!zzQoC;+jJ%~EHc*^t_D!i3G~8t2{bZ2j^~nLpA`SUAw;VpSe8XRITa`$94K2LG=DBTi!` zevu8W_n7_&M)tB4tJyz_qQ=W#AKoSyac^gMakwj<_`+4esT-zlEQL)u>&sy(`Ztg- z&-p0ZIq1(o|L;mWj!BJrO^x5phnSsAOAK=h1%UhNp!B+WBxTifV8FzlN|0gUZS+5w zLJzw6L<|BR3D^fRq7YU?I)v1EuU7D}q~@BBos7AfWto2PkWvprg;;c`sgbjL_Bh`! zN3X}Hvm-%8Q<3bsvZnLX4`KDJXCh~yq_2d_1Ua-2$V^tzh0p$1PGu##xJW((Osp0O z4?w;@LL%d#_)&vjDA7<>3YM@Ey>(f*re#xtep>nkQL-)}YSzh!EUV=egYgi4lPVf2 zI*M=&01`7<34m7p!-F@uz_WdCMs+^U6kszCoiO_>OHTO2zA+)43r*fUD`F=1)fh8{ zcP_@jiV(+S67X?<-uCUxVjpg=?`AIR0BE@+sC{a9hTLc1!9ZM>+&(E*ZQMuf#iq_{ z=p{6eWE;K9N%f17a=cuGGK*Vi`6*!K1_ER461NxUqb_*|@5b zz{wB{avKQF2n>S_E#B!>C%;b~}%<(#($kQg?d-peQ@hECBD6}+^2e`%uZ8@a5M4q8ub1*AH zl2~OGp`%jsO>oLmnT4=`I+N}ygA9k$Dsb_SeycxhqE#uzKQ{b5ND<)sXa*yIjI23o zfHsD5-Me&>!4NaiLRZDrjxtGQFqVRFRjp|9fZ2r>+^d^;1lzx0=igh{>*J2krQ#eY z{)p7H$%k=A;oTyZ{%M4ppiSgXoMLQ1b^3+$ON1PqcIMbJGo)d@iz?JtlWj_^g8 zq3$E;I`eR4rv$nPk+c3;YsuxIV+6(+fG{j)(=IY$TE@LF)w z@HiE(KH@NAc;^?hoiK=>u9Ii*^*WyV6g{H#U5=j!)PV!_A--mtg&J*-`9Ib2`Cd?iB)+{_)^%F zsK==J7s3wJ9<~)>;}U^jjjshM>k@N^5M*F=k!<7SoIzA3MA*V|FP4n#;>9*&bn9HY;uGB_$c# zZ_XV(X#v)g_N3quh}xPMYU5yo^fo7b232(9L@eETIIWb;P5zN)k9N zOVy@tML4_aq@axsKx8E2o5GI#DjsE(l$f1cFW166zc*=jv~%&fE9pD$4c)L&gTv z>JGA`{9l$#*)|lFmh3uP zvw9jE_v#4%?{Bjv(YlyLn+`+Mf@t?$K9W*lL3P6t%G2I?;ftv}>Mpv&Mw@~5e+S)G zGw>LDC28NUPMOW~&T=Y{OL)p`!6y_Vk96m$Xy{c-s35KI!~z%f+H8$7bzWpm!8=cG z*z_o~k&%bqZ4LR9yndoSt?2V<37)3LpLYuDzv3{`$f56z%_d(aOCZz)paVukAruP;MNhdz!TK0 zqhJUDnA~j{I-B=pXKHPzDXZs~Oi#TRa!e~K#|trG`@wH|5hTO+zULjvg0Q`X-rGYN z;4kw6=%dUbevAM6>&3H&@?th=bbogWmo;L4S6vv!o=AyUfarVWXw7&6>Maj)KYHv+ zl&zMm1v&PAXQE3TL+5oKtib2G9Xt~kFu;`jhLe3ys1AGeYCR6&f(ffV`rn0?5j_gc zjzH-}J$f`|8pKx`flflZ7`bawy^*(00>LG{4vU+7juCP`1KU&fi{PYR zIk;y=^37_#6X79l(C^f{_Jt6|%1ec=9src9kBSc4{yL@#$ zf6fB$i&!-U4`#R`=v74^q}pkK7!8AA((|=kzu60M zVj}6j(Z{E$n8vB%NWyX$^6xkR4NNLn%Zfv12!5{&jVx{4=V7AVrK-ePsYhPtKlR z_hG+bt8L$Ygws~;qNeNd381Al?veya7(=Yb|CryWoE#kQ*bpT$s56F9I6kjuG8-RJ zJ6WxjzxhJu)IWetkw4(U;JCX_RSG@u^MnoDh%8YZJ@iyM35=6D4ierN*t)LQqsxV{ z_j1VnV>JsS9AP1JRP#u1oW+^d3wSbpD{)j zjQ<%ow*w6_HQJs4IoCA-H6lCepQUk+!C77W^-uI$-wICW?I4KjIO|Z#s+Usk6N$zH)?|lI-#(o)S7vfebku_AE^n zqH!dnMQ@$;8tk~!laZ#V2Ib1yrPcZ1_dL-8N@LU@D7&gPU#2QCFr1fIPl8Adp({+X zPE$Ox;rKnR{GNC}FwgtkP%IH#RFv2j^k>!FV`Ywq(T@@%#Vl@thSiRVA_ZZ<9XMk< z?{ew!6$WoIH>5tDYj+j*Nk6H=06#yuRuf5c8%0OcdYe2Kss7B62Hp(_LS#R##>4Tp zW9t-AFc>UOmx*_itP-sP^>w8*S2xzyTd$1!9f`hdWyob>Jd4saU{QC z$ymBec^aKhvg?xAvs!a|k4(hM)25%GYZ4@O1&7wJ8Pwm>OJ;gYcN~O^YZdN5_UdYU z%rqMvUDyV=y@nGery-@rbjJf-^Im;%BMWWqdfmBJxzjjm9M@yAB4x!RgweXoSwL)k z9`$_ve|>5Gc8q@>v|RVS`S3Nw!0{54CnDYHcc`XY4lW1REfJXaYwtdK$ zADK8b6x0baoALKV(^OWAKgh6tbxXAbeW~v2OEx z?H^ai_31yV{08pd78@U6{!yLJFMo@Qn!4ct|3sRykO2P<)3l8S!2iqI^aliR`#V#U z8!iCy_XHK2!h6gxP*Bh~5JfHk0fY<-fY2mF2$26L6h#U^`A23_0yO`TkFsHyyKiIH|i4#Da)B;zlg_ z-za-{c4AH+_PH?MRQZlW{rCv|E77q8ntrOT6`RVaXk4}Nr(;+uMjG)Y^`baAaCAm8 z<9W)gUhff2;>E~A-gO-x+ZIZNJ&C47+yNm2L4zLJ?N(-bR%bej2#~MyGa>9Z|D2IB zkDnhg*_8UL158Vi92i1U{iZlQ1;n`RNPEwP7iFB-ND`%GEU0UzPS}%}iOAQ~4MC0$5$6BpLHK zmDag5;Buj46>VG08c({SBkdi6Z1_gPAji&hXDxYQ(Gkw4MB`1rxXNu5BawY&M(Whw zPiYe6(?C>%4j_)Zm5&fp4-0Z*#{Q7=Vm>{~x@C{8ROkldu;S3dOEKQPzVQ(Z+sPOs zlQ`KD$yTgbxJRGLl9L415qMoJ?iPv1!zn!mxi267D^$i9P9LF(A2OXqrjVZ>hJ;Xk zXsVgN2~XwijjcgB@0LWUDc^Kb&ufInj|U0lm>orrUF+nwaZ)wDz;#hdwx@$7ZEg4 zCZcFFTvue>uCctV#YFwR8z;9T5%DA@f?;@CQhSFoKjh9(thp7#s5d4Dn;L_i=Zn2j zj-ebijd*aOIU1H+g9#?iBNoJ_kt|Px=t6rVWF&fw5)eYsdEyvYlQptgVGLf67 z;qJYmQL$H){XAc>Yje#=_XqY))93e3F4K#BTj0_ox-QxPpB!_ zQI4#v$Ba{%42B+kr(_A^Gb!{Y^KlzE0uz<4iOA|D^9^k;g~8{Ajfs{8BC@EY^|1Ha zRSgP}1*UZiYmiexXO~6W&BD(R85sZ(*aZBn)lbCZv4wdWVtpfz9_CgJP5UHrfp+hq z(b-n_Q~BXmJk9|2WAH1jMARrmX6g1@l|#}};k2ch%er_trCa21#HVPw-M*2Yi(yzx z5BLtKgE9pqwAKR&?1RS7l|oY)+r_8Ce<;QE?zm21o`n2ec@a(g9NR*-&_??Oz}i?g z#s<}LT#QH!JPv#?>-gIVG7UHINJ=GKXh{XC=WzE9g-J z6AE;J1|~$st_hNV(JLoN#kg-?A>CV`j>|6K(o6u?5(i6=4^Gn{xtd1(~Q~%()bl>0zkI z+XRbaFQS8uN2xUW`*i|xKrtoja`Rd|YRs*KE2@3(n3=x~?{C-a)@KH7frWu_#5lT` za0m{=+2k^;mvoQPcevr{+vUE`Lfl+!V@N6}f*oy&#$OgM+Pp&IJ7!Gk`8dvB42`yIoIJmlpHC*dbEb=xgCS4(@YgKEd9dy&4noVDx@^~7eo7BA& zvu~>Hx^BPmG$?jbKmD!98`l!DUt`DEcw&M9B5IQ>Fsn4?c{^N8E1cuP!0ocLDr>1w zw=gIgnyM>5zN4U*281AYnNZ4l+cjD&Q6MQZl*$kTO_9Rz6&pIOQzAd!& z){|UoZ%onSIiRQ$6MWjJyQGe*3Vf+AUFS<%JNff`7){pdtB&BaNxsow9=E8fleJ$2d$Z($W7QB(4Ngn@L2L&eeS#TSqk>>JnAYJ6#vLm&TfVGnX zs~jsU9*?NId=!NXSsxu1tY3M3d5paAAa^q_LS^{$6qQ40*-Quy)}Vqd!#vt^^U+}9 zPD=;=xuTasdfCFLeU@mY_>G4UCqXPDMf^kg#36$=N60pYoj?sExb|b~I2-zB-Z5P* zK?|@wWguXb3O$ckG09})X-WHh`vyWHSlzgy}rar)JWk5 z5m?qx@1mO^ksEJEYg=~z$^`s*5_zY>3_g(*)piU?J=?)dVDO8u3;CQJ{(Z!R_!UEnvhjo;ABt$-LMt^2pAo9o|t>D4O1M6%`=+hUZ|3xVhqiYoU z*-?V;-UlV_(8{pYBj+Bb0BAJ{#6TC6JT9z`=<1D{GCf&tH|RhC6WVv}HtZB?2EPpNIGhP3zT$x3urF|(r+ahN^tQ&pRZ^&; zZzX7;TN?c#NQEB6-Xm@drqA8Utsb5g|!h{}lQ<%$0ca5518m`Ry6e$;=qg!$&6O+I$;rPNeB1;() zO-NbKk2o0+PhUUI!h(WlvNwT11!MxVzE5x{{4hT3)zNwoW=5P2{~Y<4Ui(hv6x58Z zRl*)S`!QgTO0m>M!9+6jD_|HVC8|9^2IqwSp1O#M@lRPB zhapyqu^ge7IcVA`R;93OQ$IkIzQ;qjXGHh*cad3d%pB}PVjPVU9*wzy14fYP*%`7m z^j$J@S#Gyc2M4~D-@)t@#danZ*r-okb$-1ws=)r%KDgE72U#7nHd z%o1Qr&`1e>jNA;{=9g#Z9`F7rV?gUvoERU?-R-#6{MmAzCp{4=DCS3;dD6j2+-y@bFu6r_v@W&@ z0c;jB;^?w{3wl_q(XJqRuKs{OT9^;Zv5H~5`Di|``bTSPpCE<006u^+WK$PV2-t@N z=>b$3z3!cnpTN3cgi3gukHZ7crDMY~pLpoMSC8`>I`d2d!Xofu3a+S+vfv>=eE<)b zxCDhMKiU<3dG_$JY=T1C#h0sndb_Nfam24>$iqL1(6+ zs~Hco;#%MBq>?zS@41nN7TYgzEOl5ixTsp}p}D>ZOo(V*TyY-jWd$7>Jp2Bbf~k`E zGIde+ay4+Zw=nkkW7p-tCwMM^=0h3yyEM?MtvIbj$HLn?#z`N==NFrFEvid%5%^X1 zkwpM36_81mYx%-pI2~wd9=Qyu_DQlA_-XL#%UMa4eby@86LcFLb{4kVV@aj~d*i70 zvn|uHvm%Ri2$_ns5>C{Fyrs||(qP)AN>8+b*tI)xMRZY8&$*jr{IApCS^S@1!Io#H zT|rIlZQ>+y+WsGrjL~`NuEu3{A6Dm#vh62aRB2eSt&~7Vvp=@)?2wDKp{(66UoR2g zRrBl;2Kv7j1p!TlNr*ByEv@$a3EG;MIRm%XzYUH|KJWYgGP%3`HG-Nj`Lyr2d2IXg9S#a=gA(FM03>KiGz1j>g*lo; zEC4ky|1pK4TL12S107mX;=3ORAR=G@LJ}J+RuiW!VCLV;9NrNS{*O#_2FStwCwE@P z9d+@B7@(2;hl>0s*o5H;K!E)ZQfVRv1E~K&65=8MKVp#m9pL_d7-9jrf5C@TJwJDehL!&xyPjla3M3D;ETH|D|rPFNTXPjZk;jP9M; zFV$#vbe_*tCSMl8{fLM!Uvt{=v2!D~xf@fJMY$!Vg9|7h=b3I2 zUwc0qJ}aqa#NV@In5Qc9zDDZ8iGV6b#Hh%4nr0?M9k7YT4QV!O#vDWyI^G$I8%-31 zfF`*Ow~rQnq|-?{dJ@ZNhrBagl466kAYe1hO$ebLQ4}fwL*6M9P^=cDOMi7aDNs9} z-aC&^XD$9cB}}R?wbkW8rm1JHNfU<@nIPB7_-P6(^mKna!F&&mgcX#i^qw$$ zhDH%WKT-pXIDzpVC8wh|GZjiBPvu45akU>|TwzaBBQ0%~%4IOu=-9SQiBF)*`~|lp zBg+m=ONH457EeR~#cfT`C%o{uC!i1!@fU$tkey{+g4t6)`##H-ONHxqdXb3dg!W9{ z$*M8tk?gdF2Xi9v$HxX;6NFNhH*_?ki6A+?CuQ+q(8G3;O96^>6Zcms`D2aXwEh5D zhNX1CO7WMCIZCk4#sZkt(c}}{4g8+QkR)ycG9vmz%5R90X zuE*bt-<7v~lH5DENvk;sZ-k_1_1i;)mExuN1}LaH{N1m;mm98>!g+2F;~>3!>B-~U zhXXr1*RklEcTWj^ty_PzzCLd#ii5;)W(h=bY-0{=Ihb0ItQ<=Xqu?t24_S!Aq9_%22G*tx9?mtqO3g7v2eE-ng5^;oUlDsL0}3fV5=JEn zC@)xH!J9Byvy>T$8FJoUo0O2%7kEBBJO1gjdZo-7UtdG2bukT_s|rCE#Vp^L-)Fjp zLtPh~4?zMr(iQDr+#NdUZ@E$n1A7Q5WR1)OpXHPh$5XTOr?k)-ztmEH3s}ZX0wlTA zE5XI+BjQ*sJ={aAvjHSvugGk?*KJnoDg&Pr-e^+1xF(jI%a&XxO?(;YRs_5&xwHl;rYINMGebY8}C|HAZzVhwM8t}Rg$;#)Dw z`_Q!7{OCDo@P!ProHXx9qolD#F|St+9dMNE&jo9{8g_SRXU3QF7@r{9Z7di z{QS2oi<`#i??ShI*~?t7d#{v_%qb!M$_%#XVJLOnbZ=c^bq~l;@hacl)Ykw;d}^ga z+4qU5M*Vb@4COO9_4^pA;5V{g;zer+1MxDdT>A2!uG-7$vVOrFnnaMci=AUUb`-3J z!uup;qENLeZ#@4n*1Xq8J0ssKYd9Ku{L*BY3&=qHPlNwVB6tmY%W{eb{%0N_U`|-f zCf;&D^k1O7X`})`@J|CD)BwbRfAuU~G5?2g zL+H6-37Vot091cb(@L8F^rrk}fc?L1;;jNe|2D+33Gnzwmi_?z_%~I|0pRg3;@=c? z0bu-Re%gNl9{(}4-U0A||6K`a8R&@v3k9Wz51IJ{#DT={z``~?J^(oW39-EbZ2!+* zfxwM_JMf1A7W}&%MtC6XU%MKF#O5OMcHn(vNcQ|2qCPDM3)7^H4AlKUi=qMJ|B=cd z;Qz6&TR6bLf1~CEz}ntlwe^(RL6DWgsEsg%=11(WM6B z5!u$6g!{8d-{k;W8;`>QuqfKOG`ygaST!3eNE9`Y0KE5mV`QKUoRy3SH z{npU|9c?~K5D-9yMT{0%m<0+_#xSM`6L9^?;|O+&`7EraMq}cxkf+Hpn!9Pv%1Vcr zC9_BA1;*1mzd`f#>ijIVG|rhGXG}*Kfl%J;pwyj0iL&@EE#v10CNwm!=K)VdQ;5_A z*R+aBVfzOec*HnjL6r$CN@FoK97pEiy=BA_wM?`%8pI6colMVEzQy(V= zx2jZlcIDTrlfIFGhqdR;&)oxeNI8Z@c@#exBfx(+&=2XQssmXVp&b~$B}0w%dai%s zu75LpL`Ffycz54nk(#8k>CGht_YE^+V&D}T@O5Qrx{)-0*X>BiP4}j2*%W$qLQt8U z^eyctkT?}pnGcYksy}hZ+5!Fe6JsgURdVNDd2oZshJC$tr3e0mBYeKhP`wzPvnpu? z7K5i{B$TKw3(?+EruYgCrSQ*Mhr>3JTJ!0-#>-WE_n84};A-*%p!~MtfLiXXjvpYO zsM53f<;HktozkF=!B?v43=|Y7IFiDF*k}!SxX#YUsr=CpFnJ~KqE(5s2#=$zorK|S zrEKNJYS~{s3R`j+qvw`BM@m zbZ0kL;xAT+wYe|irN@-S3zC=bYK(I6l-TwB17(UpBP}>Zy zvs|efyM*|E5+`_!8vHer5MeyYrW0Tvdt#12GeeJ&{)X7Xv#jFF%N%Y}fJd3{Y%HDzePp}lOM}41| z3X3hsPCX`$_R7VLt2)3@8kqKmawb<_{)vK4s^T-yG z@JKq2G*U4vRQ^hM5TvuY8NMd(fAOp)qL*Ka+So|uarj|x*5;cf!f*fTRlTa=Kkm;B z$AvE73XTS$M)D7=%gv*gpTL6}*UHkTnC)qBwj_78p%Sh>r_a^Ghbh84=WY9p1gXG{ zpJ3_0HjeFhj*79&)d6cWf|cN2ID5iZMA$ z{boUAz2;a^F;8x(%n25_@vISRUj5KqXNJFInc)Ap4|iKRxlk%#EM|dqoEFyN_$n!9 zPq6dstowRFvDY`-wFSOWoqMG?tiH|`;?U*nvrLK^z+83xV3++=>c6`3uAL(9*#>rN zw7mj0xY>MvL*d#w5WOm-56+CCliDLVoNPt|;apb8m_Z^BWgjdjMnyvZUZI;n#;i;H z#7)obLpj5^Ms1EfQYZoox+_)BuknR$viA4h8(Uxx_GPD>S?8tX&hHb4wlb~O?KK?5 zNmJS+D2*084<2bQRci^<^}5<+8{1bF^4k?pO*8(_I89A1X8Zx*f$AwYh9Ay2c|^n&6@kohMi90#7`t$>;>92P6C*8XLKT<3SF2Ld znmVYQJSDxO8Uq(*jRWzcfvRj6HuIE=6ayvJI1&Sc&E(Oy{e@IVk~+S-jh+XIb+!WO z{%RRm3wv0zxRc~U+})(lVO8JWRN1)84_~-hIE)&VuylR}akccv4L`Q@Es#MMdq^C; zMnv^6_?qD^1D4ioqN<#&r8!NtGeJ)?WAnC3k|v6JdwkG3KL)Dw97Xo`$oXmp} zNbkQc&bdvoJi7=OS?)%f@?Z6{8c@L*<4GnkCYDd(aicatgXnJitBDO+7Vpz;KA|Xw z)#H3CJ*C#C&HC;^w=g!;%Bimv8mP)NB?r(4i*8T`w6X|covc3bQdcDlXy@W_RE|V- zdW&9m`U>(NetN|k&7(Z4%;_LPKS{CaLt+CV)r!;b#b7#D+)4Y|F@-(|h4LtN2r(Sd z>MUcQjQyBMc8#w(L5hNJb;kl^TF)vEvt5zoBLZbK#BDgH6Sdoy32Ep}LB&x-PilhT zj|Mj%^L;rZ#q8JC#~Y7ynq=eY`;pdSw=Io`wyQotARRr$OGGQJUe33;gqoXeRW2$j z+MDT6sHf_l@vS!n`#chE(UwTI>rabs9faX76>c?L4Lu_7a1KfIjg1IOIDQwr(jhqR1==mmJw*Uk%>0p z5N{zo(hn*)I^K8bcJwGuWsz|j5FJjj*qsZ4FS56n52rq1oGscuE_J+EEEC;;D+v!3 zD;{YCzPKN#C7^Z=(SNQ1*F5>xF*Iz^H!-yBD%Ua8zx{-r8-8km)H=CCxuSiE&&a#v z##GnEZ=yVE-Vm=ZF$AYx2>00q{PDz6hGFz775GJFCVOQ(d3l=^ev(MMWuk^5sa(Pt$y=EvgC^w~sURF0oy7|J3t3A^R0qPYm-O*h|8C_TRN9va~yOqx{( zkn=pPsdkF4U4QLj-Hx);$Q}lw=_7o5xpzmVXj0zEmLrdO-59ky)#wu9{JhR-j}n^2 zc>XvgDeQ(Ql+ssJL7{U_DC$SqN)#FRzRFtyOBIybFWON-nba5I}c zpgb?o2|L!yz|%ek2_Q$&r&%&yBa8p|Emj@RhjTz|PnwZ}n?-8C&jjORvI@qb1wlNy zB%n~uZV~4PGZf6@)B?0N7^85aqg=TGEOssR$WNsFU=t7sZ~w0Eg~%-({9oC{o~Eq* zE-yys)8}dw75?vu*ehWj(l|LEVsAn91Gq<|waDL+EW(Ji)T9}2DTl$AN>x&g>6!}d z@b(H3@45E8f8FytlYDOxN_c(8Nx|`xl5!s>sdwA)H)`7}XX^*>>y?`Od`-PL#ieHd z`HiRNPMsH@j`Rt`AQO@BT|S5z&4ED+p}FJntNM4wUdo7Ma1eAUEi&tc8RnNiCM*0~ zaJSY-L=-f_IQ$~1AC>5j&XC9|N~%N5mGL}bN5J0u_|YaS2iJaE@=YQ1eGh4^-Q+7{o(}yy_T?HrsCM?{0Gan z?#t#cVa7qO!79*UN+sg@+8W(z$qMq(Qx#nx!dFrqOk^>fj}hgT{Y8uE?9aB^Px`44 zT0PPY6gEcelp*79d;AolRLjAV(y$gRj#sKt3y!~=6;Un0euA9y*|QMx4X1E{uqSwR z5GM7Gthn-V>#=4Sa+fc)oJtFtwG?DECS{z3eX1hIGAkAOlyLDN>kn0;W&U0RhHi3B z>Zk2^*N$F}NpClueiB*a)>X`Yvlwn{AI8Z*Z?sQgEn?0i`J>VJXse@x*|MC9l#FH5 zIP@ki@q({lxs~#=kUyE>?cVEAi96KRJ4_SS=$`5TUA!+j>VeQlps;ND<%7MbLd0B5 z9nKmJ{3w_k6>cddLRmupB@@CPLwvm#;^HsPpMW)m4(%%Hwwy8h!pi*!Fp5|oKMMbG z;2>Vxt?TLb<6W>bsED2rbshNKsa87DdOB7x-LL+`q@Nbo;2w>iLJ1g^-21x1CJ=MAt9= z(XWQAU>-18iL7bH3HK^}QGMNWJeIRZnBD2#WB#v_aFdo*lMad6>a8c0c{fB2|MJc< zrK_|}RiBKo?Et^~^0!tq^?vhnONg}7pM_ylmf^9rFq9}TH6!@v>+CfW{+w)bJg^n~ zSR-gy6k@1tayDJBweflqb70-S4&s=b-v$p01 zJGWQ{s%2|1%kvvwfi{c{3Vw*LTlP>6s_F7`>#(bA_Po*)Va;Rr>mhsdVvE`B0|%%D zdf?&3{e-4?h4Q(^jUCLXRLFB}!WNa!G34l%x$~c^=2z>t7M>rKKxea$b{^>+Vt;Vs z__O0mLxyqP>^X7~$eRX9Kq4!E%H6=y=G>oLN<%{&raj>dX}S0MBS| z<&yWa@FGcVQ*HIb{?xapdo!5w`+WRdx@lyg)n}62f!-bElhSycywT_QBM0whYVEU{ zBFK0{@rK>=>b=N+-OlzukJ!DX2y{F#A;2-%kk_8j$m*p)D1~5;ax9z}fii@$;6O_i z@{(oIX)s&EU=$6!Mq0Lo4~O&w(PX>FM`b2oBeOxD=n>TKMpVh6!A&YGc)FD?LWG1< z1^FWP93Sck1bqG+w}b&_PjmhpwotS+)#>xjW8l-m9^0aiVIuJ_ct(9>PjHOTqy-yGVm=Bz@x&GP5Ja1)ycc@LVvjM&5goh4TJlLx0Vq(K`F_W z^->a~>c_JJrZ^fk#E0eVx#7xmW`+nxQIRqLtSM6-ak_d1S*A}&FENfQ)sg#st4TXL z0^dd7m(DPQ=JSzk>cf#HY_*-3scG97ZpU7^#3PkGw_C1{O0aR*YB<6&!7XfL*_UeqVC@JZTWw zIjRP$6yG{#MgGyD-Z*~FQIexw+yS{yN6_|{HiiyD&9uqEcFMg;HcJrYXu<)_p3Y>Y zBz9uIwqyKO>{p2|<}CK_?!o2*y8FvjaP~BZViQEGkqLJ;=ptm9y!MCVxOj_hRzeA5Q-eTf1K^>UqB4 zIUf(`xRfaNMd^9Y6nwrv1lP*4p}R29(=3!N9ha)+Y0tYD>|LqKNx%@wL42a^m&#ai z+ScDXe|f!O&VIIcdndiBdU~ZJ@uH4Z*QWO|0OPtJ`FF_wPh@_T7ijV~Iy5h~kq7ez zq(u@#z6e9(LJWjpahlxT1JV8gXP{}#>?iHoiQs# zy)f>AZm|~bS;S?w@BQcg7tXUXiScqo-;6&se$=VB?W)>s=Pr=WIa0&O1yRZ$++1_r z%ALHp`ntIO+WLY=uflO7V0Vi?VLZAw9G)PfKw{3Dc4G!F#9BD{;=u2GkP?xIn4vOW zNQImnR9c3{NoM>5;wuG20;4`Y8;8)%{;Urm0-y=y9S>%+BT+I_7sn=+hWQv#6uB3N z=5Fi7S?1##&TV$-uIq(i^z(#zfuL%YZ!k9QyoNeDPxoYaPEm%~YYV$S7z_jj0VlE9 z&o7L8C%?%#vY(x^ZK(8K#As7%%l7EL^H0D9efX55hK$4sR5JY$3-%isMha6?Na{dM z7!RwSvJ*z1?DBbDFnU=spzFf->yJQ%+j%m)H@or%BenB zB>E)Jf^Zm{US|cu*>%tQXemvh`6fOnh&BI=?*&~n8c>3tOVWt65L=8?vp~^@C2@MN z-jMIwj`9O~#jP&gHCV=4!LV|`Uv3xw`@n3wK7&|+JP)&mCADH{41N7MO6+?SH6LAF z8Hqonff*Iq>g5w5v+8=HWicue7*zH{xV-@ef|P~`a#bBt6gn6`KfE=dP!~l^<0`d% z9SI0U=AHpFQi5=1tnll9%M_zkf;%R^>CoCN_M?vY>{CLj>A(zJv<>IlVs_m0!$D)- zOrm2?D*Cwk4L73F5IDSmDaM4v*FliI_8jA(kXV``wL=+RNQ3Xn_srYUxy zd0!~zC_KT3Ggdu+Mq%v!Wa-tb+*iip2xXCy`>u^k{tR)yPDt-Y^Ctv}b9xge!gyRo zZb(b%eHrhe0hr|scjSWQ?k;AK5I%w0i$0NDpBF-Kk9p|UV6wQBEnknkgPn97QActe zR5z0o`HOpTq?2}T;Wr-0=QT?B<$i_G-TDJ^;gn(=31*m^nT>4V%xwFLj~l#bx{Ac0 zXP0%h@lAPJMy}V<4YMwemNHXX0GqhJGA5#u5eKWAGPuKwS8Xa)uF3VeH5a0Y_+@by z!j|eM_gwz&S5D4qnfZqJm5$GSwDi42a+%Ak=2-pvCAVqrh1AfcIbQYi@o%mrj~Y6& zjQ*euq%oW;X6SRzQzMN6J2pJN52=}6iG-*fQ>FT7l_~O)ivz)FLXKU^>XCB|LWc4Na;8&ybxg-o^0??{{Ju$8G$w zzJtK8YT_Ehj*IA`y?H{NvOp{F%X{x4SFDD#gOx+aHb+k1vOV7un=UF>vQGQ;iL=VC zEqZf2EZaJzt@bY78sRb+4gI2at&65kj8(0y+E>@(&g*?mZVTfkR3~CO?4&xExxMz- znp0XU1$W&W6b3wcp;mfl#Pv@}5QVcjie;Nn1jN4DadMv!(z)RArqrDk6_>TXC{feU}0z zH+he2euIngiZN%mF)L2M@B@d+t#5hEx}hg76)$zsk+ZndA=(qe6B*l~RjeJIK%MCJN^%e+WBRZKqI(v*C;yD;cn}Q*pb&VMHgy7qHz+r$ zqF2nxhS`|HaM^^9`+NEYbn5uof_jic(x)0Z+}rj1U6pzsLlxt%B;?%2v@Awh!zjkGEK4g1a)dipTKMrJ3Bw3td;4TD{d3pt zq&zl@pZW5f92k%#Eb_qDt;3b3@6T6sT~4rzqK`MI+^vfm$7M6dG{hPqN%AgT&@?lW zD}g&Qof=!m>7b5O#vPS(qvO+u3mBnEJYBg2zNa5)CZ=jRaB zD^!I})Rqlb=cOc$JN&|zfd=Yj%TsWLXWg3(o(=z2WQ1+PGOH z99nc3sQY=)ItDJIx_)03#mS5`o@Kbg6v%hBt{&HC&#C|JPME&Dt~1jmI$BYr{3&Ho z53hEu0vuoC3K2H8I%p$)EU2i2|6?v@LJ@KMTH7%L-_}`tJNeqd!eBK|Su2sRWk;!w z;9On7<|5iIWb7HUU+wa`+Ud;uQe9BAGTX=LN^-ydI}03iBy?Ve)8N)#?!>=a{jdr5 zbF?MS2Vm`pCUEBoI!1EA`a$Kuv$(MVwv>H@^_K2JH-39*%vQaOatr&mZT`;JUz8{h z#ZQZIq82Z?VIqFftCNlg(K=f(w3fu{_XaX6y!($dn)O%B^aKx9;O^vdk4k}S8hM`yCiFoR{4Yv>a{ zCKmlDBwQPa3^0NuX#>ge-eZ05-=q9;|As~y+2zmnQ=N0P2wX)&CbWTg;JJMw09^YR( z8}M7#ya6ux;)bCb%8>n|{VX~>^Gmd6HUznD##{&UA)|i6n1GZBd#Qx4ksPHYSK?8^ z6v>IdO~l35h5L6!hZ(h#z!8k7iFz0W6i!2#Bh7LYNtm|vI}N-@1KGpb2#OF%$FB`> z{E+Z;Ny$e#!~4HI3L? ztgB$g*w5MTiDb0lEt`}VOh4exRMYPqbwq$FVeUYfLovpnVvvmqxSueO)qRt&;C)IF z{nOoh84L{MaOptWh6V&^l{-1pb}uh|f~K#_LG+;m>SejPR|lpS;<;y0A4NGU>mvAG z8Uwk#*K`-Pl^>f&(f~Lz`c3qxLQ6)mPO6MYlEn);hj6;RoHDX3m3|~D0b22=*wIDY z5bkwqPg0(TI>iIvms!orc$E4B+^056u+3a`)(= z>AQgW(op0?3aRceOoC?Oi_#OlSN`$oT3OgFnH=&?)JE0~EQ$~?gM)2jZWB&j=Z`k+ zUPk*YNVhJK7Tlufai_%;xAVpRKxZqi9Q5 ztHz=$;%qZ4OQ7^<3fV|R2OBv!t*$<0!X{~yV63^?yl(Qrs`s&aU5KcP9hd)ykbaKl zISpdI1oGV{z|x*>65~mID(Pr!M9&+-l84=`6OuW5!LwpZc8 zrv_;+6ENor!SD!p%=GL5iC1!bcd&AU(?cVly~Ivg)F8eQ(PgkpOe+Q{AaWFy`zLSy zvc+ocKtwm)^P#H+dPZLQ4_Z=Uuze1$tj6tvzPh;*wsm-~3{+Q8V?)khAm6d2A0Mue=m*^M%t8%I;< z+~&x<7E^Wh1>99$m7vTalRmk#fDg5^3GW9?Y^gM_MW{V4!Y}pv`fW5qTo7+~^Dg#% z;2u{u?-k7-A0HoHNlx#+WP`6vz>dG2YDK=*-WfeOHH$nrbZw>WEW#~$j5#8urcGMs z?vgZg`D+cHn2&V<{_jT@B5D8>9pqIX_#P}MyV@L8ooo5@7Wn7w2{t4_(H79rrT!n* zAi4C~NyRXtlFQ5{)kvY)DGs`fjK0Ve9F4zJ5bk{UdQ;Ryz2Je1N_#-2wNLPQNWavD z1FdiWD*^KzGzH$y)zJ~XE&;(5LL}EH-VmunH1}s{qu25<4($DM;}DvGb#21)b8rWO zL{2*b4lTnpb!}=qF*l8)Oaaj6EYb6#>$j99i&Y_4sKN^6Li)9YgTlf9o0z^IwXh{= z{8-1fL^U5`hWi+6(K})<8BC0Hv<&0}_r zQ7TZavEs1{93$-XO_D0&FJm@C8E}Z%mavJq8&9|z~3$9+ZJCyT-A%vw2NagI9s z13ia)hec$!-$XEyrH=x0d%6v<7s=(#D7E7G^N_Zb8vt33>sSHD30CI1qB><15X!KAs zQ!YF&|!MM1{@My_s({twF#L!w}7s?|s?efz9oTeZoEb7tH@@qGd8SXwX<_?np zx@1S}ra`l{QVJkOVo=KF>D9_)@MHIUd=&%HXGc{a;LL8T>KG2mQg@A$8Ieb0S6 zn^z96-DI9KOPf$eRS>`hfr|YH&79bsP$HYiOKWRlbDL6*{^Os@#@v?X}%}d z+&3Sd(YNjAuz40vZ?r#J;h?g7i5X`sQPU0hEiZV@=(|>`*MOrGw7?jrFex5(P}Zfr#_2#>ML(V9jQ^ zF-*Fbl?YFvHw&_!D!x{MDnF~daBL;KWd+l5r01jMA#s}5>UPuofz?*&MCY?jPYEkc z{g%@1($8P>)9(E3lAdV;wu32|DlN-3qC5`)FR5)r>s9X*ILGVHC?8Sq5duWZ278RG z;>VmpvZuE^&5K|enM?!Tx!|svlod+Lcjvky0=&r6swlFMQuwrqAB}lcM6n*Lnj3cDPLv{^HmB$PW`r_sI@P&o5PyHp$Gy8yGcO=@JJp>scDzb?a|SDbDJNBzG{rMy%e1d?xF5lSF>Y9SV-k8e@FhZZ#9l?W z+Z5}Pr&qiL9$wA(j2-nPbZ##3 zdEld2u>5$(Tw;gp9PmP{_(ENIL6jyl=K=S#fD z1wE9sER-|2j2!}X=G*g+Y`Na&C>Li2$<5DI#!s>CkcbA1IpU?F1&F@V)^Ha3_2=60 zKeS5hZS&+#hjTvpZJv|uv0~chbWB=X^=MfdOMj>hu{Wxu!2gfwXwo+YZovJQ_y)7h z_KNkEDsDpdzZ7g-$QK(R23Y=oF}RMm7~CigfO5>Sc4V+Rh_$v&$j0wkN-3K@+%L)E zd!#ssdVksCX-klrMyqz8T4C4La2{`LZ`|RY|5q>-I@ta7;p7&=My8Zo$9;w}ftPB| zpi0Fyqm2@wcPZZDh@=Nc{CNRv9)<%k{rv_PJE(~7sT$?nk0Q-)%0 zvL}y71+r!p!gee~{AMaVzV&OE{+6wc4x(ubWC3qv-fe-Oe>YgW$o4VDsafQksC{m6 z|8;(H`#5KL=jLaM&!1C0$AJJ#Bi_dmjYlt26M)6Y`&L}z1*Ffo*@KyBP7U0_?M^p| zN#}o>WCB@^rl?Q4rx#Hl#D1=Xoq9`@2Ev6<{-N{`PY`RNvlk?Nv}n2EXmo3SRUJ{7d`JFNl~lZv-Tb`2JW+w2~?r zzF@I5GB?)DL(vVPV(X~U6A0BmzN#LNC>0WGVnZqt8fmnsixlNhcEtSXpyJ!T_F(c# zKPM4^a6ruV<7~QUKTX9zbRpPmQgJ*)Y~n&x|3FeI8R)SDXS5c-d(~`HO@`z|NVS0C zQ3b#J3VVAnZHSb)M|fUdE71i=sH#}$=Sayd(M}BqojO%r5LJsZlCq>HRzyRox}0e& z1eDX*@i1xxXE7z{KQ+$Me=~~=UhyMP7?AscXhg#3G@amBg6%sr_U0EyQBJmPhT4?D z%vowOT+qOH)}JWWOB`&nf5imUGxeZ7cZP23G|?-c#%34l1NWO$t(LdE0e_DR}bK+);uGH1zya29NYxl^>ZdHqUDlS0M}ZmkVu=@it2 z^6%zs%wVSP>9+@8z|C8)b1OE3e0SLbjOV`2Z^@LuF~t@VA;zN=5?3;c1@7%_q5 z6aJBc_ka&07=`r7OP-s5h<<7(b7KV{=Rr?~zvf$tn73ReklaX2XvA9OD8@#XgRv{; zlC&w|ysftB7~Bw-c}mV}<*&VnZDvq`Y%DK+;41U~7$Gt7s!wWUqHpvk@RW7|#>aFB z`u3O$ElXe&9?-L4m3ZTsq^8$7*nACF+f97!eHLqo3a5$wF8l4buLc(83*VVb6HLQT znX1anLxdI!KaYlVmG-x|HI{%j;ZLthN3}YyY?PHe{)k*ipPl@vW{A9rR3Ggld}!8I zJq)@e1;HsJ_9|c4rJ*Zb>sY?0ik|~QyNy#c+T$_8Bt8)3v z`BK!m`hZxRm!l1AQ+9&=$;36u-uh86!>A*CaTEjrxIc5MpZ2t5y%t1to(@QK5?Qsr zyFP9ySYka$UtHiF>h^B)L5}H66?{zNRxL-z?s@apA;+!g zMOA1L9dKTcR!jE|vq}lo`Lpa3{(tupEzypkNdF9QwA<4yqrN;jt|V%v90qv3k*+4#N! z>r*m#1cd0&aW;fshG?uf;L%QrEFwx+Dn)77d+V}YOIUG^GxiC!+ljy_BhuE?7xOkI zKGTn~%&s(M!`P5hF=wA`!WT$VWhVAQ*z|g42+!iuIG6%O7{I#m+vTBXaI%B_`BRD& zDnoUwmdPK}`#F6>b7r4*b2)pCXk{1TxR?8k=hoeLW*3h&kRv z+R%%6yJMq?R)AXj?mNl0%J zeaI&jYd}{|!%|$VTm%K$&zI$A27q|@g)mE*rhfbVCy#O8Zh%n_RU&f5y5tG^7^`X^ zG-*NUG&DPs6)}j9+*~Ee1XFlv-lpsb-yI!IKuXAa&)LV$P2SQ_4`(QPY)cx^dXJ)> z3%x1V^3`A8Xuv%1q3e$jiTOr6hxVfnF8lVQJ8gq7o~lQqcy{+7zD+4hN^@uPqt&l~ zt`H`E{k#k`EllrGR1H)r^AJJ z<=jcp-}8Fvg85DM_9T&dR{!R==5j&v#YWhlm^%7Mpa+WXX=D)oBR%F*X?j(Yebr&m zuRHXwWn*+)#s+}?P!if01rJHxygs-Kv)X*JaALv(5~I=hLneKFCwetz$t$?z#cE*) z+Tq5P_|}(^8+Ve+hm*ue`}() z^LbP)xWGKN=vt?}4Ak)uG)OTTaDQgx6cx6<5LHW7^8fDI^fI&45w4;0V1UbDHpSoc z0{*^o9F?XwH@T1TGM2vRh)9K`8TT+^tfk39Wyrn5py~dye^+I%{A3?8%h7oRlW1F& zD6k^LXagq_QblDpYS595=JgAN=;; zhk4GLM^7Kod-kMGfFLbWA)vJX8f4rfi>kRH5RR9HqUD{vY8)se@tXjE=}P{zvKCIplC<25RQM{c!n`kQW< zw;*{I_7vaUIG?SMjJHL{1h=uR6zAUBaPX1J zKL`bau{U843@C4yH;H)BZCHcxp*vTpp--si9mn0ZBNA*8NX`e~MMt_ncwH#zxwf`ve&*o9&CUkm>cCIQf4+u8b>#Us zP#eN*7zo_}smSQ>r?8l#zm{MlCBbuyRQ5xfg*f;JT-`;gwpjGTc;n4~VgnLrx zqYW4d$3pG^Z;-!GgHWtC;(TJGMd1krh04^W}^Ml z&@U{GKn3A3?k?>N8>70$Mi&K;SPr_+p*2B2=VQ4&;@-UeJ&!6l1u5t>nTzzlON1mh z@CcT(Wc*4i^?J6ZWg8VKC4-=Zpnb+E!-Js2AYLh_gPcLQEzPMJx()fVupIS{?>T;V;^9;oMj7x4eTkd;Qdy<-fHl z06kAxf=jStF$~kbPOjXf1vA`~za5qdvOgeF8iu|iATBq=+^(oZl05}oGmS#Ppm(rP z#Ic9JQ?-fdd(d^R>(LlggsdfVBvJu*wLlc=I`EFxrNCKA#>(TiRAu`2vsFp57Ux;k zU|&)yjx#GO*71AJv&a@uBAOYfCmi|f83`JL;20BIbNP>t6dLcYpBLe-oDrH+b{NqO zk5k;LfMUyETd|qb%57`)d5}5U`sS#}oapIQMy>K0Uys=4brZ(kEueK!^_HfGsd!WQ zgZPoo`(Q%B9AJOR-LQv|iVAy544@>vqO3=qkO?!(d_Opzib<9k+0aLQXwl;|P-ioX za8fh_BMa=nX{PiaimOt(a(?m(fy^q{%!bD;27ZMMfD^loW!?`sUl`Pqx=G*NWna@M zH%zu6n#!Xv=&e@p<$CETxm!M|Q%sksak=}Htu)Jv4JdH)FZOUaNn=7pqDIQ{%b}Bf zwsq*{mx)DmwWqhcPh(hF<@Fy@6g6&~*fb^j%I50+4Idod25nhP?xB2d(Sv}6;gZMGSx#pmB}A1g}XAm#+m~&fw;IiXp55-wnI+cuMCg zFMlGa*)1p{6SunGweeK4YlJ*791K96@y9F~W<@{f-G!%nKQ%jx`Fu<`jR0S5vj2WR8@~H)M`J|H}^UYmM3NaiD$03t@rNXZo2&wP22dB7=Z@?Oh)OwFS=IkYz>& zC)~w+{E2ACrMoh`ErfRtXX5aSj79jke-{UYPXq{kkyW=q)`|5uFpcF zJYDNUK-s}|Vv<6Eza;3lp$xpq+0gq6)^rT0&~e}dO2HXqucroTBtx@WiTxv`tRf`7;F<(i z)gl?j3I;r;wG`VZ1&6za>ph+P`Q8MO(j;!iV(jRH3`WhW#e6p8NC13G3wV|T%PdMm zmR#?N9Mk%e38N>h_KF-fLI!S1W{xYRX(HktPgrc!o3>g0+|hKi*WDAwnI7OYxz2I- zd-{xR3|{)uAz9hxL8w@(V>!Q@M=(V@wTfcXRjRyu6VM$<8eBTrN8huu z9u2XUYn^upG3M`XiUU{pxrs3`_itPa~nrrTezVjy#gGSdyp=YjgzR zp)10o%Jns35pzat(D(12MKoe3;5w~C1PFsl%me0Th}KJq@Znb-yC7hzxd}P*bamDf z9)AvQ3dI>Uj_Dd>9_hs}sAzu246tddhZf+=uN+wN@zUeNCj>IkU#?|Hc(EVNS*kaugj`>lnfy)~O)zK3TqD;UME1FL!jDr>+n-MH;)%?~>bHaU}_p{LRjgj== zh6yfYZ|2&%Qwc~sk%kH1`vp@!CbF^{?~KJDrI%JAw5CsEX5+I7*?FlqY50XGpt-YU zpBvd#fR+qK2zV21Bv0(_)6Z`|{Iv(ZiX=w4BBTfX%yV6`W9eF*BD->EN698+P`KKd ze}t^;D%>CKHRP;^zr`mzcma_aU9w#0a^0Nm{mfMxV+8U)0UiW&Um5>-S2WkY-W4RD zr!SAdDn*23Jvy&eM}51G}vklM4h}jLLw-vR4`|FQRVC(agp?yvZ(q>dgR}Yd#`lnK{56~g4llYnW8UurQRf)Z zvnoE|xzfWewKNYhc{71HWAL@TJo8vqn4yUb&#|J% zn4z{C`G6T#fsYbr-+~<}tWkcU-yvC6Vr3mcboVV;`Gd7a?SRi$x(NZH9gj-uw(s6V_Q0ERK-w)HXH4X zBBDd_#-7X`k%rSOM-k!(Pl zNv%pA@D)*rI{^Oa<)OI9{1Gu^&I z6z`jHKaPgLfUDi(C<-(LZf9Q@T@}imARMW<@)|QdcH2bpxru{q_G-85tq$cu`nh6Yq-YcyOM0LN*ikqSaKC{M@u*H^$Gp-@UGo(nq<3VO790d=n;dSzwK# za00hrlr6SU)oa?{VMSA~&Yp>Bl@{SB`PHMBzoB_G#~`o$Nj(Pc0;h-}n3zXX4R}_o zYfjv%tjzbujH-*=dFoN1mGWcUxL#D%wAq4 z@3N*NYW&5o5K|7uc&&ioCCWIZuQ|6ldE=bWxrZ1t2KL#O_U{2RjJm=bQX&Z>KIPRk z+ZQ?hf+l#sIJF(UOrbs`6&mL_ALbbf<%WkSDF=7;T?V@}TB~QZ2-@N7hLf7TfGJse zJ!(cBDo3pBw3~9Qb!|aVR$IqMYNN; zT;vGOElG&L~1Kfbixy|#(@4n7gymHN7-kc+nIpUM1+ z_XE!&f09su-Xr1mX5(^iIP$aXc7E#iT3SCutOs7(2}T$7tP0LliAIg^0aabF$hvs6 zJef=_DgKPFP;=5abKiA~BxhWU^8d;nVO>}85NhsGJlHgyPKQ08azxi=J~6~>&xY4g zqVXR6Eu4*v<1#-$~qv2$HqrM_|bu`RKG*uyfQUq`~H`GA3~kSB}>dV*UJ% zG#%rq=w27pZKHPS%oD=&S_JwcA7$u~j_2rUk~<__@~Tgb`R6bT%hdM5i{0~P%qZY0 z++D{zo!LW(09BBb0<3=@S&G!|clUcT?JKZHz?8|0B*2JZa?_+O`w!Eghb3_~{ksLf zBk^$Rn7i-3!9dyo9SF4X4QkH>lscm+ik(j3ua4u?0IrzJb!px^>Tb7|L=BbBZsGKm zyvc)WZG^ukzdlxW%Kh-0=+{(ysLwW>GuAA+g~tvtp;z93s@C#!JH7!^;RR%!_cFkq zI+|*y1s!6$uff39aC?^q{gTzn#w%%0191u>jnSlcs2gs<{!aepNkiM|X7=NV_~XMw z#kkw(+p%BQN$HEfz)P*~yLq+Vz>}48<1YnXC^Q>KbectJ+yw4g&%;0hFotA zgs#^zk>QyU;j$dk&vqh(76Zz|!oRFP;c^IhsZ7;}0TQjDitbCTk#(`yl_bdagjzo; zTbkC~NI3I2Oy_O(8LwvH((9*5#)s;O14}}TDUt|^QQ;iw4J!qC@PF!2(g8J@ml1yb zC93kUc})aE8KhjCq-f=(qH6dVVNoo?RX2GPveT$&LaU;kV_H|LhH9?B3EilZQ#7}q zln%w-2F3HWbaqE|2qi;UX0%R<&1^jApjy{q&fD;7D^0JlUxnbAd4OSKF4t1V zVtv%z`0r+2qAXHML3F>8OwJgP@^29)9&k&Y&4SgLWCkI9Bp0X0#x5=HIH5+ffj~W!mh1qY7N0~dF3{YFC@9!!YMc08?tWfXh z)c56uzhHOmSu@+?soO2uW`*M)_~SYCx0h*bZsJbq+qJL_V-IV)johVFyu}>dj!C?J zPu8ahUS}`nR$B+UPI{0J$C&1z;t(`b;x~0%xE{IJYRWhLxOwEECrb=bC z--WIV!g-O73quf0bvu^YkQyd(x!Xmbu1nm$g!2Lm9t{sr?~$rRrSyYwo+g#^xN=N= zl}%s4rWkQm!fr`+35C%%fsWf?oKSK_zhOZiMx%!>%l-4ep?_JbJ@UBBNL@#?bOq`} zF2*Ri!t*yp`ILp~s;B7OnTAV8kC?dlR-e{Sd3?BfROHcZXed=4?R_rQ747 z$d=y;egaQI>sbb|qw=mN1(Y3(*W<|);yL2za7yBDq8l z?P)Pd`gK?At#;8`DkuAq70c^?xwWMdjPJ7Cmw-gkhCa?t4xU0@#o`_s!>&})pqkcz zX7N=Dt-`mlKY*8$j9kTvXr!HD2}l(xePLu+Jm%A0oDkA61Cu;CE&q(JWRI9h-YnOJ zjp2-@?Mk+%DjGgvnbsO5Pjr<(%yGM+OS>3~o|w?CJ87G)>Pxq%gw<-ddo@4WVMq!~ z%z=}R=|soSO5Bl6)AmGJ)?_sBrp9^eKR6C6F?1BdsLh?HRXOMl=rx*xTGcv8^( zFql{k)bN@l?R}jur-+K+ij{V>jf?bq7y{Dx^yqP4nAWd3Nd-GPC^0Kws9nqHn$S)b zC!8F-9F5K083_;EWJ6@tge3A|gYxGleCk?(j{7LmVrTuxwX^4CG zIN|e>>c>-6iS_>f146_*po;$p_k|`KgA8A@-g1PLuQ)686u})(_!ImvEB}3zd<8ZBx1i<|RO(*`;2Rh~`JYkqM(NK3@-Jr$H|l?)!O&Bt z#lT@w=)}QMTB2Y8jQ^JZK?KzNQ{GWT_w4xUGg|)47e1nZuPWy?4b14H(v|3b@>e$slEttc2zd zqww`5bN$+gk(_Mc-T9?{jee|7iWp0JD$qm?LuUh`ZE-;kk1=J;!h-;sJ7%i7re-fG z4;pSpPUz*0;~PQjBhXUxJO|wYpvHeOVyMO55I8cWni2ANKt?4;~pOg2%b znrj#EW2(0X6or$JeGZ(gW|?m(^Ty-;2O6DeZvGOHOfed&PF!pw*!~Awd$>Gho$WUIKAyKcdPj=5VJ`v zny`#aUJxfl`)Ds$Oro98m|x-IJg# zMbpw1q?rA41{^RiB8L&(NuX&POCyO6H(@fyL(3~XJTJk45~1CXEEbd=Jy{nGUF5-x zf0u1qn0lu0oFS`5L3;@RUHKGt^t3Camh*twi8mGeT75-6z{|GS6${!YoXjob)F_He z$&Hi+NGSZ}MHQNgoL-!ZHUny0CQ5&kG83p8sE!V50=x(xmLa1GViWkv#ur1=)G`sy zu+N~=NIiszkyQO-zHY|zQ7zNI2@`&zI?qGbNlLjmjAVmYA&2H)IK&oaxm(vdCe*`W zp@u3YJR|r6Eig$9B$$vTPylcuD+k^?8yMvh;w=ikzdz@JtOH+W;R2psF0&0)og-^MK36;Tp1i*Oi8m>f2A^} zvPsqbWBqMLcD_kNKLObeacRw-pu`1r%qHtMg=d?Szri4;=CMfcqlUGK7A`u43Wpd* z94E(1TG^kXe(%9)C2A(pU=_}gq>0V)NICPgC7`O=4Oe^U;LXRIllj=b6}b8L4oB8~&E||q1r_SV0MxrIDeaax zSS9QFI_YfNkmbZM2-EU{vIikXxDko@!8r(d5nC}v^0{T3DC_>L>MUTYh0zII(sH+) z91!MKWnxyWZjgc8TF4ZXkRIzd_)9MkPs`eO{-(7oLMQHp- z_V@xwOxU9P6epqPUrqklD8$ccv~!Z`v=Lr6ZCGr0lKfG{T?=Nn?aFhG%uMx^f*U)V z>ZIbdrA_?0aE3x9#*pS;O8-u3=zhz@#crY1M+ssi4h!C~ZaaE%^^4oq6Na)j|FDqR zBQ=6ttM^D!LWR>3;UwQ>!x{XD*949>suMM31sgyCWcT07*TfsRHqJ4@CNnwa4{zrW z%r5CGZ4DTf7lJJRywX4XTZ3rWOTom;CKfPD=c*7apawEhTD*f7Q$WL(aPaVaetUaA z)(|mwO(*#q%c!@GyKMzCj7E}xaNjnTDvm*r@EHBW{I0LAw5XqTbdm(fcb4ICoN6Gj zy0c&bWV<&+CbM=WLKsRuHp)3LW(~Pw$$}e|nOQ@1o9`$N>>7O}n=5f$YX*e8H~ZNv zglLE}Kq!BOGIa$@rDX#9G`q~b)5X{KDsqj>y5&Y{@KPH+n&du!ZS<0h zZDeR-LD&nSB~a*g_B+Ryt!7yt1FkgIXeFQRe#03hJORd;w9)0mE~iFhFIQ@4s4zbs zIywC9ne^oeNRSeQ3*duM5qvbkYIrgq3P~Bm1>nL9%u92;d|; zG*W2r0Q5i=5a0+eC(4|#>|Q*}o$WCOn}R%d&wwV@gHVS7V)+t6fQ>5u-;5d*2; zEo-|JRbGM_z3^(GALV?F^ z>NF4*W4$}ex@xuBIlF91oov*WUS+eylkqeK6Hb9$M4mT;t4i9GjOi_;oF;GjUeBE* z>>7%>HHT(MZeW^e7p_y{*A5k<2Ma3$FVFXJl2AQox8 z=egf6>bGxUNQbh1q5r-H#;Oc6+s?Gd1?@?@Mt9O~8pFr577m8u$fFL9q3px(&It}j zBgUqBhSXCiDW5Z?&=CL#fW}o|jIJu81r5|zbg9&{^V6&OV>0&#XHv=a9ug-6jN{xk zQ*({G(d;jkiY|Zk0=y|zdT0;HQq_$77VOxt75FpYx&!`G7$Oh zghvV|U`PSI5@30dqu-sDb7RVx!Qn)=%-R*n$vAGnwJ|J0zQ_4O0?)kS9@}h3qLrn) zl7A>^g+e3#PzrJx)W9Q4vox}V5B;&9_7$RRm&rKZBK{*W$U0=!R5m5>E6Ftzseh@C z9VEPT(P+s9PDv}eU`cfuq#6&7a`*E~i0v^oT{WUHU-pff1nNhWh@p^Uw7Og8P=%;)36qu-gCL+&ISEwO=anBL)# z>T^73_&DOW91Tf4$NUPBFymY`u;0nFP5arbbZA9Ce@Ol23B>V-$r(=KIPEb1WUkC! zN48?Yk$z}ki^c|jhxefS0C=pg1pAH6YbkKg0CTuf>?_L z5g--}@;`gj8qhLQHpu{DxM;aQ{Uk?Cb#bOJ)EnWO*##M)InK*+i;TlyQv}EXc)-Vv zGq)PsPw&rQ)#Aey6SL1amx-Wki6rI;HaU$7wR#?fLYld`v9U2{8&eWmwgc6~J19}n zzY3_kADLm)Y!^mD2(;-%4bHBPj&Y;n64*53G`V7jZ077((zoF>@MZ)}X4u`KlB+g2 zW?Uvqm6eeO`Ldf05L~FS>sVIXMZiTxMlLFp1ij$(95T$25xR$V0HyyTQ;alHZfP71 zNgEmjn5R#JFTGQRyK7peZC61FuMr2l6}Dyu+Y%26%roDtS)Y)`H;R3IYR$<4f6Ig) zNJel@6RK~gk3$VhgqyySnYK)oP6Ie1jiVtnm{Z?>w|Dt^WxoYcO(-)a$^j`bYT1Z5 zw1c1k#&ka*yLYsfDvyc;NGs|RHt7n;Z5398du;uAefkQYX zYXwl*CjBgfCs5TjCIgNH2>>b3L{MN?!u=}^l%zZ))FYi5$0Ha99RtRqUy`& zM-^ZMT5>1onn==k*QA8tC%Av2I7WP+DMnSxibX}y4qZWnGl-_KKG_V&9^{|7=qk%S z&LS z6^tF>dtb3SuY^6o9L2t2)OOfSwVry;qdJSfBPXJN~LdB$@ zj|>?hVXOP;Qz+@MO1grYE9Rf}z91OU@=m`^r(9hGxtV@YFM3M)N&v-fQ7D{>hu>$C zxLB851mo-rnE^D3Z4KB%TE(9h&_<_fL!=p&)2Xg!*UveBeh6^Aew?AdV$8`c*;wgq z?|H@OV?(*=QK4>ME!=xbTOioCsTt_SNuSS^UIJJQA5auIzmhAufU6$9mdo8*C@i@KzqIjddXhiR z`Q{Gv1JJk6J}KLmnJ=)aL^Eefht4N0H{)Xsqz2aymA_$*CR!k?sHbwyigTwmZpdxu zkHvH6*>T?Ky72S&`J?U5-&rshwB|J|LkQ&xM7eXuZ2-OfTzu6q-sn7H#v>$W#HhZ9 zri$$yXT~}Y=ZkEWSxRi-wd99AA^Z@h`3a=)-@us=_F2wp0RC>aq!+wd4h9ZkC%FLZ zD)=O_%in4c2XbWsJ-rkEF)=H12pw&>#f(d_FI*-3tIxme+iSO*D zVch48UwH%2+v2K5A<=39WnDBz_{L`K;T!n3iX+i|yw{B?dD(mvB8SiUs}OdB%q)Gl zia-g#WY5K5H2>>WmXIbF8CA0deDFuZ!j&*dp9<;R$zx@+-H!f1*Z4+dil(n)wt2q$vIy%dyOBt^VWEF$>4uK<2Uy}1BD};W@3W!-cQycst?iFGa&Y*Ls7z6371n`?-X(S9~ zda)5Z25wCEUH-KcadT)0Sx{V(_mZxkS&{YPiSl3y8XW(a;`s<;!JgcN*4F;$U&1v& zSJmUAF~P(T%1t?!1e7w=#D^t+s0U8o+q|^ShR+U}dNzLb4oZ3R&DEOyWs~Idq^c(a z(g()qbqEz>}LolAzgOn+wO18!+z6|6Lg zD@dj$uc+Dm99bI~^@5P~Y0#r81V~21H(etouvz(Gp{p9Kl^;MIl4UeNA_%b-6MlqF zL3}uSaeCpeX{2KGG(SDTJlzKjaiTewL)S@ggHMQjwxrcmt`rjTQVC{jn8BS$Qv=W5 zJxUR;_8RrSbmwrV1*wmfYt7~w*lq=pukel3!a>MV+hJDBQyjE)8$0yvzS0dLj9l8L zwQ%w4)+Xoi+loNdbWUyd7xkX6c;kEWeYgKNH3EO^+7_H!9=_+6ZXetPkk|`En?YX5w`(0^e_U!bpJDaDGf1btM1fW6h)XDM|Fg~&d8>cUY;p#W)qN7-#d10%?y0n%)v6mE%DcKPW;-Ft~ z16Y({?Hl!McWQQ|iJXz%)Ds7&EYJDJd{p5u>k<+F%Gw6*6+C5!2qv#&x5z2}K$o>K zOW*BcZ|zf;MH78%a79|M+stcKl+?~Wv(5!*2_$}Wi}-~!fWplYuWfUK*dzhLuEzMU zFQyKVwm6^J09uMFymZxip0Fl8?%}NU0=rEQf73;r_ z>A<4mV75wPEvfVSn?*%uw^ql>%ElVu?@jEk%@aHu4VJXDxs|sgxcF=0_U}BdLJ4@6 zMw{@jwy2?A2nXr4)eeT&+(<|G$DQq4)|DqfpwHdP@_ze`wRQc5JqJgcEnz{mUwds= z(eG=<|FT-^nl@ivQn-(7{XD0-JbpIUO|J=M<3H87u2dY`!D~Lt@Igkb&FU*GWNlJ) z7u_+AD3rFDY_19Tv50c0+J(a*h+@hFCDI$3Lc+6}Adm`k%#dTIOs^c%TXAs(7Qn<- z(mScnm)&JYerVOccYPzBW}>=K1$G+<z1a4L##$hhCEhPkU7-1v@{&GEIYC_oJMj>zz;?H&{~8el{Wu}TE}iSk z(m+d*DNIUSLdYyOZyKH=I7KrLM;&~n= zks;_*gt@wlu;CYWzvrRl9XcqN@!dbzPNG!#<9xU^{1R28!7D4eUiMjgLwv}bCwRmp zetQ^y67WX1uw_T=NRkNlzX36j*&;2}b6F56t*91e5ta1S5BaAlg3#fXOcka-Ce5KKqfwjSJv=z*cuZvuW)b#7gKJ8{;o zvaOgqdCIu>BEFgyRi8#->WeEID*Qu3_7Y>7-aB4XV)ExhaE4u;sn2z9%Ilm6|#i?Ns)gD5wS zxUfA3>cqKH>?T{ILl-EsR*W)Et5@V}@w!;DglBWJ;4Ga2_V(?vF6KrHqli&oa{QSY zx3psOT1H=$dQS{(^m(e(io3Vp1=IYY7Tc+zK9Z`iKjQ!?{b3Hbbd>FTQXd~nXY zsb8<15*CncO{ICoXiLe^ATyd-G$$D-YI~7f9|Xrn7$JmAky8T5fW)VyDFMiUT?{%5 z+~BfAMA9l{QDLamB>t~d!BO=NT!zr8qGK6nJm$Vt(5~d}>6DFWx@o}qnI<2Xc8RN> z!_UPWxLcVl_*-~cG4FE+q!{5jf*D?DK?+=z)hf0Aip7Fzv@PACF}cds3sq?*^}RpQ z8JRRQ+b)vnrnFs7Q8-2r9O+zunnu&%niim}mBvA1+34LBdJpIEd=C-XMOR+YYVT)f zLKXZLEzNpgGj+$cEL(h^8vII)Q6BW-RsZ>qCpz#tABh$JX@?qFE6C;!(%P<%-00uZ ze(;S4zkOsTWe|cQ0w!-Pm08j?YRXKlipcPw8sFm7 zQa>gQrG3@fqVt3u@kceuw{Zz+S<3p*^@N*17L$_Hdol~Ou7gP&4&hc!vcjVlH1k^e ze+B_fC<2!0O2SWM`^-oeXN@ob2u=}YpWAO^iNoqG<%4{51_-`s3TXpGsYR|Nlh#wS zG7>rByk}N6IXMY>G2Z3SZ<`N@Kz8uT!!f}ek#H9VW@-?+S~}v$9~FK0^L)>&kneAv zavBcu`YU(JTf$o#9iIecBHosZwfmLMx8 zYi;hpI8}@gq%P?zD4-uidcKs43v}J3VCklb%d1%Wdvsv>rEiB-q*Je~9&J+>79P;8 zqf0>7m+;AVVY(^(b%F38vrJr$vl3@4i5J7nC)BMB76q)w{Fd{@dAOyVPX_5bB-F)a z`2+jb+}u-rC&z~l(h!x?yvSLxQ+xTK%2^AjqDU%lMR);?Y=<`8z2P85+BX$LrC@ps zu9(ICUYjjde!uJn=G8b0!)EY75^tc(IEJ^UhJe^ak#1rJs`>G?{f7&jK3mSMH?<>E`wBt_Bq zbHJI0NN`nFfUbuUsW|_G`#puOae|d$l>W76HzaWlZn)2gYxZAzl=Ko^;Z%wjgK^>6z?-1HX2cY?H zPdH)#hy?p@fbGcevw%#Ao+$tl$fB++6N(8P*bfQjEeO>%QpF*2VZi&7a{GU{dZ+kG zg7y15wr$(CZJQI@=8iS7ZQGdGnb@|SOzgZl=lpKo|INPX>i+b;sIGdRTI;)dUF@@} zX{BWbewSB{F?z(^5ajs@u!F)fpxNhcIhkO>Q~RWH;R_(^)s*SsIk_XbkihB5oywf2 zH$L9(?ac+mTdQ!TZ8)JpYnAde7zxAy%@r;R~ZEeXFOm3QT z36Lp^$zK@h^w6-ip_C4FFvV_ZiS!agj($})Bg_k*(MZb_Nd>S_YdaoAxDvR-kTLfu zXFK8{3DM<&I-v+|ZrrLUiUOKb3e09weRD0C3_RxExaO$OEjHJobM9rSoIx`5WJDub z|Fh+xRI?g44w;fw7S}%0!F!F|Sb!au+Tt3J!#VWZT$(I5_0Yv}9MrKJN3pk2ca~AP z_GDuHGX|QWaQgb!(v|OLu?Yhn`T2u6kEyL7xk~PB{oU?0t)R1`fT%jZSmv0+)wPd7s`yz{`tQ@4KKt}`art`CnUt*T**Zf8*I?rKh-9pf$4K5fx<4&G=qzY9Oi z2Zcx4?&)p*OBZ}TTj(Z1S?NKBw%d5jYt@a5>ZOKz2#%+&mnWmIHdAuD{$>9KJWkc6 z8?Rkt+Cq|VHj6A0lVOzYB^6qZReO6Ka$lI(iD(o=-(l8BfK6bXt#1m7NuaYX15%$V zlbLe`Ob-1NeyCifY&Bf*L|>ZI`Mf4kRn&2z60I!9FDOb`eNs*D{%jO`GQT|yy4g_?U#7V?&Q!;P$wH0OB-JuNL~Gm2&0S-T3bh8Lhaw7V z%n^O;E^5vL1>-P=K@J2xt}XIG7IjMhu=!7e=JO{owBj@#B*l`~K@o$an0l67XoUldDPKrPHhM^$1UxYc-}Uq6Ks3=z zw$a1{I?5Orgm&NW-=+^DBmQ?8Cm}=a_uByq8H;RiV4YweiEKqWPh|2d*ubmr^%Nri zhzS>upkO#G{efkHbd)8Z8u=C1Za#YACWeWMrGydq#u|x+sqcB^qWo$QykVv0K&Q zRJ3Y|Nl`hhOd*>3Sc>I3AYM7L;?ASPb5Nv)OW;{TwwJoVLrCCIMs`0mqyPzFmn#Rr zBO*|&MTRwDeF8n&rb0R{3Av6xLMu`c_viw(QilE6HD3mIhTID!=dq^cewOq}kDVUTRW93R#d=m{5wzeb( zAD%U`3Au^#kMJr~;Yf`i_HPQi+1d;s_yR1rRr|n~n4j=i*bF1p-G?udWC97G^`^{C zTH&Y)9Y?nOMY8;-ZohE}>n3}-=|)endKX}|30Fa?XqXrLD}!fr2?Q^LTq zMpG6CR^|o2mj^b}Z=rEpKX*cV8cH`%>$oH#R#-5kYtw}LH>@m0rP-Cq(pk8w3x`z{ z)8vQ?O(9r((RunG&lfGg$V!5wNu+WXU{{P@O$IGT@hg#7Cm-ztRLw}{E1%%pm7cO_ zXHCv?C3}eZI5)#homf`?lb=P!K{qWBrg=Cl7=6Ind8JV?#x!e_r`^sgh#$0XfQjMgh*1lQ2BYEVj zA*PytAK@mMF__VSvcv3yxws$OGiJ5vrg}GnK|Y-#E@xS7M?o9S#(7L2$!f#xM^~tf zS}kAhf9UIDM(VIPZ5)wOTIBJCU(Bse;kdZ$E_8o21~{)yIC zVWesYFe-86W_lBe2m9fuq~u`ILQ&FSLe(l@|Ca?hdv@2ydI>oi`3_39 z#1=)c^vZh&>ciC)K?(RaWtYP{(7#H`KmF4q2s!zfdC;8~!Y7Xm|AAWf2U2k{i*0+I*Ata z0_0pR8CtMh92^Pwd)(AE`LVdKD+Rrg!Hx&Ieb*shPUe4W7o%VGzJ7uKKU=SUH_$=w z|8BAX(^*xfZ81KeRNw$Rn*)jeBtH|Qsn|voug1=dsEbWJBESR=;IAk`8qqsclh4a+ z>63FrN8x|hA|7EN-o^_0EV;PXihvwmk*p_)gI$h^R z010f&8+>Dl0C=puW7TcahUu_wF3X9^`hmu?72<-)s)`aiGUb>>thx! ze1QA@CKciVE3iw1vWc^@sh*4Y9mTxS!jolK1+G)BD@XdGK-b zjvXW7W%Cvq(+p75?X;&$wxdIxBMXxViOdWQL4H((&WB4M9G| zF$TjX(5_?dUIzaSEWAO2E=)kIlRg|B|HixZn)1hBteN)(I?j>42{S+7M{tEQKGUJ> zqe<^@$wr~HClsp?SOSupD|F!l4}NEkj3O%`->@N0(MmPaBxT*-l-D@`2bjfo*$eqbcj( zexp!YyAl9`YFN=^H9WYU5MAh7`nKqaJvu{|1^{lzsb><5yG!v*w)$eZ&1s^~BQ1U3 ztPMAwQ<&OVFaa)m;R*D43k&f={UBPoHs3X%nYntB4R98>^p`8_Z}9YPrSF)MzK9`y zCU8YREGRKLurhh1m`X9)rH3IQyy;X@jpL5LaRY#tJRP{oS@BlxWmTmaz(q}dCfA6} zJW|ZUk%ZxcHvw%OaLv~%wuUURtREBQI&6vQ$ZuMZ!{)d6EWe{hp~#Da9)DCEdp%XTLdrQwJAm2#L@E+-#Oa)*Sa`J+Ae=F;NtzTU z<~DA2{>1P9dSf3W=<*TrkA6@0AOTEO3UxU`&{ErEA&?2oBJ0d{c&<^Y@qC{@C`+ZB zKOG6B@H-Kxg#Ll7w2SbTGibvh=yH>udP`mmZ=h;`GWCr8oXEMmUnzzbiXs=}6)u^5OK$p2V8mA`J)W6@?!C2xIZ^^K!r(~gC z^mcrF83df;2Uy3r;6wrw23*DuxXIx>#7FG=f+Ph6UhU@*dGO-_zF;VUgyu}jG%3%u zxTq1=Br+!upBo(qF zxWsr?$k1bKFc4LKK0BS{{W9Vt9jXfP?H z!yh6FQn!^CG1&4nnPGT5k>^N~-o7w=DvFaeyv*JfxU3Irn9f~CjUuK&uX1=c0)CG;HJM)C7R zScIZa<3%$9M?k-U|MlwufxG>By^G^lfkA#wa+svjx0aM;d)cZ<{*w*cDyaAL4m~e{ zUVVZDM+{q!8i15uiLK41v|;MU@4F_KBsvumy1Re*$<}gC8S{oiIF+X~*1q zIEnLH+gay6LwDcWh65L@-F>t}Sh#6r%8@POVFcBo3$Wv!IqMMdIwD_&GURFv{n`t5dW#6*jBCG)O!Aw)l=nb8yD8k88T{43Ah~+8Yn^rD+B!L)}5QOZCF>7D=`>)$p zo;(zzwKTS|f&CHcYItpPc~M#*r^bs&&B!Q(i}U+o8u|AmVW0C?_OC$wx|$_!oSV1H znZ3kL6jEdg7k*PO&?v&!zx7=Ks+kJ0yd~ejK!Bfbz-9W*Pc@4)y7Q^Y1f#))*og8i zrzgHD`lmoB0GNJVGdNWN137b1ipHto3b3P`cw~6D$yLG2HJv2)FVn7w?W3r_m_-=i0HYKGX6ryKk=OS`0gwe z%b}fHCqk3ePD4=J6)0-s6LollDi!7xkGViNL$gEg#?gB*#9`vSKJ8NBB;?&=?~4ZW zO5b1y;a2e9z*aXOac|HE*d&y!5I}T#{-mhe+|L(^k#cni_9;&C;W#48xnl&yfG1Rm z(cq{~75u_X3;H*XE7vq!*jH5xqbSKm3yw7VI15I?o}+Tbd&hN69PE~$Ss}aZWO80s z!9NCiab0ulEa&^6k8bUEFmsMI4W36OkTW+O&9s`Ka>moD;h@xh!^aAtfFRaOJ`AHuivfbvT zzs4;MOY;33CWf7;!v61oVlc=Jg*s-W%nRMYj7A3t*t9s{PnMvxH-WM z=UVx^VP-u8RD(Bur>925sHUQ{-WX6q zXwflpYOPPlj)k<7m>(f^$rPo9hq=4?b4D8FzY6o9wqO4405=n4!9tlw`eRso)g4=H zF_FbKkqF}-<1Oi_ZZAuEf}t3RUws$EUw`lF+0@A=aiG>lbcPW3Aug>D8HuZSaNygX z60@L6{&4ywbvZ91VQJml)4z8b3g_GsBxyI?i#BXH$E3eh89!A3@jQB~P$6~AroOf% z8x$6cvVocHTfb{zn;z>kuenX{nf}?dmE;{O#wURwjS!U3H{^Jd z5sY!4_IIHSkQT#o!6-;2>EflFWRX(g1A5#py1r4>K2P+0{!u9J5vu_Sv2dX%B`Km= zbOyFSZdD!^W?NK@hno3!u3C5(o=8$9*shb=2GM-sFsw#E^to-BH&rPS{3IcB@)8@9 z?9(ClpH~)a$>sB(6O9O7MH}Mh$~)F2yFzD zHBa;`HpIQd=$jAUL_|tx$59GHz6^wH@LJob3R#0)?Z?*R(y&>X&%)Vy?GbxbGR5TG zrh8dz_dN}u1Y2NZdt^QJS>9~%3!w(Y!%g9%Dt3Nn!?+gitzB``>o}6JQ`7D8>T$@# z&dXsZ(Omqm)+)7(q|TX~&XIJUj-GEVC-Q8v__72#_D4p5uk&VgVD(dq9BB<1(fzfS zz6=J(vf2WVO7ptUN2q0BNh~#sjOTLF@7fNFhP7>g1#O)6Rl67ZK-{dpWfOB5PeQ%2 z9mk;Uc*ucj$aUjHwMPu|f)R`DtMUE^8t8XlPWh?+G!u%bg-C`rZf$jwnIbD`w2#vO zD;&pW-jsS`LOd2y?I2p)JcE`V$Cn5@hi#Xgcyg%5@DGD8Zq6yv!gO$J+2hv>?4Iq4el2v_fH9cZ}3O{zE?LViB1?hPVt$sII#$J3tty7=Fd2?v?WlMw?&er- z?>j5g;3!Zz#Grhp&%jU(GYvByfF73yqgUtJ?feANzd`S$_c1{cQrshO@5m|EETmCj z$VZL#D5GQ1x-Q6+n}toxJr59rE5Q@Fob`H_9$8R1E6`gey-jT_i0_6^80RM6;`BuQ zq$)Sg#hKMFO_8&8-o)R=kO7?^wj1uKmc8gA3P>|T-;TikkQwSkN^Is8fKiROSJeVP zh^Sd8(xNsQHc6L!lZVMueYaM0kc_}Sb98d1wTr6N4P(vgqA$Eq}q!UMSV%T+I!jymU6j%DUzfitXXAzbdoN#9*|$$${CRt-PVbn*2IoC4n1JOPe@jA;Hs zXh^f&ztk~KU^o)$SExCY?j_n?^sfUoRP6eE0nB_PgKvI?RV3y)yUx=pQ zYRG0$hAjyXq=P^voV|)OM|llHS!a(;+55sF#tOKE%47LF>cCma5vl% z36HAg_!&0CRd!kQ+}6EI(X%Y%(%F~G)f|cVm5L!JyJ5jSJ>o4$E8YPp*vBCSM$%1Y z>y6bK|3k;!uhY&g<+@h&%cHaBCG%UA1GR^%O#5+w)sZEdG>zb_3*{nN7m);(^PSgJ$JLkyHyqC#1k)^-i< zhH38kRH0&e4(2#Nm^ed6tb;uF55Fx!IG;KOAA3MtXH4ZR>kWs@c&>k&GRpdaeG$zg zvynW8&wh&pK>hS^1xH;C(YMCvG{ybGYsWG!z3$|LXgc zKkNO}TM`kNV1cq)e7dU1G=7mj^-B$^pOLXhH5rFln4TF)t6bFV#EL{W$no5;_&MX8 zEnWE(8`qFE4O|(s^ zPvHcnUClAb>x@C4-ehOsnoh}alIigKw%e<{l43c3GWI%^8;pPb|s z&@KVuy2DjQ#wX|&jjXbj8bL1q?mzJA4~ojaM5(A{eUQ9qU={|=6aF}gmPG|?YnU|% z7tDKY){y8U#*f>~x->}8yJ1zftS9f?(|OBw_~~p8@5v!LF{Ff1O>QXf1Ik=MGD!N? z;1_II;NS-)N=(FAi~>xyU566jg0loio<#j2b{pS}8%p;|)u{?U8hDpkRP7C30tvGT z;KRR|9Q`QI9iY1T&6{^z0mou7a&69igJ)a%sN-0ae~kN*;L)bnmHMK=CK>9eXYyf( z3H7bt{Xqu}E`C|+aV($80Tgn(q(N;}OtcStln6UzTppVcvC67h=hnz(8tRq_O}za+)-q@X;1qTuXh$kJ0NDE?ium2P5SO?O z{pp{B3Q2;BzJgd_fr6{Y0i0nzmrnX?UnQA4Jj0L#^N)=JcP%>5Pjts^jR4qtT}i(S zqOtJzZJu(!IQJu;&gdhIiO$~*z7_Uu3Jrb}gg5TA{kaoY8dS}Woo;HFZMt!@{!)b3 z?ib=;e)b+=PnRfuLU?j1?72J3x3vR?c_%eytKW>Ee(w*l9*%Dq-vDKMLkODhmzH0# z_PZFbC0^0h2k{H-ZP<^nlG~Re|5$E z9KYk?q}7{)VYCG|fR_H>c=|R_kpFQUlv}^1<9|@jyKrelN+gC?p^tHsk-@SAPQfL!uZ0MFi}sxk?3&HRcr8 zvTFCm7r)N|>2KKf1%N=%&tc$Zph(fUE1|85UP3>U@fI_b(owtbli^zvLDA=3c-Zr} z0(bxb_s5kROseGHV;Ak`ggW#j6eAi`X3pna6Gx4meCH#+*AJ3K2~8?&sU$oxOj&XC z19WP+JW_z>2-U>m1wa%Y4jC)G)$9d)4i|DNB%HmJb+{F@U)7miy3@qZNdvxCpOJ zT1+!}!$PwT@`P~P@6>u*r2dv$o(yG^Q#*{3aHf`F2j8t1h^00O*()a&WHYik8-hq)WgnQUs3iK9^WJa}cK$sItJV@ILt<3}@ z@X|1Ya0G8^S20X2fAc>8ecDo<)sHu7R{N`Zgx|QcfGibp38NX6Qt=?$L1Iq4!-`N? zfUL-bk1n1|4UwM`nQSl&0i$ekraz|ZbiEumGpX8n=$)Ah%fE>sM+l5WJXT~Q=#(Mj ziK;og>bP(y*}vd#AK2X=Z&xWC9g`IGr60@2oAklcBI@Ae+mayWAZkd|D{-JINU9J8 z5ch@#fESkQTq~RWBj3+so~X$Qkg*0p4bJ#FM=~~pYgE><+H@QuH6>6Ev{f&{8O2op zLBD6I8E7;F+6kNylO$ynM23iD zXU1vV-cc-Adf%&*{gh))iexPva!{5?hSxBsBo9b7E1j`-8(-H>r6^VEbdx_bAgpF| z7353MN3kQ$_L5hAjvAk8mb#%jz`~3c;4pZtJS4NRCAYY8(c0Rqx|9pOQhx#7nOPB2 zrI5l38`cwrrJMF9Ks-%EgQu%G{wX?-7cIeVPBnhCFj?P7@vO7_U^gHkUPVjfbpG?Y z!O~^=f(BeSr~}(9lr#(!*;Z(ZIpC$|Ym1)tv968YPTXAQX-KMsG$0m4<|`}!;>bjl zD?w`QLPe)>zoHp6A&t&ldfQv|kk)OewiF@O-LO)b5exislh!<)CmcxP>8&TSQ5R zScp1F%I*~o*W6%^Js)YmrJ>@|yY5h7NWV49!rsuT%nHQ0_m)WMJ}OncDUj|3j8R+@ z&oW?1L}$GAX##?dB8;*&WX@?wC5=S?*1TKAg8W55ju&+INIi zn&(*1)^XQwr55?45_>@ayda;C_UC`$SeyT+*q+g>*KEwQ%}ksf1fqNtce`5oT2c<+s0w(vcBqosci_ykC*C#x(HxTXYQ>>NHM?Ek{gb;OwE8 zEwLb|L(L!x@Sspo8C-BG zTcvX4=~kN!@ryfu@0?vCA*r05mt`i!bSTe9S}FA;*UHJ!$0|NWLH2cxMtxDN`iF-I zhMi3nu{lkO23{caT7G>UH(fNZq>W5~+5=V=_leewEWsUEI<#c3?YX1E*_{9RW-jnD ziE9rlu7R3D2d2+>{UIHmuQ5)Hm!q1bo3%wdFz1@wQFs{jyD|$8Ik*;_cH1~>-%@C^ z;r3al3=%|q%^_?fesMt z>XG1OES#IBWl7NvkUS*bw1kALH+%${Yfw_P#$&_+15M8-Xyv{mbeLVOx-UypD zXm}iJEJQu9m71Xri05`2GW+?ZrUMjTuu4Ep7doVby7u(>+(=g4aoUi-&3%V30zF>#^ko!Za#J`uH6h%R6=t*v%GEu7 z3S{xN3Pn((oWdY_*od(4R0Ju2Wtg`|F(8Pl~$Dar8 z!0yXX@38F3P??PKCEGcRTA8PV>leQf@f!aq(ws-pN zoblS#-FQp(9QMBtQ0FQmyLhEpHBMQ0$AMtAMjK{=`FSZ5I^NA+vqF3~%`ZRVLiY3c z$W)boKefpL&f{`nFaMnA>uv$@RV^U+XzVzRYNT=RM&nb$@i{Oz{arUyF!z*mpc)fi zBRqf0h6%ZE;@prn=m`V-n_lnPWc-Y5K%os+Db8O#_F&g&>m&@Gt*`io{m*&&rna5K z{*Rz2*RWyzl!j!OgDV6DMx*2+M~zygMB^-N0h%806wt-+kKo2);824wf>#30*7PJjwE2xB|R8~p#O=~QVJi#D|(KRI%i%P$S zFS+r+ifXh0#EZ-0ccnt!kXLifq0JmKMBd4EpW=BFWo8aGyFxA(^9 z_SB>NO{s@Iul5wT8_&~{z#9VIADD4>!apz~@L{!$Y6}Hbbo$(`)Z#hSs7uiR&0vay zr5^cf{4r<`*JmdDmJv@6; z&!ZgCmVkQRuJf>lN(r#&diriG4-d+>NNK!nv{QB;O%hH%)nc*|HgcvH=rcrWf z%j(!!WuB|AGxubp!_5*Z+5wnRf)deKm_~N~%~k2)3PNk!fcgW8a=)#MN1vAp{z;*u zjb?%_WCNeTt#j3@TGw#^v1b}T^$6Ln(<{yT=x}Dt*~Te>cXN}*({dytuQZx+Z)h0j^ z$)+0lF34qo+vwc8*vBB`#Rqd}cM+sS{6Sp>R)KG^gR(*SXlz>G!-uxypdW5focrXV z&WAwUFbr-K(Qjmqk`0qMSKm+`bXX4=(O~J-0a#V>vqm~HFFGCv*QGY!x5yY1Z(*h^ z{)A&=2F<>3Xupm&ApYY1KMeuuMbH*-~dcb&hEPiVBoS$STrZgi$5Y#jVrJo!? zgmuu+|HCPyZGjH_UkgD00F>$fS_TOxpxvPVd&D}%tl1Cx$M2`I$Nt|Vwv@o3+k7rS z9l-uOg6wx#w)Ia$ttgJUILTd^P=+_*X0dvR~p+ z5ji*8_=to`+m^unf*cPp6P$5SQJ*6MN+A=B2_2cTD5a*!b4Bonv9j{8l%q6d!+x@2 z*YAm)E>5vyIbuxpwmG{E-urdXoy3(x=j`|bR^n`^YL&fwrlTlFr3c?9I(T-I=gkW` zG?sCP_)Lqwk}K z6PyFJR#=cZj5PAH_O!ybo}OXPrPp_W2}BP$j1K)3X|L=2uVe`qG#0wO=qUL1yNH{F!8=y9V%sxwBwLa=bU zMDd~wHy)@HNJYI8A)5SH-^#@|0BKaeR$)2PU;FjKW9)?stmG3OzUbA zUA6Fhw&a-!6L7Irq;+N??l@@$;xIwnqdUde9RaGZ=pEeJtXn)vx@;JFh@c`d3AsXJ zBAd%kFs(Z!$rP*RsBD{u^Rmx6|5@u+EUItN3;3NG1=K~Jz;OlOEh9m9;L?l~AL;4p zPI1zoeiZEM^hRQO5<0fcdKt^{M}7(46LcIR6-oI{{mkc3Rr>tgC{#Ac@L27I91B=L z@c^O6Qla|Gk*@At=jG)?SYD%)BOC*R#V+@3u@C{(kQT90uZQ5uvG8QV8A^^UR|AY zh8(zS+Hq@zNS@IPFKVFOF_*Qa3sWxuwW^rphfzY~H43aXM!kw-Zz!#rBH$VKP}$7l zDpBY|gsxJDuCnNO&pUVif!MeUubcr&Q)6DG)7xU@66XZmmaC~-0Ac7fLRp_~4FyV- zuEtOk3545=9*VUAxrQp6E2+sAg=9HHGy;x{;0o#n4TpD|26P-jl^8S558ifYZ8nVu zOEIy;9$Hn5sP(o$0yo#qUHfg&HyEF+BeHjOzR@z}xal$zAsR?>G;*zUeiGOU7+_Td zQ$l-s?lu56Yn8DCxJT-A0h{w7`+r{S=DDUj{%uW9X@n@;7IyhY$SX#?DHS(7@Y;0N zF%m*A;M=;6N`M~Y@wim#&cgA){n+GN2}4U8zh{B#PpxLPivJqK{1!h73+9Q2<<8*n zM7SofYG23Z4`-%XK#kkcXPww&RJgM>tR?`qXQk5J3K@OX;>_9ntvkJI;{|c$(BJ@(~_T#1Mr&EWk_=+eY6$}JM zzK#J{KZ{E6v-r5an(k4)wf8q9);Iy!1u^#zqy>_sbXKPruksxvs3Z^9aRh|H zru-Z`vepV2{yoYt%h2ude>^E}AL(0XzOv{IGs*W!y}oAi%nlWq%I;JpQ^@a@&1)_~rrB z1Mp`*%<#2%%9a?&sDPG}quPA>c2XMFjSYzl+tsDKccQ%!xeaZ_R_)g8B|~{ByqFDr zJv#KW;t`fta{#^2esw@}qXW*ZtQZU!Jjd(_4QC3(I`UaN`tkNzX^7}8l-=LI7O(B) z=hM47AIfgz9#9jCRPgD0WT-(kaFJSlT`BvBgxn4aGyXkhY=*DB4}Uy}a8AWi#N)Z&jzZDC8vM z|HuZIGod3fL{O4ZVyH=}3-Pw;>q~xlm_ZmP6CsVh%y1W zkcqw={1CSm~i@iKhp$rWBM4yXnbE%0a_P-Uv-pf3oMV9aG@ z-{yG^o$@VY-+8f_8RO71vEP*9-Z&LB{nm@3g;}k%l*a08KQ#UedH8nvs2xfz4~Ezv z3NJk%L~G;)enXf{H^-|y7NVBFtjrFLR*96Ynh}`33F!zsfSbwTRkxgdZZJOpXcF)UPkrB!cm5qTT_uYS+MnO#2UV~+JxSq~DhMC|+ zUXH@YgU?3@SCqHY`@Qw`BlO-3mSbE^Ah#m@G3EiBaN~sd8pFL)DFqN*#K6 zQ}LN%WIuPl+{ICzmsq{fG)#Is=vWU46G#)t#}q{jf3TP10s zq$GXBx>^%OMl3T0vurJn%_Pl~e<@B39CVl{qN0t}Xw>~0V&IW(l*^(@TY0w|6=^40 z8^-2dfB>D*PBPz8;yD`2`nL^cFEzLFaUJ2{9_!W2a%hdIwc>CWts};l)b}9cmmFP{ zlIIK18+SKJ*G3t8~$X8IKG4U3g z_prFm0O3u-%iQUD2_Q-$mEW>6!=c6aK+&x+qrOK^u^t8Vd{a_@&O4F$@vPGIV4o#b z&4iafCB{3O8l89uw2okGka)0qMND~}n4zY*Gud{Ug#mp5)CF~%vRs;W5;uX^n<_?Q z>;{L?zq;=zM>W=}}YA(KkjM53D1E=gMx{#}pAnuCeRiN~ENxOc2 zZrDTRXZ%v!JD~&sgc9myu54N5!-D>=wcnUK`~VIfKOVR}_kV7vsEXHn>pcEfMQ^3U zh#p4kI?9eXZqzH)bObj+zY{btQYDN&w_=)jPk>Uv|8eHQ7W6(_R1RC$lc|ccGpAv< zxpP&Ok?7+uUXIk$tlAQ8$U`1yAiXok&YeEZ=C@C@@PIJ@ayoxu4H3GV zPYS``jhH2j*z7s46Fr^t4h-<}Wv&aM7*TWGXo@yyorulTU!$P(E6Znq)yjNTcsH;hzcAxtm2DA-I)3C;4nUoWn@kJs!zE1Fu>N zZpKs0*L+jZq8fX24tr=Mj>-sE%cq;z_r}5)^Qyk!Ti;{&*P7Z?2N(R+BksOy)NA{# z)@qdQYs3n+9stSSu5-6LaG5c`&YB3m zvX+mwZ-{9+{Y7_uhn$oA)t&6Z-Z}&Rn=+b1_EvppYflu?G7_&9$%G!`^rZ*ZXQIo( zWp`0R1$VVM9t~ame%3M@MS`tj2nY_;(sks5V}?@#lXzY$h>?Q7dSwh<^pK`IV}%Sm zHA2}0V3nriA$X_aXPnsv`Vfos{LeuqUCVu_<@Mu9wc>%`Wt??*eV`FdH$FbSAgWI)LJr~hUmVa77B$eoLuB{F{Tr;}M2}@U zmzq!UCV@k?*Efm{fgqJ_-1~ITORz{<2;d`_aA4=Z1?d&mhW!S|!`6>QdNma!Rh{=r z$1}EGpgB#N5{wBzUMwe4C@tZY$=X9eUucUFhhg@y5p@a|LB)=mO^1>IBJEtcDBZ_Z zD1tPjNis8hthTI&kac(=T))s-bVUS^t#9~ir z05wr3l<@B9Qv*O-65h-Ju1^h3-V(e`dck2KZp-tOw`hL8e zhJgu53#dHAbY8jdp8B%ES7~^4s%E;VrhFf)ZlXjd^EwkjEcH-835l~YFOzL5d(pz0*^cb7lOm?4&M|+d)cAbNZ{7fWu zW|wbCcb@#3`-J65MaiPTq@L_MN|SS$bS^A9huR2e+%!ndrkxgE!{S2J;|YmKc3{kL zv5A)DGl2f8a|O(QcLJ`|fkW;dw@^rj-R>p5H>d+SQ6?u;j02ZESQstuw+i~2-fQd# zY(NoNDIOvQQn~{YdFFW6IT;ahVPSgjR&XUim5kKf#i+XWz6XWK^NAm z64{KnYuuL4Wi}cgP|Jgr7=0CO0|aOGhsi%Lpe5#TCF!K=J>wX4jNR5$qg^K2&kcmQ ziL;|_&hCw-PI5JFf!W##SX&M?Td0nP7ZYzUSFBF@?`dSRbxDPFZ0RB>&jP zJTX>{dPGt6@tNylh1OIp(2ck*cB0qhBsrF_+C@ds8zY}zO!hQ+0W;p!CwpWMuLSL;xjGGd<0Ip8Yy@}3$gt_D>deJN)Z-;e)dsGd zCSBtHv-IOa{QSF>o|g@##sW5`fgPb6rPD7tnijzRf56KqEnxhA>QxFe!1jM-<*>W} z?*9=nQbGW`|7lv?VgT;{J$^_5EdQr}S;+x5{*QYx((PKkh5hlPO*q*O>-!!sr2+=o z?5F~e0S9W^oQWZN9sjPtZImLd`AJGt7`j-#j<89sTBy>@u!yE{0b*0X)#$uTq`3aI zr4AhQkqt}F3wtC3@i1{dnn-1_`1vVIC`<@atg+3_Jz46jDK0VG zJKG;@TcS_ z#{$qOIjD_WMR?MT^L$C=w|1D*OOhY~bbIq{dPuC~h=O~WfW}%H$B-r>w=t4jh3s6EKdn%W&l70c88NEU>|s}^-TLVP z*4OqKyjwE&s8pVWpHbCODcu~QUwA;w3RFax7WM)kpB)V>#t!r~C8avUXu=Bn977k3 zi$oO!82aCvRNcF~oTobSbt+fFqYtK@9ghD7G^ZS`Vy4ighU#^ZW;&yBFQcNO2kBa~ zqyk5Fc7>myy!ke$40&JoT+OsE6iM>r2xl0Y>(*W8w!FDiU|{r=Ewiv;KWKm!x&b0< zQlVXyaz<#2hTLS`mZ}VX6gVa61NcDyK%znHYWXXS z`hSNVcNf-sKI|%~AHmP`zrWz{ChEKt`-I^hN0AU7`)U5_3^XTo$lH{O2PYCMnZ>TF zo1kJoTtq%CCd^$t-;<12cmhVYHg*VdH1N-Eo?dT(izh|cZl~^zYnCPtXEc>snKWsc zo`})RIqWy21*2d$WdZR&@Lyk#an_u`@N2aXA>M@*5s0;WFqM7Wf z3%~_+`)W?o1)PBVS7TqX48}V8R^b&f|G&mQtpbMFT&D-P{@+Y1G6t0YUl?*dd~!%1 z+>ak`g8w(jVAa4NlWkPN;F?1%026SK|4}_ueVIv;wcP-$!0-Ie|1>l8E_(lg2&ouo zwZ`s8S|e{C%r$G?EP%kR3KWz!2#*vwlc0|Jx9svvtnPep=4@k*L%_J0XB|_gD{W*8L zHgy#)*}MAWvQ!e@c1W5HM~Ff=ep8 z8&lwCvD31iK5P+--8(yY?bvD2`^-064qeE~d&a2>q$WiU8&^101)+wRiQH}tcri-J z<*K#&<>$x8jHSKViLd>2zbP+Fvreg2YM4kaRiqtz$FQZlx#{V`{th>rI$*7c3;-`_ z5j1B;Sq&=yQ^lnGCosVI8W6&L`9h_q&J)_-Zs!1M=Y7^>HzZh=x}zpeRoCVtQ=nH* z<}D@(TnKchP@OX`E07SR$^(0`f#Vc#6O$Ti83G2xfhHp{Q7owEKzij=T$00uhF$w?=K0GlIbHZ3BR$gWju925`)9#Yj>!n_d{==0oV+ zCYJb2zmT%WL-nE-WPC(xAj}T@oq~oljVf^hmdhWjpiuj6IMf~lz*udh30im>>;kkg zJY(N0PCyy;uhP>KVk|LL<~&Mh_HKzuHI%BL%b1#5dTePl)U5u<_aBh8E~;qAMA+c# zZ2Vlds?tf5xR1DFm1)}BY=-3^cnk{;Wk#2gMZ2df5sFik%@0I%LQA!LxjopsCpg9c znp97d-2c6%5)yxNJ^6-S8v;w?4N`DqOCCinO)4#c+bGfvW;VQH*`RwRKl7yBC&c3; z&K*1R%%bqE$t-Utzw4}uopPV3!6H1=`1GOF4r z7g0q6?JJsL3zF*xqYA?ZSWUv$%Gq*(XoG`k6FIYKC;)i4XvJG#QrLrr@`n~Os5b&xcth)~+qckq)=+z<4;j%XwnI#6S=opMJn#QM(nmf&eOan^fr zt5ERMc^ec!-Ec~i_us>49c)RrY->i_uu=|8ZJQRdmGB8|U9qJpLBNhx%rKZ$p;RW0 z;LXyuNGNa!Jg=b3vnBKXfDao2)y@goW`#7wD3vNm<1-y(ga^2k#bSxPDXe;};73-n zV|b=yRFlaP%O65RJ0WOP3|FWu6Iu+`s;}o_x%M|lzpIM|R;tIq`Up}j5!SJ9G@ZZ{ z0@#S-Rv#3xh_fQZxW<`xRwhTVWx{)#0IGo@tYV`Q*Pu?o$$4Cs9z$QCEQ}3ZpHDf5 z%;1i5*B1Bw68t*5L0oOHGfXpa+VV!3$R=Je`O8SNv{hOoO#gw^-R!@xHF4RW3-y7h z+vY@iHX+yX4X@S)w#@NNO8Gc^T6Oo={&MlN z!}RdVT5Wmvo!vA7`d=j<58h%MP<|iwLP>k9APlmF*vZpRd8clGdVh^YF=*JLXgV5~3$rVjg%WlTgUs@u*qoZC6;1WwfesI zFr-zxizcQGxm=I-jiQt8Q#P&@PLei)FP}Ab>kbe5l_RSc263a0rw{*{mDTfZ`cSW` zU<6AbH^{FbS?guLM~bsgo?iFGM;owgmEV&VqgiB&zv|vw7w>ugD?(kUYs&9`ZyNrq zW)#Fm3uR64b1p(5R!a*P(@$E=+x(6!J|r&=4YV+^3=aSE`G&tPX_EjZrKfgrUl?le zP4agcP`oosM5S<&i6JX9%8DgQz{L|sIRYg`=ZK%fNi=j(wd3r|9t3mL=}Ssc0RxoO zti|B)phW(L0CKuDO*<9G_$19B7%D2%2zbA~6s4zpN|U0JpiGsh{C(sB2qe2C3^B%7 zabf!4*%W3T_6)3IEV@XQpI}_`V%p9-`gf)-%{Z*=wJKpT+;>0oeDb+>$goFX#Ub`T z^=;+k43mTW0E9q*m}(1a)QV$k0c%Y2V+Hl(BMSDd4$W?HBu#tu=}@z@kjugt)A>@!agKt z`(JA2WHQc&%)W@1qZsp~AZtd6p@xR8#^$a;q%;<`&PN~%eLR91PHup~RS0fuTv0JA zgh+-8Bp+cgvM(dDNLdDSQK`6IYK!fnqLfB`a(M1fVJrm5ad=Q1Xe)3^o;9n!L;t$# z28+8><%KkaMhpye0Tuh9GZ)a`JE*OV#qUIywthVK) zoGF3WYF)X}zt=C%o7v+3ApEVElhg z@V3I-;G1b#5QgXgbWh9Wuj#?>?FyJJ4PNL*r+?0(UEGcs1sz)NQ`~D=oo-E2FXNx% zqPO$TQ#=l)>9|eb)uUZ~;Z`0leB0_-TYhX!tL4K_t3)U&+EdPJmu!yz(v_-htakp@ z*p*W^8kIr~JvRfl`BALsxmb_k_lSo9YVH3Hy?w$ePM@k;^Gh(vBx5Q5F`TpZikV6=n7gmTz=44n)i!QG_2l)>%@c zJBY!-IgphzH*|D%e;M0+$^Lsd*9o!Q)M?Lan_LIJVxrqo-$E}2d}k!IJW1}> ztgl|)zc;9&+R$k_;0~&Lvi>G(^uBrdLgoo zx5=$}SLuOPn;s3vTbY?Hl}TN1GMwGb`hB~mW`iP|P)@S}J)+QsM8^>hw8~`Dm-sUX z8lE*sxszfwhjH;rL#s3g+TmA_~vua9if;0_nY8fk#QgVu*{qhRtSPdg9bnZ+ZD+VwCiEEHS zMdrhEC>Jqi)e)b3aGs5PSKU8YLj+-I+2VsY4-_MLB~RblK=j0Hr9==ljU$)^jWK#- zSx%>pGU}Q^lvEB6h+VWR%K)PxI!#+ja* z4(iLeXek9`9F`MWtL%F*TOv6F5mwCxWshOYl@M#x{*pUX^ym-y|KV}g3hRh2;?>Sw zuAWW)Csoze2z(nE5;pHE=kOYUshs-YO%+)!4ZjnsVJ?K`ipt|sRp-JZ^+ftmLVV;4 zx96Tl&9}stb@&yDo^sZBI>aq!m)p{UY$#WVTn^Iu%ia0W3C%Ul7F{0)s6i?Lzb2uh z!u1>(N9H~%9UDV^`!GYnOO^{u1j;<`M)+iuO9uCCA^Y3p+nCe%?bWF|H!rw>R{1vf zay#P$Qo_WIu#LvR2wss_8KbKa+G8I{pe2r*x1c6&qz;+o z8Cf9aPfKF`O=Og-j<81sH2PD~U@=VbaVfIRfFPabs+VFV&8=kc+jO>{&^emtv9S2$*)f%UV=ko=lK@S*fS&olCr3fpL3mE=jK)s|+#@KY4Hd!U`+7i* zI^}uJr8PxjE&c~33~mo7LY@Ng6Cz)RVp;*E1Caee9x&;|@}c!fIwjROn}EE&Rf4C+ z{4HT~&Jza*4Qh!B_z@ku;LuHKn7@+|!;t5|DTR$n9iRt}w)YSuS#VnGj3wwI3 z-g(_dxM(g)--rlB)N*v$alVvia|l?QNS5^l9-qy+H~(xmN+lH;Q%`Ny z*hMCX45IW<4DlzdEtqqfv;1Sjox6-@oKo026b6k?o09Tqz*JEC<|r}p>_KeF6XtTk z8m|un*?g};x9o_nJ3JDf3u4mW(u|LP0VCt^Jj9Z9?5SSZ7mOfe6O`0dEhHV(2FMvG z+-v)w;GF&q9FY=mT$&$47ct^xD0;`zA_5tO^LSjsk!TMy;m0j%+$?uQoSBqDC?xVO}i zSVQueZMwNG)1Us22hKMc`-ozCD2wvAqw}x;hKvBzKzuuD;yz@o=s78dz(HwADqX76 z?tQ;}xtD3*BO?Wuf#Tbu)yrC&*n&{|W+`GUIwMu0FE73czA_$v>p}`;fC5x6A@N@Iy zW$^(&k8bks5VxrDS($1N@T)3sEC?1k=)6}j3hBZE%odWg>g!xPq_rAYA!sA}UfXd@ z3>E-+8e1ssPtvio^S5{JT}n@!cbu~oI?bdHex1yl z`l2sUR^n9fIAwzU&-(^K80jSZItn}Wd}|M*Xaw@y@Q6KKajxI}9AE2@e>l&%8h&Hg-v9V!XTO8J)X)FJ0x5o7b$|1$bRe`e z)7B5TdUP{-{oIbG;a+CpK1oyPIO-~rWudW$!0j6{=~~0%#yrY2hFZz3x94^q_X2XH zNEhd7az!daeAtkebP=kzfNk>Ro=N}~@aO^s)ewYwpGvW#RxQ*sz|z3*mfRq2qKsHE z0t3MRyDrbIYv3yeS*<`uAn*FkQIb5t5Y(8WH=nEIgfWByO5T^0EAf`}ujmvsq<%d| z=)GEnEGGMj6uqkBg0Lya8)Z3Mz@ihkLTLTJ(v3K6==^2Xu6$;fsPL8p^HRh!pv6WNyVG1-&JsO&6 zxEz83TWXzO7+I`TjlNt9p`;ktXzG$9$~={PGELPlriZhT4sb#kdLJ30124l1LZq1RZY2&V9{Q@zaI zK~b0}p-~Fs$ZC>VRH|cks7AK2w<1wvq6NvE!i{+!v(x$h5(`#5yH^q6|$lR1cxZEj=H3HWrFts&`DDjy zCzY{O0h^)tW`jn5_d!p=zx7-i`S8L{o~?kcYe$c@KA3_=HtiK*h=*cX!kZ81jn6sP zU6zmAvaUlY22`U!`hqpP%n8q{EcUWn*h-CC=%4;&0T!}ZEFVEwzzu&!TR{vX_}itP zIok}upo!ta^{6!JimVZa<_kf9_@{{1s#M9_3%SZ|j?76r?? z!;Al(CHf-_EB3NFi=j#$!-3XtQN=+0zAj*|85E+>t;jJmqKGS742CwXh25%RJ|QU# z<^picX%u=02~LmS$?2|@tXc=41d&emuKS)NFIBNAbpaw(O%v=ojSm{j3gOg-oRz3$ zpXEg^)Un)ia5A1ma`9V0t!kV@ud}7!i6Rgjs8ns%16_Atei0k5k9@qJC zWLa2`J{B1Y9?3BpAFpt`L$@1{Hif0Nb6W3d9trDY`!_ix={B4+>%BQ-IR@*aLR?>v zTZ(had^gre$@ujE3Sul!cF-IGTJGJ>H&FwpoSOteM)@hy-lav}^rH*dHjU%SWRrRT z92h*Z3`+2F{BKginZf~_6HI!*X^-r(YbWwTKsOuD`BVyoCy<8H)t}V0uHLI>o(D%fAuX8!SmPZ=axfbb_;a*f*e zc7%Bo02&Oa7(LI4_$`>!`0a!>n5}5H@Yrn!#B-uLf>p- ztxKjkZ_=Aezu9_)V4Lt}z3|?-4dv!>4YwtchCr0OL^4s6re=aY+2_S;Af?J+Kvg-# zn%YUaI^q)2{fF4JN4G^9@b1m^`h_dbiO;$=tOyi4-o)wXr;)-^JEK)_QrB;VT=WZ< zQbZfrr}h?Y5NabFDG~pzg7HXK?9dlmn`Jmc&w?$ky79aFcDp_T^>^mH4vRN7wvzss zwcy|w?4<*Q&tHgPN8gGe;kT^6S!8IYEy&E_a@G*~Mt;JFkd6P^har0HGhcuDk%js- zFU0_}$H=!6HjmMM7-5_VaJ$M;<)Naz7c*+ae(1(Tp|`DphrRq_{c?_z4!g}jx<|U# z?&x$4K7mDsRyt3ku7n5pf)b!ige;88j1s# zV;duq1ET6A)wNVtUPlf0b$PZ0z z6@5yQSf{OgQW!**(QK{c{*tGGJfJg5nnC7!9i#Z;%GrwIxlzlrR!Cm>s7=AyIC25^ zQ&jMJtkP?!=KT2%8|-yS62lC2>hI_Yw9pbKK*R3)xOks#w95qsN$I z#>pXP(9uTw9g-adSQsAnz5FZV&7T7P(7D1fI1gr8hH!YYli*lx?p_vt( z4JXA-pT$h~Ml=_dMZ1=UevsP(DV;N}QF;FocLE4tOL>~AA*XW=Pk8b8zPHsE5`A+b zLE+N#cxU%w&x+x{10|xm{y`AGb=h6Ho45bGavw|m4)K1?SKXBMa)DD$m#bTrBFGEA z^4LV$KUK3`N~!hQgc9EpI1?aM|2@_)G`00!q+K_`tZ*Kmf- zsLiVuL$tPe_`U2m^rUn9U~7dD+DyR55!IX+fq4heUWkn{9_rnGRl6Aqfjb4%unPxE zrjI0>nr39Kcwn%vz!xA1qG(xz<0{vWu#5yY3wYU!GNs=-D`9g?QJwCH3iW>HY{r(m zLxRITXODjHn9%93t4wBgbwqQ*_L1XZv1FCc5(b(D+Y2)!3C|e*Zd#i?O{zBUn2t<5 zW@5gg;V2USq~csHqhpM{Lm4EsiT;BVbLhKO{B@^E`&|-5AD`;GU*rO*FI9C(2LByz zSP6ZX9=e0#Psz%Ic%-zTYFhJkeiP46keTtz?cF6gsTV+wdc+2EvF4}#e*atMM~hhg ztk}NykK~zN04mT_Vy+++P}z{?g>6=-efcXAmCF{pf;*^DHofc&sEY&n6An>=S(EB* zPN~1u>-bR+tyYaVf+YOaffqmO$~v>xl(QE1qHEGpaw(RFg+F#UB*6h!?*K)WWMTM9 zHgSfr4S*VQrhTS@%~k#lufKnt*0{EPu=578oV_6p@Jt3WU^ zrdKrp*4!u48H9Z_Hi87v4i3|atTs$xf{gJRRh_wqK5())@m0wz$RJq+$wM8J2IddS z7~QY!fVJilE@m%^tA=7VBEbhhwraWQ{q+&@RM*mMtz%6k8neXh|Y6oOYSe>sr{8(L~1)^*N6 ze9$28=n(oRT{&allDfPYIX5m|`OgK0#m=51U4dvyF#O;*-eK_MW1Q%3+7iGJE0oV3 z^HcjV0O1=c=#`x#(9@5r%UTVL((`B()aW^# zXaUk<36CkkE0vLt+^IPW%i3u7D3H>OU{a#?=IbIKng=24SX53@A2XJoJlx%GD&Bnl{y(lU`x@X& z39)2!c-gS9V>Jl-*fB6kVKYT5RWK>rt(imx`)_}8z95GA3!f7`&>#q)J}8!!{ku1d!64Ss=QAL)Y1i3kt3 z3f`OPPU?SE%gvSyIl&4!q>38Ja=e7`I1TA6lWh0QP#IoUhP`6dc;L*zJa@dOp$of7 zbhYKwkl6wSmAlRruj&myztlQIFNhhgs#i<#seJQ`Ls!$uB$f}5_8>v>aj2ccJ}H6T z+Av}~;EcFivR4%GwGI|K@a-s<#$Mv~J$}r^d>}N^ctSm6*ygRpvD{!xHjaO76?XjT zZCZ_5lRWEb7vgM^eEEpBC({1D3AH2EA--^GTbN3YHJU=5uQiL=5l*-WuPmV$L_Ohau{iq zI>ZvVXJF=&^vNPh(^BxY($C?ckFQ7Ivc>Ig4Ak@SYxwuC&3wc3@|P-=Y1H$n|Ex<@ zJlJ`gB#dhC$L(3cxAOJ4xY)7&)WxkY+<)Z07d5th!3>uI3O0{YH+b@Q52k>2uf@Au zMqc8#8Gd!}yv4LmkkeEa&p)!1ZF7E3Q%N@-bAA8n_xFXk;{;zmu4zekLcU};T_UMy zyqAyN%+Y!y=0Q&WopgT(eqLR42ggy0FX+|Z*Qdh8luwndcl3{>W6{qYOZ@vv{F}#( zK#>2+9?9Ps44mJ2=66)UiVF8L$#if~uQmUURS2`fM76g0&GJO*c-T-;TjF&yb=8k6 z9%p21WlF$GRn!$tPb!wjXI3hKqz@`Gz!%HtbaiIjUvCcuJu-tamp2IK<9>+u!4u=i zF{zjk54lKXES7{wg?8T>lvSVC*$0%Y*Vj#`vukb?t6jy9n_7)a*vn;)=R) zJY+@D%|C{sS3vU@h%Vsb-N(4a_9a*EY1$6=8U4+t)}dqY@-lbIKL7;{0*3YVD`wfn zMJJND-7k1N{manLZH&e}8`Pg)0%q>ti zZmTbSFr!SUC%(0TdRF=4oGw=iO0vs~gyIQd|%Q)WRoE*Z3r|V?5u9#<>fUY{gn%MTl(&7sn7b*auJ0t;B+5n&P%)JA` zUXAdp6AAq(vH0fnrxNLnsZc7}ES20N7a}v$K$D@<-EV7h#>O=0cW4u&5;%p`0KlJR z(`3Ecw5E*KQy3X5`4^wLXE@BG04A4!+UqGA;KC9|V^UKzTRVbncKB^tispIECLu3> z_jUE#=I7#|UX8`@I!b%wU%DC#k4ryoo<4>i>-``9d9#YvUbm6HSd-J#Uq*jC{`*^C z!Zt(Esyy};*PU+t3go)I;l+-mCH3M|8g!-m>J$8$`^SD@fHK_QpWC-x2O5uSVC@J% zAcv^CEUdM8?WW;ltUmB3)&b$xD&+TTv)>pVWG!d z)-p8;VQqzB5;syxF5iq)L-sAwDAc=TjRH|l0Qri4l z*V<>uBtp?WmR{`>^66Y?W|K>w@50Ns(SSW++tqW?U46RtNeJ}1qDJ#ay1=Y6qiy=) zCWhzqjxQI62mb$30SD@{*7@I^C;xu`QDIq^0)dufktG*wU(56=V=doI^@_x`^;gv= zS$AYk%LHy)5TYVWfXZ?!9g)fQeDBTgJ4`Y~D)y0vhE!kk5(kNoo?Qxcilb=m8C}TU zQ{0l!9xnV1!st;DH4ftkBCBzbvMczYv19mA)Jpo7GU2wmOkG}Yo-8@S{wo7%X4!Gl zp26sEH-$UG-fC_mq=K`VAUMD2&=LOB0ChR~hqINg;r!00m`#S1&3&_MlXBaY^G|!E z5^j!0wOu9r87+{DvxyO681?9qUJse5g|n4CII{stk-&4Xa$9%;h%SP>NM_^V3*QcT z--K%T#!Gljfj-bKkVdCKbV#e+zGv}i6^V}+Xl6(lnI6vMJJf(zY5N1}MA@kX3mF$t z?*!GBAYUOSxYSM^ZYv~|(t|Z7kD6F{AK#mBFc!L01OZC=eDvl=Y?T3+?EXZdZ%xh$ ztQjIAJ6nQ8PAxe*kWR=jA{n0P*aS;;)!H9VTFqv|tHVIlaP`d<4)X#6xN=j4>2DYYeP|1tA(=gK&F_B8CNZ&(uQBKSE&=@b1|G4?|%j8m{? zhWU)Id<=DUwZ0uUm1%JQX_yPI=qIJWGIefXZ>g9Q6Q^px<1O#Tj*8^3&|Z-7$~}zM z47NUX%}mF*->>@%-osd}$LB7$xMZoLDbKerYVJ6Wz|KoBDI+~FpF0u3)9$n94$_83 zox!t>>eeQ2O@~bUc7E^Iag9K8EbX$|I-3T+Gzec?H#>!VT#dfybp-$ z)2Y>UwLcPSY05M@Ydx&ZZ%GaYtOc(ZsVmU~8SVn|Vq>-`C-Tm|p7`L>|g`4$q(?)y;6heqabs&5y zSXm*a;E5UU*Uk<@>FP^V?o@SZqmW`$y(Jy^t82*~W_Kyg@`HHrz)K>aAIYw@61?X* zEAA$o*gx?6D2PGlCX5_vFHcw*!dtp`&+VNQU}rZMucpPSkA%f=fJwTCjD^^yv6$YQ zscZbY_2X(SX%3uq_u=hKhIH|`ipBKbH`J#&j```ovcc!`pW_Ey{24;qtmrwe!k>2I zkF^yVqt3CD84$Yi+Kbj*@LnuB>IeDwxb<`@96YQCKhdNsAbu0jTbqzHBRoZHZcLR{ z1L?G1K55)qb~oh@cEW4HWHcyDk_3K=O7P>S@|`|=W(N~#ojq+LJNDAu)1{k1jjFr687AcUhT5+t4b z;R^x7KFuV0u6*Jtv0F5KwO{x~ma%RYq$*q0cp+2A*MQaAuGckP5B=cmkIu`uQe z<#9!>C+A7ZLFW?rkJOk2P1*kGke={pFd9tj^8;t19)#={+7~Asj40=-2buf}7&_(~z|G0ES_8j=*Z_giitt=HG5k3pKrN zr-f~BkK*yr%&_QFD5-F`_n(~2x3-HoX)Nl+8`!`VPTfIE`tTC^uqJSBuon87;-+aY z=-tLaihCOFF)9C`!S2{R0s6$^Tx>gyAs{y`ORajDvLxQ@OM5ipRRbR-?a#@_Q`E*h ziM(uE8d3g`AO=_Gkmz)0MY(p$i{s+glSDN+i6}_n2U^MDjHA{(GgXXsMAthRy*Z!s zGVuIpy^-)Q=Lr`4_`^rn_+9=nO+H{h!$hg2_T#hw>U{qc+R*FI} z=iQl3;g>4ZTYqZbJTr|(s)eJFLmz_HNht3j7ScsHJm1yd9k#jF^Ui#2k~ zjRUlEB?OKLi7FiDqynVHNs==uDwIUz z(jicUoa(8zgiv8n%SrB!2Yhl6cozFFodSnxG*NT;ALjds+nkoY#>0~OuB!yZv*O#p zO&9oS9+Bf``Ol?Xq_Zj(XMcj=sTCTduj$JEvCn4B;3?Nj_wyNv+nXu!nh+&&WJqFYlq46Q9Y6BU@q`Davyx-SI0^3u3j z4PM6Wnj%WkA}+-c%ppTXpIO=ds)$xoT)D{i8f#V*H8)K?_Tg+a-=de4P=JU;DCy4Y zyij%cG;#cnakF=SFnZnpQ*wE98m*#A&inZ?SuAw{I2`(O)T?JbM|+fN-s6G&8TTDO z%324Z`}gJtWQDUy+L@?5W4mS|Ug+T$=zs1m2jA5Qd+ey**}A*Ni)iiUkWfGS9fcQ? z&2-hY%{<&h$!|!}hvckG{;E85FFVtx!|FlwiIVJqkPRI_XZ-VgJpS41sK-F}gDc3{ zFCTIiSW4y=RMF#NYx{Fz7=x;SkW?Y0l&ocv(yhsFdJJ6x+|Uoqb;6XADZeZK_$a9B zXL#^HWrdwg?gVF@&p`|{C|R0Dg1TS?<=&Z964ZbRC{*soB&jCh+PpA8Hh&Es8e)e_q zcy^?5jM8qT1Wr~6eJ7;E2vJgXaw+65cAVMr7bWk>}BzJs3k!dM^=L)VOk1)esJE?02;tnjZkLww|2zr%YBx zU==h%uk6Q7p(ZI2IeO3RLRjns%)S4XnUeD z+6COgsHVO{st$s9aZwddTet0*RKEtBkV0g`S#dQY4YQ4YT2ABRoP2WB?PP}DHr_qo zu^`_Q&_p+OBm@;9gJnzpAg(09aSJU3xIl{EAoGOB$26n<uSfUH@l_cDjJQZY zR|d3UNa%=(d}~Ojb{3mn*-u2Z5%fpB`fW$^WuokQ03T7-ztq3empjEcMl_qhgZ=bA zgqh0Zf(Dd>g~k+7YRR;6Xu(2u*p6`8dojj3GO&vY^FfgknKMw(V8za#YTpYGfP`$7 zT&YXVO6vg8OGqix;S8hF%2@o3)-6Nkw(bW@C+df_h@=&0Et{3W?O^u&M5C#Rj|?}p zYXgh1T^cpC?KavM8;ty~CbG@62P_!gcR61h@qPNwM`~Nz%AI}j84E5-YW_pt+$R~6 z)J2Y*C~D0#ouZMni>ch&=6;$v;Gf7d&G?n&g~lB>47*Kpcepk4xb4dV-00x%j({}R zr{UP8qVAuCjYpOUW>y+zSx)L!;;L5InL4WNiI0?dYxhsiqE76CXx4Fr~ zFe{Szte0Pnk5R$? zbaL}2ymWF;V>T8}Wn(6eK*7~8{M^?!XF+$}85=6R#$BerlZ27RrD@Ijy!IXbgExiU zGzw#4W385-fU&1|m=c#~yc^*E-BSK++YoagdLLKU;C|N~XtN7==isikCpR&~R?QW9 zX{Q>kI2l+jKa%H^eR&KUuL}H$`;oj6df-c|Gre**dHga2X$-tXW8)oLfF#CI@3lmN zqW48M<_w4ah3(ssHVH{^j>{>&vq%>iAixKQ`tC~peFL+_1atFaA ztx=f-#Jp)Ib%d&m;CRc!DUC@*hsm6Ql)^|48BdaD_u9#v|46tr znsb8ut1XYKWdSHJ8(hySBWq{gmV)gl-#x8hD znrG=*^-aO3ccvAWHF*^dpMVP*EQ#|9fZk~FE|fuBwUZPfjpZzdf!%0?wpC#sIZk`7 zshV|@d$O>^n>>*SzBG32AV?4_KjAnFWm_en#tAIGG{QaJKs%hdWBbe#{3JK%z3Y=q zsB@)UU>x$_9}bepbf`XK>Fc_Vsa&{M|xO%Kel3YkBd(@;FAv_AWXm@vsqzjHmY;_$QoCj}@K{sQuvQ zR?=>y_a$modA?k86j#nKhMtbrt~b2D$gWc|{$BDkSKl=9#=iNQz+4^lV-NGZozD2R zdJpD`(ARazQd#I%EF||{6EQ!Sp|%77LO}`iV{z3W>z{e8rvH6 z?2Ex?pL1fWj}OE*%ze4-*|o;VAjVze!;L0{<3^?SrqDU@7~6E1jk`{)9kB~I!DMre zOc;1oqF$&q6sGw;yH99VWYxz^Q^WJBv+pf0_6%3VNr{`ukF@2lsq2t1_d~cC%-?uA9I|7X7!vzsS`Yx8dMf)|!omZL1>60)C26qC{hO z^Q%yD*WUgud!t2NpH~w-f-ZTo2UyoL1Rt_!%lS3>a)1fmA#r7Tra15}xEZR@Tj2+-j;rhxzM;euaChw? zWJ8(NrsK62FPA(%a>X^}84@^GX=iTxOYcA8z_8mvFW5;dwn3sqr#SFhmKO3 z+p|-Q_#V|>Sz~HOvQ(*|q&({p=%n=YyD~U(k=OgF9y7g}n#AIioWosZHps5ku77=Z zIwhU=_=(gM0PpobL3dfE_?7EfFjYTIgq!h^nJy)xb(}>-201^t~xvpV6nM;!T?x}8&FQj#2SZ5 z*fEmBf7N`jD2Zn7Yb^W@Kd4V^ka?{DH!J!v>BFZ#H`*52^9`q#hmiOjtXALtq;AjM zq1q+F{$RHI%|)X!y|kWCl5Ph>dBe39)ajA(x;%hc*?vgWSeKl;Wr+r)luB~5b+`0M%3NmzdrmX-K= zR7^cIwxdlKN+wvesPYv%i18Jv3EG zdf>h>`R=u?r*HZ``*_=Ps$2QBUVoZG%(XYE>%<+4Jl^f-9MujJ*x~Dx#E@TD-%_nH zB`NJ}%`K3mMd5xJm)KynAj_mWNB+M2!v;$B%_w&v8ty5n;>H4yNM-*{5-kW@}Vm!Pd zf$aLdmTVc#Y**WnqvIpjzgGF~_F&+sKC?}4u5Z_8xB1ywKVRP4)k@d;f1OQmbDWhP zeU&o#ZjfC1@v-*jf<{@9)l(ZpR$uPR^YmcLkRJJ_{9tJ4b}>m|K;-GZx_UvapliN6 z|NJq$w1u5Hg1h>sSMXcutsXlr##O5*bbP5P`aIENQr(Y4gdeEjZ<%z9e-oVZn3Mb7 zxmz4!^AhD-1GhRj+8aP@Z~Dm_IM-S9>5-K1goNR_gTDwlCarD4T6V)iLD5 zt%@Y!KCu`+E56a)!FOzVH@No$k|HJzjf`h;(e~-+lFG zBDbFO#Upo2WhnQaALhTG2L#B3xyQ^Ed?f+CjA{zbJ-nx9@Lk^!OC`p&SvnRn=`yD_x*z+T&c_h+ChhP%d0=m9qqw?`_T7hg&1PA1YuC%1~CZ0_EnLK=?$?R#~K;yYTYZxY+ z?qww;bK1s#jc-iU%WXPkmsVdm)@Zh(>rjV>e_d1@$#u}?=o`N*8@9jr{!&)2irB0~n$A_~$vRSE2mno6?>WR6iUwnC^>6ab3Elp)gkGc~Q zJaS79u6KP(8esp{p?7NE&-jo!H%tJ}voLJu$94!fwntrS6FJYA1e&!Lq&I7+8Q5kKrQtssD^(sQ6Vk48w_jT17 z-{`uoaHXKCN+&F147811zuH)IJ;*)7sVj3VQF9F8I&Z_tH`8?FLhY-M2B+0(^KG98 zoG*<{CKlyvD&3bleXB2etFt$8?kc;%P z9D>;ow_K~tk*sO>Sdn?C!t0lub*AJxzHX)`wHnH-GUD7<$vas;y_5da@aA#;%c+P6 zt#aY)Q${fr!W&OI7|r}lIGC;>^O!YWVmAw&`xo}<1MA|YbHCq+|H4;HQOO=++q{lD z^pBeL3!`~!S?jchz%6-Q#TT>#6PZgqjJwELYLK*t+=w@UpuDc=+7tQSnCExJ^`iZeUrru z5eEkEi7V&KGrfK0$HG9L-6nHt@G|dVwX_RmSNd3=-B$YGqS*JyOu-AVW*UF8RP?L$@_ zsTvkDeAxB78rwp`&#`&I^W_x>ns11&xMp+E>f~tN)6Vsyj=^gW5tLgRcNp88G&?Kn znX*neVs`A0mCZL9XYV2=E&2n=GV6Lqv+C$JRZp(G$wF4gtLMn?6*30LfvOsrOd$k_szJC3Z{B*i^?+;7OM^D3g zHgp`W8uHOCUorM7;gPj)!-3y_s&=;!&ty-1t&iPG@Sc;?;2WPIhZ39r6b%26Q0^TuWZ7I_ugr^ulaNpWU%HWdhGLtgLsQ08a+#P2iO%ET8Cw_=U5Q1kIwa zZZqRgm#CvZ>uaLj8{0p^01Q6Sn|t4giFw0SZt7L{!Y?Uu#y{z{5kt)0KYbhZRXS}) z9+XHm_*iv`=^Bkb?0dQK6^qv7*7|)G>Akx{k3D#!bdvArh)YnkV1jv8Q-RMmU9$nT zrdhUAHY=^gZnckn76aX%=*7qp-_){JD>*Y%CZ88(@j5Hf+%m>DZ7)%FAvHncVahOg z;N@v^B;VgJ(^Pb-L+V3PAH29tM`!Niw%M!^_8lFaeMhr=<(BOiv)I@%`zFP!od4U+ zxCU=0U(bA&m~XG?#d5G=(&`%5qCma6b3^$U3LqUoV(v1N@gfLQr|D+>=sN+9{iXeJS z=!G&xa@98#z%3C(8%K7=%wiCF*o=~G zwgMQ6E%Y{~C30aOytdvqi9q0_b_KuZ5yL>B7$T0TWZ=yzF(*YJxcCtWyfj2BK!F${ zjmvLATACw{2ol3Vg*YOEX{yLJe(Amvfe@lYB&?#5z>+3`7~qN;OF%_ADk#RM zrNBW6L;#0oqfqG@2o&_iK{2FSit!*XP?ALO`~b$15PI<`3Z1d`2nQ#j;Ct*iSd5#<2w0XASAZMZR2P0l30k=p2;`~5(O4dW9r*s@+b55R z;WD#P+>pRuI0Hc<|6ej^QG)HFe-YdlCE^apB{(TSf<2NrL9+safTIcsH;y}x;$F!v z;uh+|1PYEm( zH2z|`Ll2+IKs|#ot@L5#DTHo+2Hmc<$bts^i8ju}qEBQ6Vmd^w|Cl1KWLKd~uOJg` zPc?drOk)4hw`sY#q(J)Qe`0&zXpx>ZO$p({og=6OElq)U6{1n@G9~i;Q<#3VORMlh zY0r%b1SuM|Ddsp^w>gmmJaJzzvtHfV@gVe#1^3cC9CpCz*EkZ{LAnOj*__Hyg{!VO zZpxL&gN-~a@wqOWnOVb}rk5rxDOR_@A!Wn>^DF0EAY2t$ zi|JPF6!Ae7zU_J+?$v3hhsYCn|D#YSo;w_BP_2nGD3})*CI25EXKZw^V~1U4!Kk96WNJC{n(c?phF)mWHMG=wHR;-K!f zFpptoEw(%&sSGDg)IfwVB|k-10!g?;*crMsmmWZ zez=g#rQinF4`y)vKwC~M-r}T=-NfY}@9p0?w!`n_*&3+x7Z^pIMUBxI48Qw}$@)E! z@4rIixjrjF9J;T9i$H+uw8fT3Bz7+{DPiMml;F&FoM2oVk-@_9@}4Z^4fKl+)PQzM zJT-}vc1=R-HtSH+EL;bQxjK#Grl(;SXhFkd$@ts&poQQw3^DD*PB)L!|I|UmB+%tt z`}QmW3hWmCmYJZs=pxj!5hM~tt)B;2uR#qvC>FU3=&~aTT>lFK61l78T zJl@YRij(2`3n#9JNMW71J}_Zk`+qwt<^793Mvq!XF6cp7dIF2Mg%LXe!9+>Gs=qKE z(|&0EqCn$&;4+>zLFC0ijse04_KJ}N09k(_4CVs{FL%J?I}Hb=7>&^k(kQ(!Xwavw zR6F%yG>f%xFpt5)SZy`w%B6?3FiDs|kfD*lr;n2c>yubkE?voA$#9mKfovzmmrg7j z!9@c^151+NUs6Ar;U30IduftnTH@9WTaviIDr;)Otu%xxesRQc8x0YD%-Q+O1#NB6 z*_p7W!+C76~0$RUX^)p!wBJ{;|eAL`(Tn7HeX>1 zD@NV@&>e^3%hpkj^)<5C^WA^7M#4(^#JG5nJc2la)cc;NV2 z1}A3eMK_>Dvy~ojT0kS;nU8zxBQ8&s7`~Hdi3lvi&MbH`fxRWd zPb!thF;(hxjDTK-jtA`1UD!4pm zg2P?ekPbAN&`AIVYicGLT0?lM2?kdJ1#p{=4Ry2O=RH&{c9oT6}Cj^=o z%UcgBqtG3L^MT*)bZkIu9W|R&)hU;8I}70pY^zNppl?l>tn= z2+(z;4&DtYeCwwLc;zRUQ5WD-m$(mZph)RqI^a7@C-~32O7no_VldFMR0BAj7INUf zr^J?SrUQO~5+_6)eOHVVbMRj;iY;01oCPR2usuiC*5m8 z&-ve!k-CO5@(o5h2d2X+8tW+_+=Y6sSmFZvdbti2zyrq`Ju4V>p?Y@~#Tnbw}pfN zx^56BzXQjyyF+pCu5$rw(SY4@e#Q+}1|_%$M@2&Xa`4Oz;lzA}-ok%5oJtX>G_&Z8=&7GuU|T_DW7q9B0LBysB~Y5cCk<7mvk7b zW$@fsj>h*IzyG4=_Mjg6D0@I#IpAglwFQ$E03n!y^c|X_tuTgJ=1UW_8svB&>ez_2 zGfHK$ffE(4*3zYTD>nih@6ZbZj%Dx+PLT(?30)LmTzt1nJBy)uGw=%@WG=QmVvrC5 ja)c26f5wysFkK%72`|Af+BLr|NM4VuWndS9m5cB{6zYp` diff --git a/LightningWeb.Gettext.html b/LightningWeb.Gettext.html index efbbe5d982..b622a482d8 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 b964a1d035..1147cc4a96 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 8d6bb3c4ef..1f2191c2fc 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 03e01b82e6..53cb2d5804 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 42453b42a9..574e7c453b 100644 --- a/changelog.html +++ b/changelog.html @@ -142,7 +142,8 @@

  • 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

+first click. #976
  • Return the pre-filled workflow names +#971
  • v0-7-0-pre5-2023-07-28

    diff --git a/dist/search_items-04557D51.js b/dist/search_items-2F0D4F12.js similarity index 93% rename from dist/search_items-04557D51.js rename to dist/search_items-2F0D4F12.js index 860b546cd3..d2d2ee90a0 100644 --- a/dist/search_items-04557D51.js +++ b/dist/search_items-2F0D4F12.js @@ -1 +1 @@ -searchNodes=[{"type":"module","title":"Lightning","doc":"Lightning keeps the contexts that define your domain and business logic. Contexts are also responsible for managing your data, regardless if it comes from the database, an external API or others.","ref":"Lightning.html"},{"type":"module","title":"Lightning.AdaptorRegistry","doc":"Registry process to query and maintain a list of adaptors available for writing jobs. Currently it queries NPM for all modules in the @openfn organization and filters out modules that are known not to be adaptors. Usage # Starting the process AdaptorRegistry . start_link ( ) # Getting a list of all adaptors Lightning.AdaptorRegistry.AdaptorRegistry . all ( ) Caching By default the results are cached to disk, and will be reused every start. In order to disable or configure caching pass see: start_link/1 . The process uses :continue to return before the adaptors have been queried. This does mean that the first call to the process will be delayed until the handle_continue/2 has finished. Timeouts There is a 'general' timeout of 30s, this is used for GenServer calls like all/1 and also internally when the modules are being queried. NPM can be extremely fast to respond if the package is cached on their side, but can take a couple of seconds if not cached.","ref":"Lightning.AdaptorRegistry.html"},{"type":"function","title":"Lightning.AdaptorRegistry.all/1","doc":"Get the current in-process list of adaptors. This call will wait behind the :continue message when the process starts up, so it may take a while the first time it is called (and the list hasn't been fetched yet).","ref":"Lightning.AdaptorRegistry.html#all/1"},{"type":"function","title":"Lightning.AdaptorRegistry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorRegistry.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorRegistry.fetch/0","doc":"Fetch a list of packages for the @openfn organisation","ref":"Lightning.AdaptorRegistry.html#fetch/0"},{"type":"function","title":"Lightning.AdaptorRegistry.latest_for/2","doc":"Get a latest version for a given module.","ref":"Lightning.AdaptorRegistry.html#latest_for/2"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_adaptor/1","doc":"","ref":"Lightning.AdaptorRegistry.html#resolve_adaptor/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name/1","doc":"Destructures an NPM style package name into module name and version. Example iex> resolve_package_name ( "@openfn/language-salesforce@1.2.3" ) { "@openfn/language-salesforce" , "1.2.3" } iex> resolve_package_name ( "@openfn/language-salesforce" ) { "@openfn/language-salesforce" , nil }","ref":"Lightning.AdaptorRegistry.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name!/1","doc":"Same as resolve_package_name/1 except will throw an exception if a package name cannot be matched.","ref":"Lightning.AdaptorRegistry.html#resolve_package_name!/1"},{"type":"function","title":"Lightning.AdaptorRegistry.start_link/1","doc":"Starts the AdaptorRegistry Options :use_cache (defaults to false) - stores the last set of results on disk and uses the cached file for every subsequent start. It can either be a boolean, or a string - the latter being a file path to set where the cache file is located. :name (defaults to AdaptorRegistry) - the name of the process, useful for testing and/or running multiple versions of the registry","ref":"Lightning.AdaptorRegistry.html#start_link/1"},{"type":"function","title":"Lightning.AdaptorRegistry.versions_for/2","doc":"Get a list of versions for a given module.","ref":"Lightning.AdaptorRegistry.html#versions_for/2"},{"type":"module","title":"Lightning.AdaptorRegistry.Npm","doc":"NPM API functions","ref":"Lightning.AdaptorRegistry.Npm.html"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.package_detail/1","doc":"Retrieve all details for an NPM package","ref":"Lightning.AdaptorRegistry.Npm.html#package_detail/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_body/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_options/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_params/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_url/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AdaptorRegistry.Npm.html#request/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AdaptorRegistry.Npm.html#request/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AdaptorRegistry.Npm.html#request!/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AdaptorRegistry.Npm.html#start/0"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#stream_next/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.user_packages/1","doc":"Retrieve all packages for a given user or organization. Return empty list if application cannot connect to NPM. (E.g., because it's started offline.)","ref":"Lightning.AdaptorRegistry.Npm.html#user_packages/1"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.body/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:body/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.headers/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:headers/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.method/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:method/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.options/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:options/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.params/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:params/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.request/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:request/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.url/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:url/0"},{"type":"module","title":"Lightning.AdaptorService","doc":"The Adaptor Service is use to query and install adaptors in order to run jobs. On startup, it queries the filesystem for package.json files and builds up a list of available adaptors. Configuration The service requires at least :adaptors_path , which is used to both query which adaptors are installed and when to install new adaptors. Another optional setting is: :repo , which must point at a module that will be used to do the querying and installing. Installing Adaptors Using the install/2 function an adaptor can be installed, which will also add it to the list of available adaptors. The adaptor is marked as :installing , to allow for conditional behaviour elsewhere such as delaying or rejecting processing until the adaptor becomes available. Looking up adaptors The module leans on Elixir's built-in Version module to provide version lookups. When looking up an adaptor, either a string or a tuple can be used. In the case of requesting the latest version, any one of these will return the latest version the service is aware of. @openfn/language-http @openfn/language-http@latest {"@openfn/language-http", nil} {"@openfn/language-http", "latest"} {~r/language-http/, "latest"} You can also request a specific version, or use a range specification: @openfn/language-http@1.2.3 {"@openfn/language-http", "~> 1.2.0"} {"@openfn/language-http", "< 2.0.0"} NOTE More complex npm style install strings like: ">=0.1.0 <0.2.0" are not supported. Generally the tuple style is preferred when using range specifications as the npm style strings have a simplistic regex splitter. See Version for more details on matching versions.","ref":"Lightning.AdaptorService.html"},{"type":"function","title":"Lightning.AdaptorService.build_aliased_name/1","doc":"Turns a package name and version into a string for NPM. Since multiple versions of the same package can be installed, it's important to rely on npms built-in package aliasing. E.g. @openfn/language-http@1.2.8 turns into: ` @openfn / language - http - 1.2 . 8 @npm : @openfn / language - http @ 1.2 . 8 ` Which is pretty long winded but necessary for the reason above. If using this module as a base, it's likely you would need to adaptor this to suit your particular naming strategy.","ref":"Lightning.AdaptorService.html#build_aliased_name/1"},{"type":"function","title":"Lightning.AdaptorService.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorService.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorService.find_adaptor/2","doc":"","ref":"Lightning.AdaptorService.html#find_adaptor/2"},{"type":"function","title":"Lightning.AdaptorService.get_adaptors/1","doc":"","ref":"Lightning.AdaptorService.html#get_adaptors/1"},{"type":"function","title":"Lightning.AdaptorService.install/2","doc":"","ref":"Lightning.AdaptorService.html#install/2"},{"type":"function","title":"Lightning.AdaptorService.install!/2","doc":"","ref":"Lightning.AdaptorService.html#install!/2"},{"type":"function","title":"Lightning.AdaptorService.installed?/2","doc":"","ref":"Lightning.AdaptorService.html#installed?/2"},{"type":"function","title":"Lightning.AdaptorService.resolve_package_name/1","doc":"","ref":"Lightning.AdaptorService.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorService.start_link/1","doc":"","ref":"Lightning.AdaptorService.html#start_link/1"},{"type":"type","title":"Lightning.AdaptorService.package_spec/0","doc":"","ref":"Lightning.AdaptorService.html#t:package_spec/0"},{"type":"module","title":"Lightning.Attempt","doc":"Ecto model for Attempts.","ref":"Lightning.Attempt.html"},{"type":"function","title":"Lightning.Attempt.new/1","doc":"","ref":"Lightning.Attempt.html#new/1"},{"type":"type","title":"Lightning.Attempt.t/0","doc":"","ref":"Lightning.Attempt.html#t:t/0"},{"type":"module","title":"Lightning.AttemptRun","doc":"Ecto model for an Attempts Runs.","ref":"Lightning.AttemptRun.html"},{"type":"function","title":"Lightning.AttemptRun.new/1","doc":"","ref":"Lightning.AttemptRun.html#new/1"},{"type":"function","title":"Lightning.AttemptRun.new/2","doc":"","ref":"Lightning.AttemptRun.html#new/2"},{"type":"type","title":"Lightning.AttemptRun.t/0","doc":"","ref":"Lightning.AttemptRun.html#t:t/0"},{"type":"module","title":"Lightning.AttemptService","doc":"The Attempts context.","ref":"Lightning.AttemptService.html"},{"type":"function","title":"Lightning.AttemptService.append/2","doc":"Adds an Attempt to an unsaved Run When given an Attempt, it simply adds the Run to a new AttemptRun. However when given an AttemptRun, the Run (from the AttemptRun) is set as the previous Run for the new unsaved Run.","ref":"Lightning.AttemptService.html#append/2"},{"type":"function","title":"Lightning.AttemptService.build_attempt/2","doc":"","ref":"Lightning.AttemptService.html#build_attempt/2"},{"type":"function","title":"Lightning.AttemptService.calculate_runs/3","doc":"","ref":"Lightning.AttemptService.html#calculate_runs/3"},{"type":"function","title":"Lightning.AttemptService.create_attempt/3","doc":"Create an attempt Examples iex> create_attempt ( %{ field : value } ) { :ok , % Attempt { } } iex> create_attempt ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.AttemptService.html#create_attempt/3"},{"type":"function","title":"Lightning.AttemptService.get_for_rerun/2","doc":"","ref":"Lightning.AttemptService.html#get_for_rerun/2"},{"type":"function","title":"Lightning.AttemptService.get_last_attempt_for/1","doc":"Get the latest attempt associated to a given run","ref":"Lightning.AttemptService.html#get_last_attempt_for/1"},{"type":"function","title":"Lightning.AttemptService.get_workflow_for/1","doc":"","ref":"Lightning.AttemptService.html#get_workflow_for/1"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_job/2","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of workorder ids that are associated to the given Job","ref":"Lightning.AttemptService.html#list_for_rerun_from_job/2"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_start/1","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of work order ids.","ref":"Lightning.AttemptService.html#list_for_rerun_from_start/1"},{"type":"function","title":"Lightning.AttemptService.retry/3","doc":"Creates a new Attempt starting from a given run. All upstream/prior Runs that were performed on that attempt are associated with the new Attempt, where as the specified run is used to create a new one and is added to the Attempt. Any runs downstream from the Run given are ignored.","ref":"Lightning.AttemptService.html#retry/3"},{"type":"function","title":"Lightning.AttemptService.retry_many/2","doc":"Creates new Attempts for each pair of corresponding AttemptRun and InvocationReason.","ref":"Lightning.AttemptService.html#retry_many/2"},{"type":"module","title":"Lightning.Auditing","doc":"Context for working with Audit records.","ref":"Lightning.Auditing.html"},{"type":"function","title":"Lightning.Auditing.list_all/1","doc":"","ref":"Lightning.Auditing.html#list_all/1"},{"type":"module","title":"Lightning.Auditing.Model","doc":"Macro module to add common model behaviour to a given Ecto model","ref":"Lightning.Auditing.Model.html"},{"type":"function","title":"Lightning.Auditing.Model.event/5","doc":"Creates a schema changeset for the event identified by row_id and caused by actor_id . The given metadata can be either nil , Ecto.Changeset , struct or map. It returns :no_changes in case of an Ecto.Changeset metadata that changed nothing or an Ecto.Changeset with the event ready to be inserted.","ref":"Lightning.Auditing.Model.html#event/5"},{"type":"function","title":"Lightning.Auditing.Model.save/2","doc":"Saves the event to the Repo . In case of nothing changes, do nothing. It returns {:ok, :no_changes} if nothing changed, {:ok, struct} if the log has been successfully saved or {:error, changeset} in case of error.","ref":"Lightning.Auditing.Model.html#save/2"},{"type":"module","title":"Lightning.AuthProviders","doc":"Context module for dealing with external Auth Providers.","ref":"Lightning.AuthProviders.html"},{"type":"function","title":"Lightning.AuthProviders.build_handler/2","doc":"","ref":"Lightning.AuthProviders.html#build_handler/2"},{"type":"function","title":"Lightning.AuthProviders.create/1","doc":"","ref":"Lightning.AuthProviders.html#create/1"},{"type":"function","title":"Lightning.AuthProviders.create_handler/1","doc":"","ref":"Lightning.AuthProviders.html#create_handler/1"},{"type":"function","title":"Lightning.AuthProviders.delete!/1","doc":"","ref":"Lightning.AuthProviders.html#delete!/1"},{"type":"function","title":"Lightning.AuthProviders.get_authorize_url/1","doc":"Retrieve the authorization url for a given handler or handler name.","ref":"Lightning.AuthProviders.html#get_authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.get_existing/0","doc":"","ref":"Lightning.AuthProviders.html#get_existing/0"},{"type":"function","title":"Lightning.AuthProviders.get_existing/1","doc":"","ref":"Lightning.AuthProviders.html#get_existing/1"},{"type":"function","title":"Lightning.AuthProviders.get_handler/1","doc":"","ref":"Lightning.AuthProviders.html#get_handler/1"},{"type":"function","title":"Lightning.AuthProviders.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.new/0","doc":"","ref":"Lightning.AuthProviders.html#new/0"},{"type":"function","title":"Lightning.AuthProviders.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.html#remove_handler/1"},{"type":"function","title":"Lightning.AuthProviders.update/2","doc":"","ref":"Lightning.AuthProviders.html#update/2"},{"type":"module","title":"Lightning.AuthProviders.AuthConfig","doc":"AuthProvider model","ref":"Lightning.AuthProviders.AuthConfig.html"},{"type":"type","title":"Lightning.AuthProviders.AuthConfig.t/0","doc":"","ref":"Lightning.AuthProviders.AuthConfig.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.CacheWarmer","doc":"Dummy warmer which caches database rows every 30s.","ref":"Lightning.AuthProviders.CacheWarmer.html"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AuthProviders.CacheWarmer.html#child_spec/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.execute/1","doc":"Executes this cache warmer with a connection.","ref":"Lightning.AuthProviders.CacheWarmer.html#execute/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.interval/0","doc":"Returns the interval for this warmer.","ref":"Lightning.AuthProviders.CacheWarmer.html#interval/0"},{"type":"module","title":"Lightning.AuthProviders.Google","doc":"","ref":"Lightning.AuthProviders.Google.html"},{"type":"function","title":"Lightning.AuthProviders.Google.authorize_url/2","doc":"","ref":"Lightning.AuthProviders.Google.html#authorize_url/2"},{"type":"function","title":"Lightning.AuthProviders.Google.build_client/1","doc":"Builds a new client","ref":"Lightning.AuthProviders.Google.html#build_client/1"},{"type":"function","title":"Lightning.AuthProviders.Google.get_config/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_config/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown!/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown!/0"},{"type":"function","title":"Lightning.AuthProviders.Google.refresh_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#refresh_token/2"},{"type":"module","title":"Lightning.AuthProviders.Handler","doc":"Module which wraps Oauth configuration and a WellKnown document into a convenient struct that can be used to authenticate users against any OIDC compliant provider.","ref":"Lightning.AuthProviders.Handler.html"},{"type":"function","title":"Lightning.AuthProviders.Handler.authorize_url/1","doc":"","ref":"Lightning.AuthProviders.Handler.html#authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.from_model/1","doc":"Returns a Handler from a given AuthConfig","ref":"Lightning.AuthProviders.Handler.html#from_model/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_token/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.new/2","doc":"Create a new Provider struct, expects a name and opts: :client_id - The providers issued id :client_secret - Secret for the client :redirect_uri - The URI for redirecting after authentication, usually the callback url in the router. :wellknown - A AuthProviders.WellKnown struct with the providers .well-known/openid-configuration .","ref":"Lightning.AuthProviders.Handler.html#new/2"},{"type":"type","title":"Lightning.AuthProviders.Handler.opts/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:opts/0"},{"type":"type","title":"Lightning.AuthProviders.Handler.t/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.Store","doc":"Store module for caching Handlers. Since Handlers often have to fetch their .well-known files when being initialized we cache these in order to avoid repeatedly making HTTP requests to a providers API.","ref":"Lightning.AuthProviders.Store.html"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.Store.put_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#put_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.Store.html#remove_handler/1"},{"type":"type","title":"Lightning.AuthProviders.Store.finder/0","doc":"","ref":"Lightning.AuthProviders.Store.html#t:finder/0"},{"type":"module","title":"Lightning.AuthProviders.WellKnown","doc":"A datastructure to fetch and hold information about a given OIDC/OAuth provider","ref":"Lightning.AuthProviders.WellKnown.html"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch!/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch!/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.new/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#new/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_options/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_params/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AuthProviders.WellKnown.html#request/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AuthProviders.WellKnown.html#request/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AuthProviders.WellKnown.html#request!/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AuthProviders.WellKnown.html#start/0"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#stream_next/1"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.body/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:body/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.headers/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:headers/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.method/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:method/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.options/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:options/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.params/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:params/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.request/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:request/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.t/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:t/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.url/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:url/0"},{"type":"module","title":"Lightning.CLI","doc":"Module providing facilities to make calls to the OpenFn CLI. See @openfn/cli","ref":"Lightning.CLI.html"},{"type":"function","title":"Lightning.CLI.execute/1","doc":"Execute a command in a child process and parse the results.","ref":"Lightning.CLI.html#execute/1"},{"type":"function","title":"Lightning.CLI.metadata/2","doc":"Retrieve metadata for a given adaptor and configuration.","ref":"Lightning.CLI.html#metadata/2"},{"type":"module","title":"Lightning.CLI.Result","doc":"Struct that wraps the output of an OpenFn CLI call. Containing the keys: start_time end_time status logs Logs The OpenFn CLI returns JSON formatted log lines, which are decoded and added to a Result struct. There are two kinds of output: { "level" :"<<level>>" , "name" :"<<module>>" , "message" :"..." ] , "time" :< < timestamp >> } These are usually for general logging, and debugging. { "message" : [ "<<message|filepath|output>>" ] } The above is the equivalent of the output of a command","ref":"Lightning.CLI.Result.html"},{"type":"function","title":"Lightning.CLI.Result.get_messages/1","doc":"Returns message type log lines from a Result .","ref":"Lightning.CLI.Result.html#get_messages/1"},{"type":"function","title":"Lightning.CLI.Result.new/1","doc":"","ref":"Lightning.CLI.Result.html#new/1"},{"type":"function","title":"Lightning.CLI.Result.parse/2","doc":"","ref":"Lightning.CLI.Result.html#parse/2"},{"type":"type","title":"Lightning.CLI.Result.t/0","doc":"","ref":"Lightning.CLI.Result.html#t:t/0"},{"type":"module","title":"Lightning.Demo","doc":"Demo encapsulates logic for setting up a demonstration site.","ref":"Lightning.Demo.html"},{"type":"function","title":"Lightning.Demo.reset_demo/0","doc":"Deletes everything in the database including the superuser and creates a set of publicly available users for a demo site via a command that can be run on Kubernetes-deployed systems.","ref":"Lightning.Demo.html#reset_demo/0"},{"type":"module","title":"Lightning.ExportUtils","doc":"Module that expose a function generating a complete and valid yaml string from a project and its workflows.","ref":"Lightning.ExportUtils.html"},{"type":"function","title":"Lightning.ExportUtils.build_yaml_tree/2","doc":"","ref":"Lightning.ExportUtils.html#build_yaml_tree/2"},{"type":"function","title":"Lightning.ExportUtils.generate_new_yaml/1","doc":"","ref":"Lightning.ExportUtils.html#generate_new_yaml/1"},{"type":"module","title":"Lightning.FailureNotifierView","doc":"","ref":"Lightning.FailureNotifierView.html"},{"type":"function","title":"Lightning.FailureNotifierView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"Lightning.FailureNotifierView.html#__resource__/0"},{"type":"function","title":"Lightning.FailureNotifierView.failure_alert.html/1","doc":"","ref":"Lightning.FailureNotifierView.html#failure_alert.html/1"},{"type":"function","title":"Lightning.FailureNotifierView.render/2","doc":"Renders the given template locally.","ref":"Lightning.FailureNotifierView.html#render/2"},{"type":"function","title":"Lightning.FailureNotifierView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"Lightning.FailureNotifierView.html#template_not_found/2"},{"type":"module","title":"Lightning.Helpers","doc":"Common functions for the context","ref":"Lightning.Helpers.html"},{"type":"function","title":"Lightning.Helpers.coerce_json_field/2","doc":"Changes a given maps field from a json string to a map. If it cannot be converted, it leaves the original value","ref":"Lightning.Helpers.html#coerce_json_field/2"},{"type":"function","title":"Lightning.Helpers.indefinite_article/1","doc":"","ref":"Lightning.Helpers.html#indefinite_article/1"},{"type":"function","title":"Lightning.Helpers.json_safe/1","doc":"Recursively ensures a given map is safe to convert to JSON, where all keys are strings and all values are json safe (primitive values).","ref":"Lightning.Helpers.html#json_safe/1"},{"type":"function","title":"Lightning.Helpers.ms_to_human/1","doc":"Converts milliseconds (integer) to a human duration, such as "1 minute" or "45 years, 6 months, 5 days, 21 hours, 12 minutes, 34 seconds" using Timex.Format.Duration.Formatters.Humanized.format() .","ref":"Lightning.Helpers.html#ms_to_human/1"},{"type":"module","title":"Lightning.MetadataService","doc":"Retrieves metadata for a given credential and adaptor using the OpenFn CLI.","ref":"Lightning.MetadataService.html"},{"type":"function","title":"Lightning.MetadataService.fetch/2","doc":"Retrieve metadata for a given adaptor and credential. The adaptor must be an npm specification.","ref":"Lightning.MetadataService.html#fetch/2"},{"type":"module","title":"Lightning.Name","doc":"Generates a random names.","ref":"Lightning.Name.html"},{"type":"function","title":"Lightning.Name.generate/1","doc":"","ref":"Lightning.Name.html#generate/1"},{"type":"module","title":"Lightning.ObanManager","doc":"The Oban Manager","ref":"Lightning.ObanManager.html"},{"type":"function","title":"Lightning.ObanManager.handle_event/4","doc":"","ref":"Lightning.ObanManager.html#handle_event/4"},{"type":"module","title":"Lightning.Policies.Permissions","doc":"This module defines a unique interface managing authorizations in Lightning. Users in Lightning have instance-wide and project-wide roles which determine their level of access to resources in the application. Fo rmore details see the documentation . These authorizations policies are all implemented under the lib/lightning/policies folder. In that folder you can find 3 files: The users.ex file has all the policies for the instances wide access levels The project_users.ex file has all the policies for the project wide access levels The permissions.ex file defines the Lightning.Policies.Permissions.can/4 interface. Which is a wrapper around the Bodyguard.permit/4 function. We use that interface to be able to harmonize the use of policies accross the entire app. All the policies are tested in the test/lightning/policies folder. And the test are written in a way that allows the reader to quickly who can do what in the app. We have two variants of the Lightning.Policies.Permissions.can/4 interface: Lightning.Policies.Permissions.can(policy, action, actor, resource) returns :ok if the actor can perform the action on the resource and {:error, :unauthorized} otherwise. Lightning.Policies.Permissions.can?(policy, action, actor, resource) returns true if the actor can perform the action on the resource and false otherwise. Here is an example of how we the Lightning.Policies.Permissions.can/4 interface to check if the a user can edit a job or not can_edit_job = Lightning.Policies.ProjectUsers |> Lightning.Policies.Permissions . can? ( :edit_job , socket . assigns . current_user , socket . assigns . project ) if can_edit_job do # allow user to edit the job else # quick user out end","ref":"Lightning.Policies.Permissions.html"},{"type":"function","title":"Lightning.Policies.Permissions.can/4","doc":"checks if user has the permissions to apply action using some policy module Returns :ok if user can apply action and {:error, :unauthorized} otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) :ok iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) { :error , :unauthorized }","ref":"Lightning.Policies.Permissions.html#can/4"},{"type":"function","title":"Lightning.Policies.Permissions.can?/4","doc":"same as can/4 but returns true if user can apply action and false otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) true iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) false","ref":"Lightning.Policies.Permissions.html#can?/4"},{"type":"module","title":"Lightning.Policies.ProjectUsers","doc":"The Bodyguard Policy module for projects members roles.","ref":"Lightning.Policies.ProjectUsers.html"},{"type":"function","title":"Lightning.Policies.ProjectUsers.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action in that project and false if they cannot. Note that permissions are grouped by action, rather than by user role. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.ProjectUsers.html#authorize/3"},{"type":"type","title":"Lightning.Policies.ProjectUsers.actions/0","doc":"","ref":"Lightning.Policies.ProjectUsers.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Provisioning","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Provisioning.html"},{"type":"function","title":"Lightning.Policies.Provisioning.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied. Only a superuser can provision a new project. Owners and admins can update existing projects.","ref":"Lightning.Policies.Provisioning.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Provisioning.actions/0","doc":"","ref":"Lightning.Policies.Provisioning.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Users","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Users.html"},{"type":"function","title":"Lightning.Policies.Users.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.Users.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Users.actions/0","doc":"","ref":"Lightning.Policies.Users.html#t:actions/0"},{"type":"module","title":"Lightning.Release","doc":"Used for executing DB release tasks when run in production without Mix installed.","ref":"Lightning.Release.html"},{"type":"function","title":"Lightning.Release.create_db/0","doc":"","ref":"Lightning.Release.html#create_db/0"},{"type":"function","title":"Lightning.Release.load_app/0","doc":"","ref":"Lightning.Release.html#load_app/0"},{"type":"function","title":"Lightning.Release.migrate/0","doc":"","ref":"Lightning.Release.html#migrate/0"},{"type":"function","title":"Lightning.Release.rollback/2","doc":"","ref":"Lightning.Release.html#rollback/2"},{"type":"module","title":"Lightning.Repo","doc":"","ref":"Lightning.Repo.html"},{"type":"function","title":"Lightning.Repo.aggregate/3","doc":"Callback implementation for Ecto.Repo.aggregate/3 .","ref":"Lightning.Repo.html#aggregate/3"},{"type":"function","title":"Lightning.Repo.aggregate/4","doc":"Callback implementation for Ecto.Repo.aggregate/4 .","ref":"Lightning.Repo.html#aggregate/4"},{"type":"function","title":"Lightning.Repo.all/2","doc":"Callback implementation for Ecto.Repo.all/2 .","ref":"Lightning.Repo.html#all/2"},{"type":"function","title":"Lightning.Repo.checked_out?/0","doc":"Callback implementation for Ecto.Repo.checked_out?/0 .","ref":"Lightning.Repo.html#checked_out?/0"},{"type":"function","title":"Lightning.Repo.checkout/2","doc":"Callback implementation for Ecto.Repo.checkout/2 .","ref":"Lightning.Repo.html#checkout/2"},{"type":"function","title":"Lightning.Repo.child_spec/1","doc":"","ref":"Lightning.Repo.html#child_spec/1"},{"type":"function","title":"Lightning.Repo.config/0","doc":"Callback implementation for Ecto.Repo.config/0 .","ref":"Lightning.Repo.html#config/0"},{"type":"function","title":"Lightning.Repo.default_options/1","doc":"Callback implementation for Ecto.Repo.default_options/1 .","ref":"Lightning.Repo.html#default_options/1"},{"type":"function","title":"Lightning.Repo.delete/2","doc":"Callback implementation for Ecto.Repo.delete/2 .","ref":"Lightning.Repo.html#delete/2"},{"type":"function","title":"Lightning.Repo.delete!/2","doc":"Callback implementation for Ecto.Repo.delete!/2 .","ref":"Lightning.Repo.html#delete!/2"},{"type":"function","title":"Lightning.Repo.delete_all/2","doc":"Callback implementation for Ecto.Repo.delete_all/2 .","ref":"Lightning.Repo.html#delete_all/2"},{"type":"function","title":"Lightning.Repo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the pool to disconnect within the given interval. See Ecto.Adapters.SQL.disconnect_all/3 for more information.","ref":"Lightning.Repo.html#disconnect_all/2"},{"type":"function","title":"Lightning.Repo.exists?/2","doc":"Callback implementation for Ecto.Repo.exists?/2 .","ref":"Lightning.Repo.html#exists?/2"},{"type":"function","title":"Lightning.Repo.explain/3","doc":"A convenience function for SQL-based repositories that executes an EXPLAIN statement or similar depending on the adapter to obtain statistics for the given query. See Ecto.Adapters.SQL.explain/4 for more information.","ref":"Lightning.Repo.html#explain/3"},{"type":"function","title":"Lightning.Repo.get/3","doc":"Callback implementation for Ecto.Repo.get/3 .","ref":"Lightning.Repo.html#get/3"},{"type":"function","title":"Lightning.Repo.get!/3","doc":"Callback implementation for Ecto.Repo.get!/3 .","ref":"Lightning.Repo.html#get!/3"},{"type":"function","title":"Lightning.Repo.get_by/3","doc":"Callback implementation for Ecto.Repo.get_by/3 .","ref":"Lightning.Repo.html#get_by/3"},{"type":"function","title":"Lightning.Repo.get_by!/3","doc":"Callback implementation for Ecto.Repo.get_by!/3 .","ref":"Lightning.Repo.html#get_by!/3"},{"type":"function","title":"Lightning.Repo.get_dynamic_repo/0","doc":"Callback implementation for Ecto.Repo.get_dynamic_repo/0 .","ref":"Lightning.Repo.html#get_dynamic_repo/0"},{"type":"function","title":"Lightning.Repo.in_transaction?/0","doc":"Callback implementation for Ecto.Repo.in_transaction?/0 .","ref":"Lightning.Repo.html#in_transaction?/0"},{"type":"function","title":"Lightning.Repo.insert/2","doc":"Callback implementation for Ecto.Repo.insert/2 .","ref":"Lightning.Repo.html#insert/2"},{"type":"function","title":"Lightning.Repo.insert!/2","doc":"Callback implementation for Ecto.Repo.insert!/2 .","ref":"Lightning.Repo.html#insert!/2"},{"type":"function","title":"Lightning.Repo.insert_all/3","doc":"Callback implementation for Ecto.Repo.insert_all/3 .","ref":"Lightning.Repo.html#insert_all/3"},{"type":"function","title":"Lightning.Repo.insert_or_update/2","doc":"Callback implementation for Ecto.Repo.insert_or_update/2 .","ref":"Lightning.Repo.html#insert_or_update/2"},{"type":"function","title":"Lightning.Repo.insert_or_update!/2","doc":"Callback implementation for Ecto.Repo.insert_or_update!/2 .","ref":"Lightning.Repo.html#insert_or_update!/2"},{"type":"function","title":"Lightning.Repo.load/2","doc":"Callback implementation for Ecto.Repo.load/2 .","ref":"Lightning.Repo.html#load/2"},{"type":"function","title":"Lightning.Repo.one/2","doc":"Callback implementation for Ecto.Repo.one/2 .","ref":"Lightning.Repo.html#one/2"},{"type":"function","title":"Lightning.Repo.one!/2","doc":"Callback implementation for Ecto.Repo.one!/2 .","ref":"Lightning.Repo.html#one!/2"},{"type":"function","title":"Lightning.Repo.paginate/2","doc":"","ref":"Lightning.Repo.html#paginate/2"},{"type":"function","title":"Lightning.Repo.preload/3","doc":"Callback implementation for Ecto.Repo.preload/3 .","ref":"Lightning.Repo.html#preload/3"},{"type":"function","title":"Lightning.Repo.prepare_query/3","doc":"Callback implementation for Ecto.Repo.prepare_query/3 .","ref":"Lightning.Repo.html#prepare_query/3"},{"type":"function","title":"Lightning.Repo.put_dynamic_repo/1","doc":"Callback implementation for Ecto.Repo.put_dynamic_repo/1 .","ref":"Lightning.Repo.html#put_dynamic_repo/1"},{"type":"function","title":"Lightning.Repo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query/4 for more information.","ref":"Lightning.Repo.html#query/3"},{"type":"function","title":"Lightning.Repo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query!/4 for more information.","ref":"Lightning.Repo.html#query!/3"},{"type":"function","title":"Lightning.Repo.query_many/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many/4 for more information.","ref":"Lightning.Repo.html#query_many/3"},{"type":"function","title":"Lightning.Repo.query_many!/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many!/4 for more information.","ref":"Lightning.Repo.html#query_many!/3"},{"type":"function","title":"Lightning.Repo.reload/2","doc":"Callback implementation for Ecto.Repo.reload/2 .","ref":"Lightning.Repo.html#reload/2"},{"type":"function","title":"Lightning.Repo.reload!/2","doc":"Callback implementation for Ecto.Repo.reload!/2 .","ref":"Lightning.Repo.html#reload!/2"},{"type":"function","title":"Lightning.Repo.rollback/1","doc":"Callback implementation for Ecto.Repo.rollback/1 .","ref":"Lightning.Repo.html#rollback/1"},{"type":"function","title":"Lightning.Repo.scrivener_defaults/0","doc":"","ref":"Lightning.Repo.html#scrivener_defaults/0"},{"type":"function","title":"Lightning.Repo.start_link/1","doc":"Callback implementation for Ecto.Repo.start_link/1 .","ref":"Lightning.Repo.html#start_link/1"},{"type":"function","title":"Lightning.Repo.stop/1","doc":"Callback implementation for Ecto.Repo.stop/1 .","ref":"Lightning.Repo.html#stop/1"},{"type":"function","title":"Lightning.Repo.stream/2","doc":"Callback implementation for Ecto.Repo.stream/2 .","ref":"Lightning.Repo.html#stream/2"},{"type":"function","title":"Lightning.Repo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL. See Ecto.Adapters.SQL.to_sql/3 for more information.","ref":"Lightning.Repo.html#to_sql/2"},{"type":"function","title":"Lightning.Repo.transaction/2","doc":"Callback implementation for Ecto.Repo.transaction/2 .","ref":"Lightning.Repo.html#transaction/2"},{"type":"function","title":"Lightning.Repo.update/2","doc":"Callback implementation for Ecto.Repo.update/2 .","ref":"Lightning.Repo.html#update/2"},{"type":"function","title":"Lightning.Repo.update!/2","doc":"Callback implementation for Ecto.Repo.update!/2 .","ref":"Lightning.Repo.html#update!/2"},{"type":"function","title":"Lightning.Repo.update_all/3","doc":"Callback implementation for Ecto.Repo.update_all/3 .","ref":"Lightning.Repo.html#update_all/3"},{"type":"module","title":"Lightning.RunSearchForm","doc":"Run filtering search form.","ref":"Lightning.RunSearchForm.html"},{"type":"module","title":"Lightning.SafetyString","doc":"Utilities for securely encoding serializable structs, lists and strings into URL-safe strings. In order to pass the state around in a URL, in a manner that protects secrets from leaking - and allows us to avoid persistance we take a set of parameters and: Encode into a URI query string gzip it to save characters encrypt the string base64 encode it for URI encoding safety","ref":"Lightning.SafetyString.html"},{"type":"function","title":"Lightning.SafetyString.decode/1","doc":"","ref":"Lightning.SafetyString.html#decode/1"},{"type":"function","title":"Lightning.SafetyString.encode/1","doc":"","ref":"Lightning.SafetyString.html#encode/1"},{"type":"module","title":"Lightning.Scrubber","doc":"Process used to scrub strings of sensitive information. Can be started via start_link/1 . { :ok , scrubber } = Lightning.Scrubber . start_link ( samples : Lightning.Credentials . sensitive_values_for ( credential ) ) Takes an optional :name key, in case you need to name the process.","ref":"Lightning.Scrubber.html"},{"type":"function","title":"Lightning.Scrubber.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Scrubber.html#child_spec/1"},{"type":"function","title":"Lightning.Scrubber.encode_samples/1","doc":"Prepare a list of sensitive samples (strings) into a potentially bigger list composed of variations a sample may appear.","ref":"Lightning.Scrubber.html#encode_samples/1"},{"type":"function","title":"Lightning.Scrubber.samples/1","doc":"","ref":"Lightning.Scrubber.html#samples/1"},{"type":"function","title":"Lightning.Scrubber.scrub/2","doc":"","ref":"Lightning.Scrubber.html#scrub/2"},{"type":"function","title":"Lightning.Scrubber.start_link/1","doc":"","ref":"Lightning.Scrubber.html#start_link/1"},{"type":"module","title":"Lightning.SetupUtils","doc":"SetupUtils encapsulates logic for setting up initial data for various sites.","ref":"Lightning.SetupUtils.html"},{"type":"function","title":"Lightning.SetupUtils.add_and_update_runs/3","doc":"","ref":"Lightning.SetupUtils.html#add_and_update_runs/3"},{"type":"function","title":"Lightning.SetupUtils.create_dhis2_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_dhis2_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_openhie_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_openhie_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_starter_project/2","doc":"","ref":"Lightning.SetupUtils.html#create_starter_project/2"},{"type":"function","title":"Lightning.SetupUtils.setup_demo/1","doc":"Creates initial data and returns the created records.","ref":"Lightning.SetupUtils.html#setup_demo/1"},{"type":"function","title":"Lightning.SetupUtils.tear_down/1","doc":"","ref":"Lightning.SetupUtils.html#tear_down/1"},{"type":"module","title":"Lightning.TaskWorker","doc":"A TaskWorker with concurrency limits. A simple concurrency limiter that wraps Task.Supervisor , which already does have the ability to specify max_children ; it throws an error when that limit is exceeded. To use it, start it like any other process; ideally in your supervision tree. ... , { Lightning.TaskWorker , name : :cli_task_worker , max_tasks : 4 } Options :max_tasks Defaults to the number of system schedulers available to the vm.","ref":"Lightning.TaskWorker.html"},{"type":"function","title":"Lightning.TaskWorker.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.TaskWorker.html#child_spec/1"},{"type":"function","title":"Lightning.TaskWorker.get_status/1","doc":"","ref":"Lightning.TaskWorker.html#get_status/1"},{"type":"function","title":"Lightning.TaskWorker.start_link/1","doc":"","ref":"Lightning.TaskWorker.html#start_link/1"},{"type":"function","title":"Lightning.TaskWorker.start_task/2","doc":"","ref":"Lightning.TaskWorker.html#start_task/2"},{"type":"module","title":"Lightning.Validators","doc":"Extra validators for Ecto.Changeset.","ref":"Lightning.Validators.html"},{"type":"function","title":"Lightning.Validators.validate_exclusive/3","doc":"Validate that only one of the fields is set at a time. Example: changeset |> validate_exclusive ( [ :source_job_id , :source_trigger_id ] , "source_job_id and source_trigger_id are mutually exclusive" )","ref":"Lightning.Validators.html#validate_exclusive/3"},{"type":"function","title":"Lightning.Validators.validate_one_required/3","doc":"Validate that at least one of the fields is set.","ref":"Lightning.Validators.html#validate_one_required/3"},{"type":"module","title":"Lightning.Vault","doc":"Module for handling the encryption and decryption of database fields.","ref":"Lightning.Vault.html"},{"type":"function","title":"Lightning.Vault.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Vault.html#child_spec/1"},{"type":"function","title":"Lightning.Vault.start_link/1","doc":"","ref":"Lightning.Vault.html#start_link/1"},{"type":"module","title":"Lightning.WorkOrder","doc":"Ecto model for Workorders.","ref":"Lightning.WorkOrder.html"},{"type":"function","title":"Lightning.WorkOrder.new/0","doc":"","ref":"Lightning.WorkOrder.html#new/0"},{"type":"type","title":"Lightning.WorkOrder.t/0","doc":"","ref":"Lightning.WorkOrder.html#t:t/0"},{"type":"module","title":"Lightning.WorkOrderService","doc":"The WorkOrderService.","ref":"Lightning.WorkOrderService.html"},{"type":"function","title":"Lightning.WorkOrderService.attempt_updated/1","doc":"","ref":"Lightning.WorkOrderService.html#attempt_updated/1"},{"type":"function","title":"Lightning.WorkOrderService.build/2","doc":"","ref":"Lightning.WorkOrderService.html#build/2"},{"type":"function","title":"Lightning.WorkOrderService.create_manual_workorder/3","doc":"","ref":"Lightning.WorkOrderService.html#create_manual_workorder/3"},{"type":"function","title":"Lightning.WorkOrderService.create_webhook_workorder/2","doc":"","ref":"Lightning.WorkOrderService.html#create_webhook_workorder/2"},{"type":"function","title":"Lightning.WorkOrderService.create_work_order/1","doc":"Creates a work_order. Examples iex> create_work_order ( %{ field : value } ) { :ok , % WorkOrder { } } iex> create_work_order ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.WorkOrderService.html#create_work_order/1"},{"type":"function","title":"Lightning.WorkOrderService.multi_for/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for/3"},{"type":"function","title":"Lightning.WorkOrderService.multi_for_manual/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for_manual/3"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_run/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_run/2"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_runs/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_runs/2"},{"type":"function","title":"Lightning.WorkOrderService.subscribe/1","doc":"","ref":"Lightning.WorkOrderService.html#subscribe/1"},{"type":"module","title":"Lightning.Workflows","doc":"The Workflows context.","ref":"Lightning.Workflows.html"},{"type":"function","title":"Lightning.Workflows.build_trigger/1","doc":"Builds a Trigger","ref":"Lightning.Workflows.html#build_trigger/1"},{"type":"function","title":"Lightning.Workflows.change_workflow/2","doc":"Returns an %Ecto.Changeset{} for tracking workflow changes. Examples iex> change_workflow ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#change_workflow/2"},{"type":"function","title":"Lightning.Workflows.create_edge/1","doc":"Creates an edge","ref":"Lightning.Workflows.html#create_edge/1"},{"type":"function","title":"Lightning.Workflows.create_workflow/1","doc":"Creates a workflow. Examples iex> create_workflow ( %{ field : value } ) { :ok , % Workflow { } } iex> create_workflow ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#create_workflow/1"},{"type":"function","title":"Lightning.Workflows.delete_workflow/1","doc":"Deletes a workflow. Examples iex> delete_workflow ( workflow ) { :ok , % Workflow { } } iex> delete_workflow ( workflow ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#delete_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_edge_by_webhook/1","doc":"Gets a Single Edge by it's webhook trigger.","ref":"Lightning.Workflows.html#get_edge_by_webhook/1"},{"type":"function","title":"Lightning.Workflows.get_edges_for_cron_execution/1","doc":"Returns a list of edges with jobs to execute, given a current timestamp in Unix. This is used by the scheduler, which calls this function once every minute.","ref":"Lightning.Workflows.html#get_edges_for_cron_execution/1"},{"type":"function","title":"Lightning.Workflows.get_workflow/1","doc":"","ref":"Lightning.Workflows.html#get_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_workflow!/1","doc":"Gets a single workflow. Raises Ecto.NoResultsError if the Workflow does not exist. Examples iex> get_workflow! ( 123 ) % Workflow { } iex> get_workflow! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Workflows.html#get_workflow!/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for/1","doc":"Retrieves a list of Workflows with their jobs and triggers preloaded.","ref":"Lightning.Workflows.html#get_workflows_for/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for_query/1","doc":"","ref":"Lightning.Workflows.html#get_workflows_for_query/1"},{"type":"function","title":"Lightning.Workflows.list_workflows/0","doc":"Returns the list of workflows. Examples iex> list_workflows ( ) [ % Workflow { } , ... ]","ref":"Lightning.Workflows.html#list_workflows/0"},{"type":"function","title":"Lightning.Workflows.mark_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the workflow request_deletion. Examples iex> change_request_deletion ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#mark_for_deletion/2"},{"type":"function","title":"Lightning.Workflows.to_project_space/1","doc":"","ref":"Lightning.Workflows.html#to_project_space/1"},{"type":"function","title":"Lightning.Workflows.update_trigger/2","doc":"Updates a trigger","ref":"Lightning.Workflows.html#update_trigger/2"},{"type":"function","title":"Lightning.Workflows.update_workflow/2","doc":"Updates a workflow. Examples iex> update_workflow ( workflow , %{ field : new_value } ) { :ok , % Workflow { } } iex> update_workflow ( workflow , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#update_workflow/2"},{"type":"module","title":"Lightning.Workflows.Edge","doc":"Ecto model for Workflow Edges. A Workflow Edge represents a connection between two jobs (or a trigger and a job) in a workflow. The source of the edge is either a job or a trigger. The target of the edge is always a job.","ref":"Lightning.Workflows.Edge.html"},{"type":"function","title":"Lightning.Workflows.Edge.changeset/2","doc":"","ref":"Lightning.Workflows.Edge.html#changeset/2"},{"type":"function","title":"Lightning.Workflows.Edge.new/1","doc":"","ref":"Lightning.Workflows.Edge.html#new/1"},{"type":"function","title":"Lightning.Workflows.Edge.validate/1","doc":"","ref":"Lightning.Workflows.Edge.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Edge.edge_condition/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:edge_condition/0"},{"type":"type","title":"Lightning.Workflows.Edge.t/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:t/0"},{"type":"module","title":"Lightning.Workflows.Graph","doc":"Utility to construct and manipulate a graph/plan made out of Jobs","ref":"Lightning.Workflows.Graph.html"},{"type":"function","title":"Lightning.Workflows.Graph.new/1","doc":"","ref":"Lightning.Workflows.Graph.html#new/1"},{"type":"function","title":"Lightning.Workflows.Graph.remove/2","doc":"","ref":"Lightning.Workflows.Graph.html#remove/2"},{"type":"function","title":"Lightning.Workflows.Graph.vertices/1","doc":"","ref":"Lightning.Workflows.Graph.html#vertices/1"},{"type":"type","title":"Lightning.Workflows.Graph.t/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:t/0"},{"type":"type","title":"Lightning.Workflows.Graph.vertex/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:vertex/0"},{"type":"module","title":"Lightning.Workflows.Workflow","doc":"Ecto model for Workflows. A Workflow contains the fields for defining a workflow. name A plain text identifier","ref":"Lightning.Workflows.Workflow.html"},{"type":"function","title":"Lightning.Workflows.Workflow.request_deletion_changeset/2","doc":"","ref":"Lightning.Workflows.Workflow.html#request_deletion_changeset/2"},{"type":"function","title":"Lightning.Workflows.Workflow.validate/1","doc":"","ref":"Lightning.Workflows.Workflow.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Workflow.t/0","doc":"","ref":"Lightning.Workflows.Workflow.html#t:t/0"},{"type":"module","title":"Lightning.Workorders.SearchParams","doc":"This module is used to parse search parameters for workorders and provide a query to the database.","ref":"Lightning.Workorders.SearchParams.html"},{"type":"function","title":"Lightning.Workorders.SearchParams.new/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#new/1"},{"type":"function","title":"Lightning.Workorders.SearchParams.to_uri_params/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#to_uri_params/1"},{"type":"type","title":"Lightning.Workorders.SearchParams.t/0","doc":"","ref":"Lightning.Workorders.SearchParams.html#t:t/0"},{"type":"module","title":"LightningWeb","doc":"The entrypoint for defining your web interface, such as controllers, views, channels and so on. This can be used in your application as: use LightningWeb , :controller use LightningWeb , :view The definitions below will be executed for every view, controller, etc, so keep them short and clean, focused on imports, uses and aliases. Do NOT define functions inside the quoted expressions below. Instead, define any helper function in modules and import those modules here.","ref":"LightningWeb.html"},{"type":"macro","title":"LightningWeb.__using__/1","doc":"When used, dispatch to the appropriate controller/view/etc.","ref":"LightningWeb.html#__using__/1"},{"type":"function","title":"LightningWeb.channel/0","doc":"","ref":"LightningWeb.html#channel/0"},{"type":"function","title":"LightningWeb.component/0","doc":"","ref":"LightningWeb.html#component/0"},{"type":"function","title":"LightningWeb.controller/0","doc":"","ref":"LightningWeb.html#controller/0"},{"type":"function","title":"LightningWeb.html/0","doc":"","ref":"LightningWeb.html#html/0"},{"type":"function","title":"LightningWeb.live_component/0","doc":"","ref":"LightningWeb.html#live_component/0"},{"type":"function","title":"LightningWeb.live_view/1","doc":"","ref":"LightningWeb.html#live_view/1"},{"type":"function","title":"LightningWeb.router/0","doc":"","ref":"LightningWeb.html#router/0"},{"type":"function","title":"LightningWeb.static_paths/0","doc":"","ref":"LightningWeb.html#static_paths/0"},{"type":"function","title":"LightningWeb.verified_routes/0","doc":"","ref":"LightningWeb.html#verified_routes/0"},{"type":"function","title":"LightningWeb.view/0","doc":"","ref":"LightningWeb.html#view/0"},{"type":"module","title":"LightningWeb.API.Helpers","doc":"Helpers for the API views","ref":"LightningWeb.API.Helpers.html"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_link/3","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_link/3"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_links/2","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_links/2"},{"type":"function","title":"LightningWeb.API.Helpers.url_for/2","doc":"","ref":"LightningWeb.API.Helpers.html#url_for/2"},{"type":"module","title":"LightningWeb.API.JobController","doc":"","ref":"LightningWeb.API.JobController.html"},{"type":"function","title":"LightningWeb.API.JobController.index/2","doc":"","ref":"LightningWeb.API.JobController.html#index/2"},{"type":"function","title":"LightningWeb.API.JobController.show/2","doc":"","ref":"LightningWeb.API.JobController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProjectController","doc":"","ref":"LightningWeb.API.ProjectController.html"},{"type":"function","title":"LightningWeb.API.ProjectController.index/2","doc":"","ref":"LightningWeb.API.ProjectController.html#index/2"},{"type":"function","title":"LightningWeb.API.ProjectController.show/2","doc":"","ref":"LightningWeb.API.ProjectController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProvisioningController","doc":"","ref":"LightningWeb.API.ProvisioningController.html"},{"type":"function","title":"LightningWeb.API.ProvisioningController.create/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#create/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#show/2"},{"type":"module","title":"LightningWeb.API.RunController","doc":"","ref":"LightningWeb.API.RunController.html"},{"type":"function","title":"LightningWeb.API.RunController.index/2","doc":"","ref":"LightningWeb.API.RunController.html#index/2"},{"type":"function","title":"LightningWeb.API.RunController.show/2","doc":"","ref":"LightningWeb.API.RunController.html#show/2"},{"type":"module","title":"LightningWeb.AuditLive.Index","doc":"LiveView for listing Audit events","ref":"LightningWeb.AuditLive.Index.html"},{"type":"function","title":"LightningWeb.AuditLive.Index.diff/1","doc":"","ref":"LightningWeb.AuditLive.Index.html#diff/1"},{"type":"function","title":"LightningWeb.AuditLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.AuditLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.AuthProvidersLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.AuthProvidersLive.FormComponent.html"},{"type":"module","title":"LightningWeb.AuthProvidersLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.AuthProvidersLive.Index.html"},{"type":"module","title":"LightningWeb.ChangesetJSON","doc":"Renders changesets as JSON.","ref":"LightningWeb.ChangesetJSON.html"},{"type":"function","title":"LightningWeb.ChangesetJSON.error/1","doc":"","ref":"LightningWeb.ChangesetJSON.html#error/1"},{"type":"module","title":"LightningWeb.ChangesetView","doc":"","ref":"LightningWeb.ChangesetView.html"},{"type":"function","title":"LightningWeb.ChangesetView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ChangesetView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ChangesetView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ChangesetView.html#render/2"},{"type":"function","title":"LightningWeb.ChangesetView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ChangesetView.html#template_not_found/2"},{"type":"function","title":"LightningWeb.ChangesetView.translate_errors/1","doc":"Traverses and translates changeset errors. See Ecto.Changeset.traverse_errors/2 and LightningWeb.ErrorHelpers.translate_error/1 for more details.","ref":"LightningWeb.ChangesetView.html#translate_errors/1"},{"type":"module","title":"LightningWeb.Components.Modal","doc":"A modal component that can be used to display a modal on the page. This currently isn't used anywhere but should be used in the future to replace the existing modal implementations.","ref":"LightningWeb.Components.Modal.html"},{"type":"module","title":"LightningWeb.CredentialLive.Edit","doc":"LiveView for editing a single Credential, which inturn uses LightningWeb.CredentialLive.FormComponent for common functionality.","ref":"LightningWeb.CredentialLive.Edit.html"},{"type":"function","title":"LightningWeb.CredentialLive.Edit.handle_info/2","doc":"A generic handler for forwarding updates from PubSub","ref":"LightningWeb.CredentialLive.Edit.html#handle_info/2"},{"type":"module","title":"LightningWeb.CredentialLive.FormComponent","doc":"Form Component for working with a single Credential","ref":"LightningWeb.CredentialLive.FormComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.credential_transfer/1","doc":"Attributes users ( :list ) (required) form ( :map ) (required)","ref":"LightningWeb.CredentialLive.FormComponent.html#credential_transfer/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.form_component/1","doc":"Switcher components for different types of credentials. Attributes type ( :string ) (required) form ( :map ) (required) update_body ( :any ) Slots inner_block","ref":"LightningWeb.CredentialLive.FormComponent.html#form_component/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.project_credentials/1","doc":"Attributes form ( :map ) (required) projects ( :list ) (required) selected ( :map ) (required) phx_target ( :any ) - Defaults to nil .","ref":"LightningWeb.CredentialLive.FormComponent.html#project_credentials/1"},{"type":"module","title":"LightningWeb.CredentialLive.GoogleSheetsComponent","doc":"Form component to setup a Google Sheets component. This component has several moving parts: Subscribes to a PubSub topic specially link to the component id See: LightningWeb.OauthCredentialHelper . Uses the Lightning.Google module to set up an OAuth client for generating urls, exchanging the code and requesting a new access_token . The flow for creating a new token is: Generate an authorization link which contains: The authorization url from the Google client with the applications callback_url A state string that is an encrypted set of data with the components module and id in it Once the user authorizes the client the callback is requested with a code The LightningWeb.OidcController decodes the state returned to it and does a 'broadcast_forward' which is simply a message expected to be received by a LiveView and applied to Phoenix.LiveView.send_update/3 . The component receives the code and requests a token. Any changes to the token (Credential body) are still handled by the parent component and so a update_body function is passed in to send params changes back up to update the form.","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.disabled_authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#disabled_authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.error_block/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#error_block/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.fieldset/1","doc":"Attributes form ( :map ) (required) update_body ( :any ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.userinfo/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#userinfo/1"},{"type":"module","title":"LightningWeb.CredentialLive.Index","doc":"LiveView for listing and managing credentials","ref":"LightningWeb.CredentialLive.Index.html"},{"type":"function","title":"LightningWeb.CredentialLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.CredentialLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.schema_input/1","doc":"Attributes form ( :map ) (required) schema ( :map ) (required) field ( :any ) (required)","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#schema_input/1"},{"type":"module","title":"LightningWeb.CredentialLive.RawBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.RawBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.RawBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.RawBodyComponent.html#fieldset/1"},{"type":"module","title":"LightningWeb.CredentialLive.TypePicker","doc":"","ref":"LightningWeb.CredentialLive.TypePicker.html"},{"type":"module","title":"LightningWeb.DataclipLive.Edit","doc":"LiveView for editing a single dataclip.","ref":"LightningWeb.DataclipLive.Edit.html"},{"type":"function","title":"LightningWeb.DataclipLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.FormComponent","doc":"Form Component for working with a single dataclip","ref":"LightningWeb.DataclipLive.FormComponent.html"},{"type":"function","title":"LightningWeb.DataclipLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.DataclipLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.Index","doc":"LiveView for listing and working with a list of Dataclips","ref":"LightningWeb.DataclipLive.Index.html"},{"type":"function","title":"LightningWeb.DataclipLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.DownloadsController","doc":"","ref":"LightningWeb.DownloadsController.html"},{"type":"function","title":"LightningWeb.DownloadsController.download_project_yaml/2","doc":"","ref":"LightningWeb.DownloadsController.html#download_project_yaml/2"},{"type":"module","title":"LightningWeb.Endpoint","doc":"","ref":"LightningWeb.Endpoint.html"},{"type":"function","title":"LightningWeb.Endpoint.broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast/3 .","ref":"LightningWeb.Endpoint.html#broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast!/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast!/3 .","ref":"LightningWeb.Endpoint.html#broadcast!/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from!/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from!/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from!/4"},{"type":"function","title":"LightningWeb.Endpoint.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.Endpoint.html#call/2"},{"type":"function","title":"LightningWeb.Endpoint.child_spec/1","doc":"Returns the child specification to start the endpoint under a supervision tree.","ref":"LightningWeb.Endpoint.html#child_spec/1"},{"type":"function","title":"LightningWeb.Endpoint.config/2","doc":"Returns the endpoint configuration for key Returns default if the key does not exist.","ref":"LightningWeb.Endpoint.html#config/2"},{"type":"function","title":"LightningWeb.Endpoint.config_change/2","doc":"Reloads the configuration given the application environment changes.","ref":"LightningWeb.Endpoint.html#config_change/2"},{"type":"function","title":"LightningWeb.Endpoint.host/0","doc":"Returns the host for the given endpoint.","ref":"LightningWeb.Endpoint.html#host/0"},{"type":"function","title":"LightningWeb.Endpoint.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.Endpoint.html#init/1"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast/3 .","ref":"LightningWeb.Endpoint.html#local_broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#local_broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.path/1","doc":"Generates the path information when routing to this endpoint.","ref":"LightningWeb.Endpoint.html#path/1"},{"type":"function","title":"LightningWeb.Endpoint.script_name/0","doc":"Generates the script name.","ref":"LightningWeb.Endpoint.html#script_name/0"},{"type":"function","title":"LightningWeb.Endpoint.start_link/1","doc":"Starts the endpoint supervision tree. All other options are merged into the endpoint configuration.","ref":"LightningWeb.Endpoint.html#start_link/1"},{"type":"function","title":"LightningWeb.Endpoint.static_integrity/1","doc":"Generates a base64-encoded cryptographic hash (sha512) to a static file in priv/static . Meant to be used for Subresource Integrity with CDNs.","ref":"LightningWeb.Endpoint.html#static_integrity/1"},{"type":"function","title":"LightningWeb.Endpoint.static_lookup/1","doc":"Returns a two item tuple with the first item being the static_path and the second item being the static_integrity .","ref":"LightningWeb.Endpoint.html#static_lookup/1"},{"type":"function","title":"LightningWeb.Endpoint.static_path/1","doc":"Generates a route to a static file in priv/static .","ref":"LightningWeb.Endpoint.html#static_path/1"},{"type":"function","title":"LightningWeb.Endpoint.static_url/0","doc":"Generates the static URL without any path information. It uses the configuration under :static_url to generate such. It falls back to :url if :static_url is not set.","ref":"LightningWeb.Endpoint.html#static_url/0"},{"type":"function","title":"LightningWeb.Endpoint.struct_url/0","doc":"Generates the endpoint base URL but as a URI struct. It uses the configuration under :url to generate such. Useful for manipulating the URL data and passing it to URL helpers.","ref":"LightningWeb.Endpoint.html#struct_url/0"},{"type":"function","title":"LightningWeb.Endpoint.subscribe/2","doc":"Callback implementation for Phoenix.Endpoint.subscribe/2 .","ref":"LightningWeb.Endpoint.html#subscribe/2"},{"type":"function","title":"LightningWeb.Endpoint.unsubscribe/1","doc":"Callback implementation for Phoenix.Endpoint.unsubscribe/1 .","ref":"LightningWeb.Endpoint.html#unsubscribe/1"},{"type":"function","title":"LightningWeb.Endpoint.url/0","doc":"Generates the endpoint base URL without any path information. It uses the configuration under :url to generate such.","ref":"LightningWeb.Endpoint.html#url/0"},{"type":"module","title":"LightningWeb.ErrorView","doc":"","ref":"LightningWeb.ErrorView.html"},{"type":"function","title":"LightningWeb.ErrorView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ErrorView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ErrorView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ErrorView.html#render/2"},{"type":"function","title":"LightningWeb.ErrorView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ErrorView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.FallbackController","doc":"Translates controller action results into valid Plug.Conn responses. See Phoenix.Controller.action_fallback/1 for more details.","ref":"LightningWeb.FallbackController.html"},{"type":"module","title":"LightningWeb.FirstSetupLive.Superuser","doc":"Superuser setup liveview Allows the creation on the first user in the system. It has only one action: :show","ref":"LightningWeb.FirstSetupLive.Superuser.html"},{"type":"function","title":"LightningWeb.FirstSetupLive.Superuser.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.FirstSetupLive.Superuser.html#render/1"},{"type":"module","title":"LightningWeb.FormHelpers","doc":"Conveniences for building forms.","ref":"LightningWeb.FormHelpers.html"},{"type":"module","title":"LightningWeb.Gettext","doc":"A module providing Internationalization with a gettext-based API. By using Gettext , your module gains a set of macros for translations, for example: import LightningWeb.Gettext # Simple translation gettext ( "Here is the string to translate" ) # Plural translation ngettext ( "Here is the string to translate" , "Here are the strings to translate" , 3 ) # Domain-based translation dgettext ( "errors" , "Here is the error message to translate" ) See the Gettext Docs for detailed usage.","ref":"LightningWeb.Gettext.html"},{"type":"macro","title":"LightningWeb.Gettext.dgettext/3","doc":"Callback implementation for Gettext.Backend.dgettext/3 .","ref":"LightningWeb.Gettext.html#dgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.dgettext_noop/2","doc":"Callback implementation for Gettext.Backend.dgettext_noop/2 .","ref":"LightningWeb.Gettext.html#dgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.dngettext/5","doc":"Callback implementation for Gettext.Backend.dngettext/5 .","ref":"LightningWeb.Gettext.html#dngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.dngettext_noop/3","doc":"Callback implementation for Gettext.Backend.dngettext_noop/3 .","ref":"LightningWeb.Gettext.html#dngettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext/4","doc":"Callback implementation for Gettext.Backend.dpgettext/4 .","ref":"LightningWeb.Gettext.html#dpgettext/4"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#dpgettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext/6","doc":"Callback implementation for Gettext.Backend.dpngettext/6 .","ref":"LightningWeb.Gettext.html#dpngettext/6"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext_noop/4","doc":"","ref":"LightningWeb.Gettext.html#dpngettext_noop/4"},{"type":"macro","title":"LightningWeb.Gettext.gettext/2","doc":"Callback implementation for Gettext.Backend.gettext/2 .","ref":"LightningWeb.Gettext.html#gettext/2"},{"type":"macro","title":"LightningWeb.Gettext.gettext_comment/1","doc":"Callback implementation for Gettext.Backend.gettext_comment/1 .","ref":"LightningWeb.Gettext.html#gettext_comment/1"},{"type":"macro","title":"LightningWeb.Gettext.gettext_noop/1","doc":"Callback implementation for Gettext.Backend.gettext_noop/1 .","ref":"LightningWeb.Gettext.html#gettext_noop/1"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_bindings/2","doc":"Callback implementation for Gettext.Backend.handle_missing_bindings/2 .","ref":"LightningWeb.Gettext.html#handle_missing_bindings/2"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_plural_translation/7","doc":"Callback implementation for Gettext.Backend.handle_missing_plural_translation/7 .","ref":"LightningWeb.Gettext.html#handle_missing_plural_translation/7"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_translation/5","doc":"Callback implementation for Gettext.Backend.handle_missing_translation/5 .","ref":"LightningWeb.Gettext.html#handle_missing_translation/5"},{"type":"function","title":"LightningWeb.Gettext.lgettext/5","doc":"","ref":"LightningWeb.Gettext.html#lgettext/5"},{"type":"function","title":"LightningWeb.Gettext.lngettext/7","doc":"","ref":"LightningWeb.Gettext.html#lngettext/7"},{"type":"macro","title":"LightningWeb.Gettext.ngettext/4","doc":"Callback implementation for Gettext.Backend.ngettext/4 .","ref":"LightningWeb.Gettext.html#ngettext/4"},{"type":"macro","title":"LightningWeb.Gettext.ngettext_noop/2","doc":"Callback implementation for Gettext.Backend.ngettext_noop/2 .","ref":"LightningWeb.Gettext.html#ngettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pgettext/3","doc":"Callback implementation for Gettext.Backend.pgettext/3 .","ref":"LightningWeb.Gettext.html#pgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.pgettext_noop/2","doc":"","ref":"LightningWeb.Gettext.html#pgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pngettext/5","doc":"Callback implementation for Gettext.Backend.pngettext/5 .","ref":"LightningWeb.Gettext.html#pngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.pngettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#pngettext_noop/3"},{"type":"module","title":"LightningWeb.HealthCheck","doc":"","ref":"LightningWeb.HealthCheck.html"},{"type":"function","title":"LightningWeb.HealthCheck.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.HealthCheck.html#call/2"},{"type":"function","title":"LightningWeb.HealthCheck.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.HealthCheck.html#init/1"},{"type":"module","title":"LightningWeb.Hooks","doc":"LiveView Hooks","ref":"LightningWeb.Hooks.html"},{"type":"function","title":"LightningWeb.Hooks.on_mount/4","doc":"Finds and assigns a project to the socket, if a user doesn't have access they are redirected and shown a 'No Access' screen via a :nav flash message. There is a fallthru function, when there is no project_id in the params - this is for liveviews that may or may not have a project_id depending on usage - like DashboardLive .","ref":"LightningWeb.Hooks.html#on_mount/4"},{"type":"module","title":"LightningWeb.InitAssigns","doc":"Ensures common assigns are applied to all LiveViews attaching this hook.","ref":"LightningWeb.InitAssigns.html"},{"type":"function","title":"LightningWeb.InitAssigns.on_mount/4","doc":"","ref":"LightningWeb.InitAssigns.html#on_mount/4"},{"type":"module","title":"LightningWeb.JobLive.AdaptorPicker","doc":"Component allowing selecting an adaptor and it's version","ref":"LightningWeb.JobLive.AdaptorPicker.html"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.display_name_for_adaptor/1","doc":"Converts standard adaptor names into "label","value" lists and returns non-standard names as merely "value"; both can be passed directly into a select option list.","ref":"LightningWeb.JobLive.AdaptorPicker.html#display_name_for_adaptor/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.get_adaptor_version_options/1","doc":"","ref":"LightningWeb.JobLive.AdaptorPicker.html#get_adaptor_version_options/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.render/1","doc":"Attributes form ( :map ) (required) on_change ( :any ) - Defaults to nil . disabled ( :boolean ) - Defaults to false .","ref":"LightningWeb.JobLive.AdaptorPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CredentialPicker","doc":"Component allowing selecting a credential or creating a new one via a modal.","ref":"LightningWeb.JobLive.CredentialPicker.html"},{"type":"function","title":"LightningWeb.JobLive.CredentialPicker.render/1","doc":"Attributes form ( :map ) (required) disabled ( :boolean ) - Defaults to false . credentials ( :list ) (required) on_change ( :any ) - Defaults to nil .","ref":"LightningWeb.JobLive.CredentialPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CronSetupComponent","doc":"A live component for managing cron setup in a form. The CronSetupComponent provides an interactive form for configuring cron settings. It includes fields for specifying the frequency, minute, hour, weekday, monthday, and cron expression. Usage Include the CronSetupComponent in your live view or template. Pass the necessary assigns to the component, such as form , on_change , and disabled . Handle the cron_expression_change event to capture changes in the form inputs. Use the updated cron_expression in your application logic.","ref":"LightningWeb.JobLive.CronSetupComponent.html"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.build_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#build_cron_expression/2"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.frequency_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#frequency_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.get_cron_data/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#get_cron_data/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.hour_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#hour_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.minute_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#minute_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.monthday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#monthday_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.render/1","doc":"Attributes id ( :string ) (required) form ( :map ) (required) on_change ( :any ) (required) disabled ( :boolean ) (required)","ref":"LightningWeb.JobLive.CronSetupComponent.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.time_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#time_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.weekday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#weekday_field/1"},{"type":"module","title":"LightningWeb.JobLive.JobBuilder","doc":"Job Builder Panel","ref":"LightningWeb.JobLive.JobBuilder.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.delete_title/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#delete_title/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.follow_run/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#follow_run/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.render/1","doc":"Attributes return_to ( :string ) (required) params ( :map ) - Defaults to %{} . can_edit_job ( :boolean ) (required) can_run_job ( :boolean ) (required)","ref":"LightningWeb.JobLive.JobBuilder.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_adaptor/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_adaptor/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_credential/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_credential/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.update_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#update_cron_expression/2"},{"type":"module","title":"LightningWeb.JobLive.JobBuilderComponents","doc":"","ref":"LightningWeb.JobLive.JobBuilderComponents.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.job_editor_component/1","doc":"Attributes adaptor ( :string ) (required) disabled ( :boolean ) - Defaults to false . source ( :string ) (required) change_event ( :string ) - Defaults to "job_body_changed" . Global attributes are accepted.","ref":"LightningWeb.JobLive.JobBuilderComponents.html#job_editor_component/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.trigger_picker/1","doc":"Attributes form ( :map ) (required) upstream_jobs ( :list ) (required) on_cron_change ( :any ) (required) disabled ( :boolean ) - Defaults to true .","ref":"LightningWeb.JobLive.JobBuilderComponents.html#trigger_picker/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.when_invalid/1","doc":"Attributes changeset ( :map ) (required) field ( :atom ) (required) Slots inner_block (required)","ref":"LightningWeb.JobLive.JobBuilderComponents.html#when_invalid/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.render/1","doc":"Attributes job ( :map ) (required) on_run ( :any ) (required) user ( :map ) (required) selected_dataclip_id ( :string ) (required)","ref":"LightningWeb.JobLive.ManualRunComponent.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.changeset/2","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html#changeset/2"},{"type":"module","title":"LightningWeb.LiveHelpers","doc":"General purpose LiveView helper functions","ref":"LightningWeb.LiveHelpers.html"},{"type":"function","title":"LightningWeb.LiveHelpers.display_short_uuid/1","doc":"","ref":"LightningWeb.LiveHelpers.html#display_short_uuid/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_in/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_in/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_out/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_out/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_error_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_error_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_info_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_info_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_nav_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_nav_block/1"},{"type":"module","title":"LightningWeb.ModalPortal","doc":"Component for rendering content inside layout without full DOM patch.","ref":"LightningWeb.ModalPortal.html"},{"type":"function","title":"LightningWeb.ModalPortal.close_modal/0","doc":"","ref":"LightningWeb.ModalPortal.html#close_modal/0"},{"type":"function","title":"LightningWeb.ModalPortal.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.ModalPortal.html#handle_event/3"},{"type":"function","title":"LightningWeb.ModalPortal.on_hide/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_hide/2"},{"type":"function","title":"LightningWeb.ModalPortal.on_show/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_show/2"},{"type":"function","title":"LightningWeb.ModalPortal.open_modal/2","doc":"","ref":"LightningWeb.ModalPortal.html#open_modal/2"},{"type":"function","title":"LightningWeb.ModalPortal.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ModalPortal.html#render/1"},{"type":"function","title":"LightningWeb.ModalPortal.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.ModalPortal.html#update/2"},{"type":"module","title":"LightningWeb.OauthCredentialHelper","doc":"A set of helper functions to encodes state and coordinate OAuth callbacks back to a LiveView component.","ref":"LightningWeb.OauthCredentialHelper.html"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast/2","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#broadcast/2"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast_forward/3","doc":"Broadcast a message specifically for forwarding a message to a component. It expects a subscription_id , the module of the component and opts being a keyword list containing an :id key of the specific component. See: Phoenix.LiveView.send_update/3 for more info. A corresponding LiveView (that is subscribed) is expected to have a matching handle_info/2 that looks like this: def handle_info ( { :forward , mod , opts } , socket ) do send_update ( mod , opts ) { :noreply , socket } end","ref":"LightningWeb.OauthCredentialHelper.html#broadcast_forward/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.build_state/3","doc":"Encode and encrypt the callback data which will be sent so a provider as the state key in the request. The values are: subscription_id The same ID used to subscribe. The component module The LiveView component that is going to receive update The component id The ID of the component","ref":"LightningWeb.OauthCredentialHelper.html#build_state/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.decode_state/1","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#decode_state/1"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.subscribe/1","doc":"Subscribe to the oauth_credential topic. It expects the a unique ID for the topic, usually the LiveView's socket.id .","ref":"LightningWeb.OauthCredentialHelper.html#subscribe/1"},{"type":"module","title":"LightningWeb.OidcController","doc":"","ref":"LightningWeb.OidcController.html"},{"type":"function","title":"LightningWeb.OidcController.new/2","doc":"Once the user has completed the authorization flow from above, they are returned here, and the authorization code is used to log them in.","ref":"LightningWeb.OidcController.html#new/2"},{"type":"function","title":"LightningWeb.OidcController.show/2","doc":"Given a known provider, redirect them to the authorize url on the provider","ref":"LightningWeb.OidcController.html#show/2"},{"type":"module","title":"LightningWeb.PageView","doc":"","ref":"LightningWeb.PageView.html"},{"type":"function","title":"LightningWeb.PageView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.PageView.html#__resource__/0"},{"type":"function","title":"LightningWeb.PageView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.PageView.html#render/2"},{"type":"function","title":"LightningWeb.PageView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.PageView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.Pagination","doc":"Pagination Components This has been extracted and adapted from scrivener_html . See: https://github.com/mgwidmann/scrivener_html","ref":"LightningWeb.Pagination.html"},{"type":"function","title":"LightningWeb.Pagination.page_link/1","doc":"","ref":"LightningWeb.Pagination.html#page_link/1"},{"type":"function","title":"LightningWeb.Pagination.pagination_bar/1","doc":"","ref":"LightningWeb.Pagination.html#pagination_bar/1"},{"type":"function","title":"LightningWeb.Pagination.raw_pagination_links/2","doc":"Returns the raw data in order to generate the proper HTML for pagination links. Data is returned in a {text, page_number} format where text is intended to be the text of the link and page_number is the page it should go to. Defaults are already supplied and they are as follows: [ distance : 5 , next : :next , previous : :previous , first : true , last : true , ellipsis : :ellipsis ] distance must be a positive non-zero integer or an exception is raised. next and previous should be strings but can be anything you want as long as it is truthy, falsey values will remove them from the output. first and last are only booleans, and they just include/remove their respective link from output. An example of the data returned: iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 10 , page_number : 5 } ) [ { "<<" , 4 } , { 1 , 1 } , { 2 , 2 } , { 3 , 3 } , { 4 , 4 } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { ">>" , 6 } ] iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 20 , page_number : 10 } , first : [ "←" ] , last : [ "→" ] ) [ { "<<" , 9 } , { [ "←" ] , 1 } , { :ellipsis , { :safe , "&hellip;" } } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { 11 , 11 } , { 12 , 12 } , { 13 , 13 } , { 14 , 14 } , { 15 , 15 } , { :ellipsis , { :safe , "&hellip;" } } , { [ "→" ] , 20 } , { ">>" , 11 } ] Simply loop and pattern match over each item and transform it to your custom HTML.","ref":"LightningWeb.Pagination.html#raw_pagination_links/2"},{"type":"module","title":"LightningWeb.Plugs.FirstSetup","doc":"Plug to redirect HTTP requests to /first_setup if there are no superusers in the system yet.","ref":"LightningWeb.Plugs.FirstSetup.html"},{"type":"module","title":"LightningWeb.ProfileLive.Edit","doc":"LiveView for user profile page.","ref":"LightningWeb.ProfileLive.Edit.html"},{"type":"function","title":"LightningWeb.ProfileLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProfileLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.FormComponent","doc":"Form component update profile email and password","ref":"LightningWeb.ProfileLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.MfaComponent","doc":"Component to enable MFA on a User's account","ref":"LightningWeb.ProfileLive.MfaComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.MfaComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.MfaComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.ProjectLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProjectLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProjectLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.ProjectLive.Index.html"},{"type":"function","title":"LightningWeb.ProjectLive.Index.delete_action/1","doc":"","ref":"LightningWeb.ProjectLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.ProjectLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Settings","doc":"Index Liveview for Runs","ref":"LightningWeb.ProjectLive.Settings.html"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.digest/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#digest/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.failure_alert/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#failure_alert/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Settings.html#render/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.role/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#role/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.user/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#user/1"},{"type":"module","title":"LightningWeb.RouteHelpers","doc":"Convenience functions for generating paths.","ref":"LightningWeb.RouteHelpers.html"},{"type":"function","title":"LightningWeb.RouteHelpers.oidc_callback_url/0","doc":"","ref":"LightningWeb.RouteHelpers.html#oidc_callback_url/0"},{"type":"function","title":"LightningWeb.RouteHelpers.show_run_url/2","doc":"","ref":"LightningWeb.RouteHelpers.html#show_run_url/2"},{"type":"module","title":"LightningWeb.Router","doc":"","ref":"LightningWeb.Router.html"},{"type":"function","title":"LightningWeb.Router.api/2","doc":"","ref":"LightningWeb.Router.html#api/2"},{"type":"function","title":"LightningWeb.Router.browser/2","doc":"","ref":"LightningWeb.Router.html#browser/2"},{"type":"function","title":"LightningWeb.Router.call/2","doc":"Callback invoked by Plug on every request.","ref":"LightningWeb.Router.html#call/2"},{"type":"function","title":"LightningWeb.Router.init/1","doc":"Callback required by Plug that initializes the router for serving web requests.","ref":"LightningWeb.Router.html#init/1"},{"type":"function","title":"LightningWeb.Router.storybook_assets/2","doc":"","ref":"LightningWeb.Router.html#storybook_assets/2"},{"type":"function","title":"LightningWeb.Router.storybook_browser/2","doc":"","ref":"LightningWeb.Router.html#storybook_browser/2"},{"type":"module","title":"LightningWeb.RunLive.Index","doc":"Index Liveview for Runs","ref":"LightningWeb.RunLive.Index.html"},{"type":"function","title":"LightningWeb.RunLive.Index.checked/2","doc":"","ref":"LightningWeb.RunLive.Index.html#checked/2"},{"type":"function","title":"LightningWeb.RunLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.RunLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.RunLive.RerunJobComponent","doc":"Rerun job component","ref":"LightningWeb.RunLive.RerunJobComponent.html"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#handle_event/3"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#render/1"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#update/2"},{"type":"module","title":"LightningWeb.RunLive.RunViewerLive","doc":"","ref":"LightningWeb.RunLive.RunViewerLive.html"},{"type":"function","title":"LightningWeb.RunLive.RunViewerLive.handle_info/2","doc":"Reload the run when any update messages arrive.","ref":"LightningWeb.RunLive.RunViewerLive.html#handle_info/2"},{"type":"module","title":"LightningWeb.RunLive.Show","doc":"Show page for individual runs.","ref":"LightningWeb.RunLive.Show.html"},{"type":"function","title":"LightningWeb.RunLive.Show.apply_action/3","doc":"","ref":"LightningWeb.RunLive.Show.html#apply_action/3"},{"type":"module","title":"LightningWeb.RunLive.WorkOrderComponent","doc":"Workorder component","ref":"LightningWeb.RunLive.WorkOrderComponent.html"},{"type":"function","title":"LightningWeb.RunLive.WorkOrderComponent.render/1","doc":"Attributes show_details ( :boolean ) - Defaults to false . entry_selected ( :boolean ) - Defaults to false .","ref":"LightningWeb.RunLive.WorkOrderComponent.html#render/1"},{"type":"module","title":"LightningWeb.Telemetry","doc":"Assorted metrics to collect during runtime. See https://hexdocs.pm/phoenix/telemetry.html","ref":"LightningWeb.Telemetry.html"},{"type":"function","title":"LightningWeb.Telemetry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"LightningWeb.Telemetry.html#child_spec/1"},{"type":"function","title":"LightningWeb.Telemetry.metrics/0","doc":"","ref":"LightningWeb.Telemetry.html#metrics/0"},{"type":"function","title":"LightningWeb.Telemetry.start_link/1","doc":"","ref":"LightningWeb.Telemetry.html#start_link/1"},{"type":"module","title":"LightningWeb.TokensLive.Index","doc":"LiveView for listing and managing tokens","ref":"LightningWeb.TokensLive.Index.html"},{"type":"function","title":"LightningWeb.TokensLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.TokensLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserAuth","doc":"The UserAuth controller.","ref":"LightningWeb.UserAuth.html"},{"type":"function","title":"LightningWeb.UserAuth.authenticate_bearer/2","doc":"","ref":"LightningWeb.UserAuth.html#authenticate_bearer/2"},{"type":"function","title":"LightningWeb.UserAuth.fetch_current_user/2","doc":"Authenticates the user by looking into the session and remember me token.","ref":"LightningWeb.UserAuth.html#fetch_current_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_in_user/2","doc":"Logs the user in by creating a new session token.","ref":"LightningWeb.UserAuth.html#log_in_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_out_user/1","doc":"Logs the user out. It clears all session data for safety. See renew_session.","ref":"LightningWeb.UserAuth.html#log_out_user/1"},{"type":"function","title":"LightningWeb.UserAuth.mark_totp_pending/1","doc":"","ref":"LightningWeb.UserAuth.html#mark_totp_pending/1"},{"type":"function","title":"LightningWeb.UserAuth.new_session/2","doc":"Assigns the token to a new session. It renews the session ID and clears the whole session to avoid fixation attacks. See the renew_session function to customize this behaviour. It also sets a :live_socket_id key in the session, so LiveView sessions are identified and automatically disconnected on log out. The line can be safely removed if you are not using LiveView.","ref":"LightningWeb.UserAuth.html#new_session/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_if_user_is_authenticated/2","doc":"Used for routes that require the user to not be authenticated.","ref":"LightningWeb.UserAuth.html#redirect_if_user_is_authenticated/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_with_return_to/2","doc":"Returns to or redirects to the dashboard and potentially set remember_me token.","ref":"LightningWeb.UserAuth.html#redirect_with_return_to/2"},{"type":"function","title":"LightningWeb.UserAuth.require_authenticated_user/2","doc":"Used for routes that require the user to be authenticated. If you want to enforce the user email is confirmed before they use the application at all, here would be a good place.","ref":"LightningWeb.UserAuth.html#require_authenticated_user/2"},{"type":"function","title":"LightningWeb.UserAuth.totp_pending?/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_pending?/1"},{"type":"function","title":"LightningWeb.UserAuth.totp_validated/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_validated/1"},{"type":"module","title":"LightningWeb.UserConfirmationController","doc":"","ref":"LightningWeb.UserConfirmationController.html"},{"type":"function","title":"LightningWeb.UserConfirmationController.confirm_email/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#confirm_email/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.create/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#create/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.edit/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#edit/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.new/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#new/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.update/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#update/2"},{"type":"module","title":"LightningWeb.UserLive.Edit","doc":"LiveView for editing a single job, which inturn uses LightningWeb.JobLive.BigFormComponent for common functionality.","ref":"LightningWeb.UserLive.Edit.html"},{"type":"function","title":"LightningWeb.UserLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.FormComponent","doc":"Form component for creating and editing users","ref":"LightningWeb.UserLive.FormComponent.html"},{"type":"function","title":"LightningWeb.UserLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.UserLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.Index","doc":"Index page for listing users","ref":"LightningWeb.UserLive.Index.html"},{"type":"function","title":"LightningWeb.UserLive.Index.delete_action/1","doc":"","ref":"LightningWeb.UserLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.UserLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserRegistrationController","doc":"","ref":"LightningWeb.UserRegistrationController.html"},{"type":"function","title":"LightningWeb.UserRegistrationController.create/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#create/2"},{"type":"function","title":"LightningWeb.UserRegistrationController.new/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#new/2"},{"type":"module","title":"LightningWeb.UserResetPasswordController","doc":"","ref":"LightningWeb.UserResetPasswordController.html"},{"type":"function","title":"LightningWeb.UserResetPasswordController.create/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#create/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.edit/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#edit/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.new/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#new/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.update/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#update/2"},{"type":"module","title":"LightningWeb.UserSessionController","doc":"","ref":"LightningWeb.UserSessionController.html"},{"type":"function","title":"LightningWeb.UserSessionController.auth_handler_url/0","doc":"","ref":"LightningWeb.UserSessionController.html#auth_handler_url/0"},{"type":"function","title":"LightningWeb.UserSessionController.create/2","doc":"","ref":"LightningWeb.UserSessionController.html#create/2"},{"type":"function","title":"LightningWeb.UserSessionController.delete/2","doc":"","ref":"LightningWeb.UserSessionController.html#delete/2"},{"type":"function","title":"LightningWeb.UserSessionController.exchange_token/2","doc":"","ref":"LightningWeb.UserSessionController.html#exchange_token/2"},{"type":"function","title":"LightningWeb.UserSessionController.new/2","doc":"","ref":"LightningWeb.UserSessionController.html#new/2"},{"type":"module","title":"LightningWeb.UserTOTPController","doc":"","ref":"LightningWeb.UserTOTPController.html"},{"type":"function","title":"LightningWeb.UserTOTPController.create/2","doc":"","ref":"LightningWeb.UserTOTPController.html#create/2"},{"type":"function","title":"LightningWeb.UserTOTPController.new/2","doc":"","ref":"LightningWeb.UserTOTPController.html#new/2"},{"type":"module","title":"LightningWeb.WebhooksController","doc":"","ref":"LightningWeb.WebhooksController.html"},{"type":"function","title":"LightningWeb.WebhooksController.create/2","doc":"","ref":"LightningWeb.WebhooksController.html#create/2"},{"type":"module","title":"LightningWeb.WorkflowLive.EditorPane","doc":"","ref":"LightningWeb.WorkflowLive.EditorPane.html"},{"type":"function","title":"LightningWeb.WorkflowLive.EditorPane.render/1","doc":"Attributes id ( :string ) (required) disabled ( :boolean ) - Defaults to false . class ( :string ) - Defaults to "" . on_change ( :any ) (required) adaptor ( :string ) (required) source ( :string ) (required) job_id ( :string ) (required)","ref":"LightningWeb.WorkflowLive.EditorPane.html#render/1"},{"type":"module","title":"LightningWeb.WorkflowLive.JobView","doc":"","ref":"LightningWeb.WorkflowLive.JobView.html"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.container/1","doc":"Attributes id ( :string ) (required) Slots top column - Accepts attributes: class ( :string ) - Extra CSS classes for the column. bottom","ref":"LightningWeb.WorkflowLive.JobView.html#container/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.input_pane/1","doc":"Attributes job ( :map ) (required) user ( :map ) (required) project ( :map ) (required) on_run ( :any ) (required) - Callback to run a job manually. can_run_job ( :boolean ) - Defaults to true .","ref":"LightningWeb.WorkflowLive.JobView.html#input_pane/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.job_edit_view/1","doc":"Attributes job ( :map ) (required) form ( :map ) (required) - A form built from a job. current_user ( :map ) (required) project ( :map ) (required) close_url ( :any ) (required) socket ( :any ) (required) on_run ( :any ) (required) - Callback to run a job manually. follow_run_id ( :any ) - Defaults to nil . Slots footer","ref":"LightningWeb.WorkflowLive.JobView.html#job_edit_view/1"},{"type":"module","title":"LightningWeb.WorkflowNewLive.WorkflowParams","doc":"Various function for reconciling changes to a workflow params map. The front end editor uses JSON patches to represent changes to the workflow.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_form_params/2","doc":"Produce a new set of params by applying the given form params to the current params.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_form_params/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_patches/2","doc":"Produce a new set of params by applying the given patches to the current parms","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_patches/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_map/1","doc":"Convert a changeset to a serializable map of workflow params, suitable for sending to the front end editor. It uses Lightning.Helpers.json_safe/1 to ensure that the map is safe to serialize to JSON. This is necessary because the underlying model may contain atom values.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_map/1"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_patches/2","doc":"Produce a set of patches that represent the difference between the initial params and the target params. This usually is used to produce a set of patches that represent the changes introduced by a changeset.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_patches/2"},{"type":"task","title":"Mix.Tasks.Lightning.GenEncryptionKey","doc":"Helper to generate a unique encryption key for Vault","ref":"Mix.Tasks.Lightning.GenEncryptionKey.html"},{"type":"task","title":"Mix.Tasks.Lightning.InstallRuntime","doc":"Installs the following NodeJS packages: core language-common","ref":"Mix.Tasks.Lightning.InstallRuntime.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.packages/0","doc":"","ref":"Mix.Tasks.Lightning.InstallRuntime.html#packages/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallRuntime.html#run/1"},{"type":"task","title":"Mix.Tasks.Lightning.InstallSchemas","doc":"Install the credential json schemas Use --exclude language-package1, language-package2 to exclude specific packages","ref":"Mix.Tasks.Lightning.InstallSchemas.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.fetch_schemas/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#fetch_schemas/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.parse_excluded/1","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#parse_excluded/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.persist_schema/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#persist_schema/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_options/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_params/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_chunk/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request!/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#run/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#start/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#stream_next/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.write_schema/3","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#write_schema/3"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.body/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:body/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.headers/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:headers/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.method/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:method/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.options/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:options/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.params/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:params/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.request/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:request/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.url/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:url/0"},{"type":"module","title":"ObanPruner","doc":"The Oban Pruner removes completed Oban jobs. It leaves everything else for manual inspection.","ref":"ObanPruner.html"},{"type":"function","title":"ObanPruner.perform/1","doc":"Deletes completed Oban jobs, leaving discarded for manual inspection.","ref":"ObanPruner.html#perform/1"},{"type":"module","title":"Storybook.Root","doc":"","ref":"Storybook.Root.html"},{"type":"module","title":"Lightning.Accounts","doc":"The Accounts context.","ref":"Lightning.Accounts.html"},{"type":"function","title":"Lightning.Accounts.apply_user_email/3","doc":"Emulates that the email will change without actually changing it in the database. Examples iex> apply_user_email ( user , "valid password" , %{ email : ... } ) { :ok , % User { } } role : :superuser iex> apply_user_email ( user , "invalid password" , %{ email : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#apply_user_email/3"},{"type":"function","title":"Lightning.Accounts.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Accounts.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Accounts.change_scheduled_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the user scheduled_deletion. Examples iex> change_scheduled_deletion ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_scheduled_deletion/2"},{"type":"function","title":"Lightning.Accounts.change_superuser_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking superuser changes. Examples iex> change_superuser_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_superuser_registration/1"},{"type":"function","title":"Lightning.Accounts.change_user_details/2","doc":"","ref":"Lightning.Accounts.html#change_user_details/2"},{"type":"function","title":"Lightning.Accounts.change_user_email/2","doc":"Returns an %Ecto.Changeset{} for changing the user email. Examples iex> change_user_email ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_email/2"},{"type":"function","title":"Lightning.Accounts.change_user_password/2","doc":"Returns an %Ecto.Changeset{} for changing the user password. Examples iex> change_user_password ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_password/2"},{"type":"function","title":"Lightning.Accounts.change_user_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking user changes. Examples iex> change_user_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_registration/1"},{"type":"function","title":"Lightning.Accounts.confirm_user/1","doc":"Confirms a user by the given token. If the token matches, the user account is marked as confirmed and the token is deleted.","ref":"Lightning.Accounts.html#confirm_user/1"},{"type":"function","title":"Lightning.Accounts.delete_auth_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_auth_token/1"},{"type":"function","title":"Lightning.Accounts.delete_session_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_session_token/1"},{"type":"function","title":"Lightning.Accounts.delete_token/1","doc":"Deletes a token. Examples iex> delete_token ( token ) { :ok , % UserToken { } } iex> delete_token ( token ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_token/1"},{"type":"function","title":"Lightning.Accounts.delete_user/1","doc":"Deletes a user. Examples iex> delete_user ( user ) { :ok , % User { } } iex> delete_user ( user ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_user/1"},{"type":"function","title":"Lightning.Accounts.delete_user_totp/1","doc":"Deletes the given user's TOTP","ref":"Lightning.Accounts.html#delete_user_totp/1"},{"type":"function","title":"Lightning.Accounts.deliver_update_email_instructions/3","doc":"Delivers the update email instructions to the given user. Examples iex> deliver_update_email_instructions ( user , current_email , & Routes . user_update_email_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_update_email_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/2","doc":"Delivers the confirmation email instructions to the given user. Examples iex> deliver_user_confirmation_instructions ( user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } } iex> deliver_user_confirmation_instructions ( confirmed_user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :error , :already_confirmed }","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/3","doc":"","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_reset_password_instructions/2","doc":"Delivers the reset password email to the given user. Examples iex> deliver_user_reset_password_instructions ( user , & Routes . user_reset_password_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_user_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.exchange_auth_token/1","doc":"Exchanges an auth token for a session token. The auth token is removed from the database if successful.","ref":"Lightning.Accounts.html#exchange_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_api_token/1","doc":"Generates an API token for a user.","ref":"Lightning.Accounts.html#generate_api_token/1"},{"type":"function","title":"Lightning.Accounts.generate_auth_token/1","doc":"Generates an auth token.","ref":"Lightning.Accounts.html#generate_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_user_session_token/1","doc":"Generates a session token.","ref":"Lightning.Accounts.html#generate_user_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_token!/1","doc":"Gets a single token. Raises Ecto.NoResultsError if the UserToken does not exist. Examples iex> get_token! ( 123 ) % UserToken { } iex> get_token! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_token!/1"},{"type":"function","title":"Lightning.Accounts.get_user!/1","doc":"Gets a single user. Raises Ecto.NoResultsError if the User does not exist. Examples iex> get_user! ( 123 ) % User { } iex> get_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_user!/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_api_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_api_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_auth_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_auth_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email/1","doc":"Gets a user by email. Examples iex> get_user_by_email ( "foo@example.com" ) % User { } iex> get_user_by_email ( "unknown@example.com" ) nil","ref":"Lightning.Accounts.html#get_user_by_email/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email_and_password/2","doc":"Gets a user by email and password. Examples iex> get_user_by_email_and_password ( "foo@example.com" , "correct_password" ) % User { } iex> get_user_by_email_and_password ( "foo@example.com" , "invalid_password" ) nil","ref":"Lightning.Accounts.html#get_user_by_email_and_password/2"},{"type":"function","title":"Lightning.Accounts.get_user_by_reset_password_token/1","doc":"Gets the user by reset password token. Examples iex> get_user_by_reset_password_token ( "validtoken" ) % User { } iex> get_user_by_reset_password_token ( "invalidtoken" ) nil","ref":"Lightning.Accounts.html#get_user_by_reset_password_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_session_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_totp/1","doc":"Gets a single UserTOTP if any exists.","ref":"Lightning.Accounts.html#get_user_totp/1"},{"type":"function","title":"Lightning.Accounts.get_users_to_alert_for_project/1","doc":"Gets all users to alert of workflow failure for a project","ref":"Lightning.Accounts.html#get_users_to_alert_for_project/1"},{"type":"function","title":"Lightning.Accounts.has_activity_in_projects?/1","doc":"","ref":"Lightning.Accounts.html#has_activity_in_projects?/1"},{"type":"function","title":"Lightning.Accounts.has_one_superuser?/0","doc":"Used to determine if there is at least one Superuser in the system. This triggers the setup page on fresh installs.","ref":"Lightning.Accounts.html#has_one_superuser?/0"},{"type":"function","title":"Lightning.Accounts.list_api_tokens/1","doc":"Lists all user tokens","ref":"Lightning.Accounts.html#list_api_tokens/1"},{"type":"function","title":"Lightning.Accounts.list_users/0","doc":"Returns the list of users. Examples iex> list_users ( ) [ % User { } , ... ]","ref":"Lightning.Accounts.html#list_users/0"},{"type":"function","title":"Lightning.Accounts.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find users that are ready for permanent deletion and purge them.","ref":"Lightning.Accounts.html#perform/1"},{"type":"function","title":"Lightning.Accounts.purge_user/1","doc":"","ref":"Lightning.Accounts.html#purge_user/1"},{"type":"function","title":"Lightning.Accounts.register_superuser/1","doc":"Registers a superuser. Examples iex> register_superuser ( %{ field : value } ) { :ok , % User { } } iex> register_superuser ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_superuser/1"},{"type":"function","title":"Lightning.Accounts.register_user/1","doc":"Registers a user. Examples iex> register_user ( %{ field : value } ) { :ok , % User { } } iex> register_user ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_user/1"},{"type":"function","title":"Lightning.Accounts.reset_user_password/2","doc":"Resets the user password. Examples iex> reset_user_password ( user , %{ password : "new long password" , password_confirmation : "new long password" } ) { :ok , % User { } } iex> reset_user_password ( user , %{ password : "valid" , password_confirmation : "not the same" } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#reset_user_password/2"},{"type":"function","title":"Lightning.Accounts.schedule_user_deletion/2","doc":"Given a user and a confirmation email, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic user purge cronjob will never run. (Note that subsequent logins will be blocked for users pending deletion.)","ref":"Lightning.Accounts.html#schedule_user_deletion/2"},{"type":"function","title":"Lightning.Accounts.update_user_details/2","doc":"","ref":"Lightning.Accounts.html#update_user_details/2"},{"type":"function","title":"Lightning.Accounts.update_user_email/2","doc":"Updates the user email using the given token. If the token matches, the user email is updated and the token is deleted. The confirmed_at date is also updated to the current time.","ref":"Lightning.Accounts.html#update_user_email/2"},{"type":"function","title":"Lightning.Accounts.update_user_password/3","doc":"Updates the user password. Examples iex> update_user_password ( user , "valid password" , %{ password : ... } ) { :ok , % User { } } iex> update_user_password ( user , "invalid password" , %{ password : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#update_user_password/3"},{"type":"function","title":"Lightning.Accounts.upsert_user_totp/2","doc":"Updates or Inserts the user's TOTP","ref":"Lightning.Accounts.html#upsert_user_totp/2"},{"type":"function","title":"Lightning.Accounts.valid_user_totp?/2","doc":"Validates if the given TOTP code is valid.","ref":"Lightning.Accounts.html#valid_user_totp?/2"},{"type":"function","title":"Lightning.Accounts.validate_change_user_email/2","doc":"","ref":"Lightning.Accounts.html#validate_change_user_email/2"},{"type":"module","title":"Lightning.Accounts.User","doc":"The User model.","ref":"Lightning.Accounts.User.html"},{"type":"function","title":"Lightning.Accounts.User.confirm_changeset/1","doc":"Confirms the account by setting confirmed_at .","ref":"Lightning.Accounts.User.html#confirm_changeset/1"},{"type":"function","title":"Lightning.Accounts.User.details_changeset/2","doc":"A user changeset for user details: email first_name last_name role","ref":"Lightning.Accounts.User.html#details_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.email_changeset/2","doc":"A user changeset for changing the email. It requires the email to change otherwise an error is added.","ref":"Lightning.Accounts.User.html#email_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.password_changeset/3","doc":"A user changeset for changing the password. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#password_changeset/3"},{"type":"function","title":"Lightning.Accounts.User.scheduled_deletion_changeset/2","doc":"A user changeset for changing the scheduled_deletion property.","ref":"Lightning.Accounts.User.html#scheduled_deletion_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.superuser_registration_changeset/2","doc":"A superuser changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#superuser_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.user_registration_changeset/2","doc":"A user changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#user_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.valid_password?/2","doc":"Verifies the password. If there is no user or the user doesn't have a password, we call Bcrypt.no_user_verify/0 to avoid timing attacks.","ref":"Lightning.Accounts.User.html#valid_password?/2"},{"type":"function","title":"Lightning.Accounts.User.validate_current_password/2","doc":"Validates the current password otherwise adds an error to the changeset.","ref":"Lightning.Accounts.User.html#validate_current_password/2"},{"type":"type","title":"Lightning.Accounts.User.t/0","doc":"","ref":"Lightning.Accounts.User.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.User.RolesEnum","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.create_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Accounts.User.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schema/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Accounts.User.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Accounts.User.RolesEnum.t/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserNotifier","doc":"The UserNotifier module.","ref":"Lightning.Accounts.UserNotifier.html"},{"type":"function","title":"Lightning.Accounts.UserNotifier.build_digest_url/3","doc":"","ref":"Lightning.Accounts.UserNotifier.html#build_digest_url/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/2","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/3","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_addition_notification/2","doc":"Deliver email to notify user of his addition of a project.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_addition_notification/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_digest/2","doc":"Deliver a project digest of daily/weekly or monthly activity to a user.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_digest/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_reset_password_instructions/2","doc":"Deliver instructions to reset a user password.","ref":"Lightning.Accounts.UserNotifier.html#deliver_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_instructions/2","doc":"Deliver instructions to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_warning/2","doc":"Deliver warning to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_warning/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.notify_project_deletion/2","doc":"","ref":"Lightning.Accounts.UserNotifier.html#notify_project_deletion/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.send_deletion_notification_email/1","doc":"Deliver an email to notify the user about their account being deleted","ref":"Lightning.Accounts.UserNotifier.html#send_deletion_notification_email/1"},{"type":"module","title":"Lightning.Accounts.UserTOTP","doc":"User Time based OTPs schema","ref":"Lightning.Accounts.UserTOTP.html"},{"type":"function","title":"Lightning.Accounts.UserTOTP.changeset/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserTOTP.valid_totp?/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#valid_totp?/2"},{"type":"type","title":"Lightning.Accounts.UserTOTP.t/0","doc":"","ref":"Lightning.Accounts.UserTOTP.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserToken","doc":"The UserToken model. The reason why we store session tokens in the database, even though Phoenix already provides a session cookie, is because Phoenix' default session cookies are not persisted, they are simply signed and potentially encrypted. This means they are valid indefinitely, unless you change the signing/encryption salt. Therefore, storing them allows individual user sessions to be expired. The token system can also be extended to store additional data, such as the device used for logging in. You could then use this information to display all valid sessions and devices in the UI and allow users to explicitly expire any session they deem invalid.","ref":"Lightning.Accounts.UserToken.html"},{"type":"function","title":"Lightning.Accounts.UserToken.build_email_token/3","doc":"Builds a token and its hash to be delivered to the user's email. The non-hashed token is sent to the user email while the hashed part is stored in the database. The original token cannot be reconstructed, which means anyone with read-only access to the database cannot directly use the token in the application to gain access. Furthermore, if the user changes their email in the system, the tokens sent to the previous email are no longer valid. Users can easily adapt the existing code to provide other types of delivery methods, for example, by phone numbers.","ref":"Lightning.Accounts.UserToken.html#build_email_token/3"},{"type":"function","title":"Lightning.Accounts.UserToken.build_token/2","doc":"Generates a token that will be stored in a signed place, such as session or cookie. As they are signed, those tokens do not need to be hashed.","ref":"Lightning.Accounts.UserToken.html#build_token/2"},{"type":"function","title":"Lightning.Accounts.UserToken.changeset/2","doc":"","ref":"Lightning.Accounts.UserToken.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign/2","doc":"Combines generate_claims/1 and encode_and_sign/2","ref":"Lightning.Accounts.UserToken.html#generate_and_sign/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign!/2","doc":"Same as generate_and_sign/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#generate_and_sign!/2"},{"type":"function","title":"Lightning.Accounts.UserToken.last_used_changeset/1","doc":"Update when the api token was last used by setting last_used_at .","ref":"Lightning.Accounts.UserToken.html#last_used_changeset/1"},{"type":"function","title":"Lightning.Accounts.UserToken.token_and_context_query/2","doc":"Returns the token struct for the given token value and context.","ref":"Lightning.Accounts.UserToken.html#token_and_context_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.user_and_contexts_query/2","doc":"Gets all tokens for the given user for the given contexts.","ref":"Lightning.Accounts.UserToken.html#user_and_contexts_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate/3","doc":"Combines verify/2 and validate/2","ref":"Lightning.Accounts.UserToken.html#verify_and_validate/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate!/3","doc":"Same as verify_and_validate/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#verify_and_validate!/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_change_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. This is used to validate requests to change the user email. It is different from verify_email_token_query/2 precisely because verify_email_token_query/2 validates the email has not changed, which is the starting point by this function. The given token is valid if it matches its hashed counterpart in the database and if it has not expired (after @change_email_validity_in_days). The context must always start with "change:".","ref":"Lightning.Accounts.UserToken.html#verify_change_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The given token is valid if it matches its hashed counterpart in the database and the user email has not changed. This function also checks if the token is being used within a certain period, depending on the context. The default contexts supported by this function are either "confirm", for account confirmation emails, and "reset_password", for resetting the password. For verifying requests to change the email, see verify_change_email_token_query/2 .","ref":"Lightning.Accounts.UserToken.html#verify_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The token is valid if it matches the value in the database and it has not expired (after @auth_validity_in_seconds or @session_validity_in_days).","ref":"Lightning.Accounts.UserToken.html#verify_token_query/2"},{"type":"module","title":"Lightning.Credentials","doc":"The Credentials context.","ref":"Lightning.Credentials.html"},{"type":"function","title":"Lightning.Credentials.change_credential/2","doc":"Returns an %Ecto.Changeset{} for tracking credential changes. Examples iex> change_credential ( credential ) % Ecto.Changeset { data : % Credential { } }","ref":"Lightning.Credentials.html#change_credential/2"},{"type":"function","title":"Lightning.Credentials.create_credential/1","doc":"Creates a credential. Examples iex> create_credential ( %{ field : value } ) { :ok , % Credential { } } iex> create_credential ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#create_credential/1"},{"type":"function","title":"Lightning.Credentials.delete_credential/1","doc":"Deletes a credential. Examples iex> delete_credential ( credential ) { :ok , % Credential { } } iex> delete_credential ( credential ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#delete_credential/1"},{"type":"function","title":"Lightning.Credentials.get_credential!/1","doc":"Gets a single credential. Raises Ecto.NoResultsError if the Credential does not exist. Examples iex> get_credential! ( 123 ) % Credential { } iex> get_credential! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Credentials.html#get_credential!/1"},{"type":"function","title":"Lightning.Credentials.invalid_projects_for_user/2","doc":"Given a credential and a user, returns a list of invalid projects—i.e., those that the credential is shared with but that the user does not have access to. This is used to generate a validation error when a credential cannot be transferred. Examples iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ ] iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ "52ea8758-6ce5-43d7-912f-6a1e1f11dc55" ]","ref":"Lightning.Credentials.html#invalid_projects_for_user/2"},{"type":"function","title":"Lightning.Credentials.list_credentials/0","doc":"Returns the list of credentials. Examples iex> list_credentials ( ) [ % Credential { } , ... ]","ref":"Lightning.Credentials.html#list_credentials/0"},{"type":"function","title":"Lightning.Credentials.list_credentials/1","doc":"","ref":"Lightning.Credentials.html#list_credentials/1"},{"type":"function","title":"Lightning.Credentials.list_credentials_for_user/1","doc":"Returns the list of credentials for a given user. Examples iex> list_credentials_for_user ( 123 ) [ % Credential { user_id : 123 } , % Credential { user_id : 123 } , ... ]","ref":"Lightning.Credentials.html#list_credentials_for_user/1"},{"type":"function","title":"Lightning.Credentials.maybe_refresh_token/1","doc":"","ref":"Lightning.Credentials.html#maybe_refresh_token/1"},{"type":"function","title":"Lightning.Credentials.sensitive_values_for/1","doc":"","ref":"Lightning.Credentials.html#sensitive_values_for/1"},{"type":"function","title":"Lightning.Credentials.update_credential/2","doc":"Updates a credential. Examples iex> update_credential ( credential , %{ field : new_value } ) { :ok , % Credential { } } iex> update_credential ( credential , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#update_credential/2"},{"type":"module","title":"Lightning.Credentials.Audit","doc":"Model for storing changes to Credentials","ref":"Lightning.Credentials.Audit.html"},{"type":"function","title":"Lightning.Credentials.Audit.event/4","doc":"","ref":"Lightning.Credentials.Audit.html#event/4"},{"type":"function","title":"Lightning.Credentials.Audit.save/1","doc":"","ref":"Lightning.Credentials.Audit.html#save/1"},{"type":"module","title":"Lightning.Credentials.Credential","doc":"The Credential model.","ref":"Lightning.Credentials.Credential.html"},{"type":"type","title":"Lightning.Credentials.Credential.t/0","doc":"","ref":"Lightning.Credentials.Credential.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.Schema","doc":"Structure that can parse JsonSchemas (using ExJsonSchema ) and validate changesets for a given schema.","ref":"Lightning.Credentials.Schema.html"},{"type":"function","title":"Lightning.Credentials.Schema.new/2","doc":"","ref":"Lightning.Credentials.Schema.html#new/2"},{"type":"function","title":"Lightning.Credentials.Schema.properties/2","doc":"","ref":"Lightning.Credentials.Schema.html#properties/2"},{"type":"function","title":"Lightning.Credentials.Schema.required?/2","doc":"","ref":"Lightning.Credentials.Schema.html#required?/2"},{"type":"function","title":"Lightning.Credentials.Schema.validate/2","doc":"","ref":"Lightning.Credentials.Schema.html#validate/2"},{"type":"type","title":"Lightning.Credentials.Schema.t/0","doc":"","ref":"Lightning.Credentials.Schema.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.SchemaDocument","doc":"Provides facilities to dynamically create and validate a changeset for a given Schema","ref":"Lightning.Credentials.SchemaDocument.html"},{"type":"function","title":"Lightning.Credentials.SchemaDocument.changeset/3","doc":"","ref":"Lightning.Credentials.SchemaDocument.html#changeset/3"},{"type":"module","title":"Lightning.Credentials.SensitiveValues","doc":"Functions to pull out sensitive values inside a credential. These values are used to scrub logs for leaked secrets.","ref":"Lightning.Credentials.SensitiveValues.html"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.flatten_map/1","doc":"Keys that are not considered sensitive","ref":"Lightning.Credentials.SensitiveValues.html#flatten_map/1"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.secret_values/1","doc":"Given a map, find all values allowed (via @safe_keys ) and return them as a list.","ref":"Lightning.Credentials.SensitiveValues.html#secret_values/1"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:pairs/0"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.raw_pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:raw_pairs/0"},{"type":"module","title":"Lightning.Invocation","doc":"The Invocation context.","ref":"Lightning.Invocation.html"},{"type":"function","title":"Lightning.Invocation.change_dataclip/2","doc":"Returns an %Ecto.Changeset{} for tracking dataclip changes. Examples iex> change_dataclip ( dataclip ) % Ecto.Changeset { data : % Dataclip { } }","ref":"Lightning.Invocation.html#change_dataclip/2"},{"type":"function","title":"Lightning.Invocation.change_run/2","doc":"Returns an %Ecto.Changeset{} for tracking run changes. Examples iex> change_run ( run ) % Ecto.Changeset { data : % Run { } }","ref":"Lightning.Invocation.html#change_run/2"},{"type":"function","title":"Lightning.Invocation.create_dataclip/1","doc":"Creates a dataclip. Examples iex> create_dataclip ( %{ field : value } ) { :ok , % Dataclip { } } iex> create_dataclip ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_dataclip/1"},{"type":"function","title":"Lightning.Invocation.create_log_line/2","doc":"","ref":"Lightning.Invocation.html#create_log_line/2"},{"type":"function","title":"Lightning.Invocation.create_run/1","doc":"Creates a run. Examples iex> create_run ( %{ field : value } ) { :ok , % Run { } } iex> create_run ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_run/1"},{"type":"function","title":"Lightning.Invocation.delete_dataclip/1","doc":"Deletes a dataclip. Examples iex> delete_dataclip ( dataclip ) { :ok , % Dataclip { } } iex> delete_dataclip ( dataclip ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_dataclip/1"},{"type":"function","title":"Lightning.Invocation.delete_run/1","doc":"Deletes a run. Examples iex> delete_run ( run ) { :ok , % Run { } } iex> delete_run ( run ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_run/1"},{"type":"function","title":"Lightning.Invocation.filter_run_body_and_logs_where/2","doc":"","ref":"Lightning.Invocation.html#filter_run_body_and_logs_where/2"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_before_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_status_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_status_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workflow_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workflow_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_before_where/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip/1","doc":"Gets a single dataclip given one of: a Dataclip uuid a Run model Returns nil if the Dataclip does not exist. Examples iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) % Dataclip { } iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) nil iex> get_dataclip ( % Run { id : "a uuid" } ) % Dataclip { }","ref":"Lightning.Invocation.html#get_dataclip/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip!/1","doc":"Gets a single dataclip. Raises Ecto.NoResultsError if the Dataclip does not exist. Examples iex> get_dataclip! ( 123 ) % Dataclip { } iex> get_dataclip! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_dataclip!/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip_query/1","doc":"Query for retrieving the dataclip that a runs starting dataclip.","ref":"Lightning.Invocation.html#get_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_result_dataclip_query/1","doc":"Query for retrieving the dataclip that was the result of a successful run.","ref":"Lightning.Invocation.html#get_result_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_run!/1","doc":"Gets a single run. Raises Ecto.NoResultsError if the Run does not exist. Examples iex> get_run! ( 123 ) % Run { } iex> get_run! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_run!/1"},{"type":"function","title":"Lightning.Invocation.get_run_with_job!/1","doc":"Fetches a run and preloads the job via the run's event.","ref":"Lightning.Invocation.html#get_run_with_job!/1"},{"type":"function","title":"Lightning.Invocation.get_workorders_by_ids/1","doc":"","ref":"Lightning.Invocation.html#get_workorders_by_ids/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips/0","doc":"Returns the list of dataclips. Examples iex> list_dataclips ( ) [ % Dataclip { } , ... ]","ref":"Lightning.Invocation.html#list_dataclips/0"},{"type":"function","title":"Lightning.Invocation.list_dataclips/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_for_job/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_for_job/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_query/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_query/1"},{"type":"function","title":"Lightning.Invocation.list_runs/0","doc":"Returns the list of runs. Examples iex> list_runs ( ) [ % Run { } , ... ]","ref":"Lightning.Invocation.html#list_runs/0"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project/2","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project/2"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project_query/1","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project_query/1"},{"type":"function","title":"Lightning.Invocation.list_work_orders_for_project_query/2","doc":"","ref":"Lightning.Invocation.html#list_work_orders_for_project_query/2"},{"type":"function","title":"Lightning.Invocation.search_workorders/1","doc":"","ref":"Lightning.Invocation.html#search_workorders/1"},{"type":"function","title":"Lightning.Invocation.search_workorders/3","doc":"","ref":"Lightning.Invocation.html#search_workorders/3"},{"type":"function","title":"Lightning.Invocation.update_dataclip/2","doc":"Updates a dataclip. Examples iex> update_dataclip ( dataclip , %{ field : new_value } ) { :ok , % Dataclip { } } iex> update_dataclip ( dataclip , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_dataclip/2"},{"type":"function","title":"Lightning.Invocation.update_run/2","doc":"Updates a run. Examples iex> update_run ( run , %{ field : new_value } ) { :ok , % Run { } } iex> update_run ( run , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_run/2"},{"type":"function","title":"Lightning.Invocation.with_attempts/1","doc":"","ref":"Lightning.Invocation.html#with_attempts/1"},{"type":"module","title":"Lightning.Invocation.Dataclip","doc":"Ecto model for Dataclips. Dataclips represent some data that arrived in the system, and records both the data and the source of the data. Types :http_request The data arrived via a webhook. :global Was created manually, and is intended to be used multiple times. When repetitive static data is needed to be maintained, instead of hard-coding into a Job - a more convenient solution is to create a :global Dataclip and access it inside the Job. :run_result The final state of a successful run. :saved_input An arbitrary input, created by a user. (Only configuration will be overwritten.)","ref":"Lightning.Invocation.Dataclip.html"},{"type":"function","title":"Lightning.Invocation.Dataclip.get_types/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#get_types/0"},{"type":"function","title":"Lightning.Invocation.Dataclip.new/1","doc":"","ref":"Lightning.Invocation.Dataclip.html#new/1"},{"type":"function","title":"Lightning.Invocation.Dataclip.validate_by_type/1","doc":"Append validations based on the type of the Dataclip. :run_result must have an associated Run model.","ref":"Lightning.Invocation.Dataclip.html#validate_by_type/1"},{"type":"type","title":"Lightning.Invocation.Dataclip.source_type/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:source_type/0"},{"type":"type","title":"Lightning.Invocation.Dataclip.t/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.LogLine","doc":"Ecto model for run logs.","ref":"Lightning.Invocation.LogLine.html"},{"type":"function","title":"Lightning.Invocation.LogLine.validate/1","doc":"","ref":"Lightning.Invocation.LogLine.html#validate/1"},{"type":"type","title":"Lightning.Invocation.LogLine.t/0","doc":"","ref":"Lightning.Invocation.LogLine.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.Query","doc":"Query functions for working with Runs and Dataclips","ref":"Lightning.Invocation.Query.html"},{"type":"function","title":"Lightning.Invocation.Query.last_run_for_job/1","doc":"The last run for a job","ref":"Lightning.Invocation.Query.html#last_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.last_successful_run_for_job/1","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#last_successful_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_for/1","doc":"Runs for a specific user","ref":"Lightning.Invocation.Query.html#runs_for/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_with_code/2","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#runs_with_code/2"},{"type":"module","title":"Lightning.Invocation.Run","doc":"Ecto model for Runs. A run represents the work initiated for a Job with an input dataclip. Once completed (successfully) it will have an output_dataclip associated with it as well.","ref":"Lightning.Invocation.Run.html"},{"type":"function","title":"Lightning.Invocation.Run.new/1","doc":"","ref":"Lightning.Invocation.Run.html#new/1"},{"type":"function","title":"Lightning.Invocation.Run.new_from/1","doc":"Creates a new Run changeset, but copies over certain fields. This is used to create new runs for retrys.","ref":"Lightning.Invocation.Run.html#new_from/1"},{"type":"type","title":"Lightning.Invocation.Run.t/0","doc":"","ref":"Lightning.Invocation.Run.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReason","doc":"Ecto model for InvocationReasons.","ref":"Lightning.InvocationReason.html"},{"type":"function","title":"Lightning.InvocationReason.new/1","doc":"","ref":"Lightning.InvocationReason.html#new/1"},{"type":"function","title":"Lightning.InvocationReason.validate_by_trigger_type/1","doc":"","ref":"Lightning.InvocationReason.html#validate_by_trigger_type/1"},{"type":"type","title":"Lightning.InvocationReason.source_type/0","doc":"","ref":"Lightning.InvocationReason.html#t:source_type/0"},{"type":"type","title":"Lightning.InvocationReason.t/0","doc":"","ref":"Lightning.InvocationReason.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReasons","doc":"The InvocationReasons context.","ref":"Lightning.InvocationReasons.html"},{"type":"function","title":"Lightning.InvocationReasons.build/2","doc":"","ref":"Lightning.InvocationReasons.html#build/2"},{"type":"function","title":"Lightning.InvocationReasons.create_reason/1","doc":"Creates a reason. Examples iex> create_reason ( %{ field : value } ) { :ok , % InvocationReason { } } iex> create_reason ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.InvocationReasons.html#create_reason/1"},{"type":"type","title":"Lightning.InvocationReasons.reason_type/0","doc":"","ref":"Lightning.InvocationReasons.html#t:reason_type/0"},{"type":"module","title":"Lightning.Pipeline","doc":"Service class to coordinate the running of jobs, and their downstream jobs.","ref":"Lightning.Pipeline.html"},{"type":"function","title":"Lightning.Pipeline.assemble_logs_for_run/1","doc":"Return all logs for a run as a string of text, separated by new line breaks","ref":"Lightning.Pipeline.html#assemble_logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.logs_for_run/1","doc":"Return all logs for a run as a list","ref":"Lightning.Pipeline.html#logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.process/1","doc":"","ref":"Lightning.Pipeline.html#process/1"},{"type":"module","title":"Lightning.Pipeline.Runner","doc":"Job running entrypoint","ref":"Lightning.Pipeline.Runner.html"},{"type":"function","title":"Lightning.Pipeline.Runner.create_dataclip_from_result/2","doc":"Creates a dataclip linked to the run that just finished. If either the file doesn't exist or there is a JSON decoding error, it logs and returns an error tuple.","ref":"Lightning.Pipeline.Runner.html#create_dataclip_from_result/2"},{"type":"function","title":"Lightning.Pipeline.Runner.find_or_install_adaptor/1","doc":"Make sure an adaptor matching the name is available. If it is available, return it's Engine.Adaptor struct - if not then install it.","ref":"Lightning.Pipeline.Runner.html#find_or_install_adaptor/1"},{"type":"function","title":"Lightning.Pipeline.Runner.scrub_result/1","doc":"Scrubs values from all keys in configuration, will be replaced by extensions to scrubber.ex, which is currently only used for logs.","ref":"Lightning.Pipeline.Runner.html#scrub_result/1"},{"type":"function","title":"Lightning.Pipeline.Runner.start/2","doc":"Execute a Run. Given a valid run: Persist the Dataclip and the Job's body to disk Create a blank output file on disk Build up a %Lightning.Runtime.Runspec{} with the paths, and adaptor module name And start it via Handler.start/2 . The callbacks implemented on Handler ( c:Handler.on_start/1 and c:Handler.on_finish/2 ) update the run when a Run is started and when it's finished, attaching the exit_code and log when they are available.","ref":"Lightning.Pipeline.Runner.html#start/2"},{"type":"module","title":"Lightning.Pipeline.Runner.Handler","doc":"Custom handler callbacks for Lightnings use of Engine to execute runs.","ref":"Lightning.Pipeline.Runner.Handler.html"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.on_start/1","doc":"The on_start handler updates the run, setting the started_at time and stamping the run with the ID of the credential that was used, if any, to facilitate easier auditing.","ref":"Lightning.Pipeline.Runner.Handler.html#on_start/1"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.stop/1","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#stop/1"},{"type":"type","title":"Lightning.Pipeline.Runner.Handler.handler_opts/0","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#t:handler_opts/0"},{"type":"module","title":"Lightning.Pipeline.StateAssembler","doc":"Facilities for building the state for a Run How state is assembled For the most common jobs, an inbound webhook will store an :http_request type dataclip. The reason that is created is associated with the dataclip. At runtime, the initial state for a Run will be in the shape of: { "data" : < the dataclip > , "configuration" : < the job ' s credential > } Saved inputs Saved custom inputs will only have state.configuration changed, everything else will remain as displayed. Flow Jobs When a Job is triggered by a previous Jobs success or failure these are the rules for constructing that Jobs state: For jobs that trigger on it's upstream jobs failure, the event will have the previous runs input dataclip as its input dataclip. The state will also have the log of the previous run attached on the error key. For Jobs triggered by a previous success, the run will have the previous runs output dataclip as its input dataclip. :run_result dataclips are expected to already have a data key, and are merged into the root.","ref":"Lightning.Pipeline.StateAssembler.html"},{"type":"function","title":"Lightning.Pipeline.StateAssembler.assemble/1","doc":"Assemble state for use in a Run.","ref":"Lightning.Pipeline.StateAssembler.html#assemble/1"},{"type":"module","title":"Lightning.Jobs","doc":"The Jobs context.","ref":"Lightning.Jobs.html"},{"type":"function","title":"Lightning.Jobs.change_job/2","doc":"Returns an %Ecto.Changeset{} for tracking job changes. Examples iex> change_job ( job ) % Ecto.Changeset { data : % Job { } }","ref":"Lightning.Jobs.html#change_job/2"},{"type":"function","title":"Lightning.Jobs.create_job/1","doc":"Creates a job. Examples iex> create_job ( %{ field : value } ) { :ok , % Job { } } iex> create_job ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#create_job/1"},{"type":"function","title":"Lightning.Jobs.delete_job/1","doc":"Deletes a job. Examples iex> delete_job ( job ) { :ok , % Job { } } iex> delete_job ( job ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#delete_job/1"},{"type":"function","title":"Lightning.Jobs.get_downstream_jobs_for/2","doc":"Returns the list of downstream jobs for a given job, optionally matching a specific trigger type. When downstream_jobs_for is called without a trigger that means its between jobs when it called with a trigger that means we are starting from outside the pipeline","ref":"Lightning.Jobs.html#get_downstream_jobs_for/2"},{"type":"function","title":"Lightning.Jobs.get_job/1","doc":"","ref":"Lightning.Jobs.html#get_job/1"},{"type":"function","title":"Lightning.Jobs.get_job!/1","doc":"Gets a single job. Raises Ecto.NoResultsError if the Job does not exist. Examples iex> get_job! ( 123 ) % Job { } iex> get_job! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Jobs.html#get_job!/1"},{"type":"function","title":"Lightning.Jobs.get_upstream_jobs_for/1","doc":"Returns the list of jobs excluding the one given.","ref":"Lightning.Jobs.html#get_upstream_jobs_for/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project_query/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project_query/1"},{"type":"function","title":"Lightning.Jobs.list_active_cron_jobs/0","doc":"","ref":"Lightning.Jobs.html#list_active_cron_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs/0","doc":"Returns the list of jobs.","ref":"Lightning.Jobs.html#list_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs_for_workflow/1","doc":"","ref":"Lightning.Jobs.html#list_jobs_for_workflow/1"},{"type":"function","title":"Lightning.Jobs.update_job/2","doc":"Updates a job. Examples iex> update_job ( job , %{ field : new_value } ) { :ok , % Job { } } iex> update_job ( job , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#update_job/2"},{"type":"module","title":"Lightning.Jobs.Job","doc":"Ecto model for Jobs. A Job contains the fields for defining a job. body The expression/javascript code name A plain text identifier adaptor An NPM style string that contains both the module name and it's version. E.g. @openfn/language-http@v1.2.3 or @openfn/language-foo@latest . While the version suffix isn't enforced here as it's not strictly necessary in this context, the front end will ensure a version is stated ( @latest being the default).","ref":"Lightning.Jobs.Job.html"},{"type":"function","title":"Lightning.Jobs.Job.new/1","doc":"","ref":"Lightning.Jobs.Job.html#new/1"},{"type":"function","title":"Lightning.Jobs.Job.put_project_credential/2","doc":"","ref":"Lightning.Jobs.Job.html#put_project_credential/2"},{"type":"function","title":"Lightning.Jobs.Job.put_workflow/2","doc":"Attaches a workflow to a job, this is useful when you have an unpersisted Workflow changeset - and want it to be created at the same time as a Job. Example: workflow = Ecto.Changeset . cast ( % Lightning.Workflows.Workflow { } , %{ "project_id" => attrs [ :project_id ] , "id" => Ecto.UUID . generate ( ) } , [ :project_id , :id ] ) job = % Job { } |> Ecto.Changeset . change ( ) |> Job . put_workflow ( workflow ) |> Job . changeset ( attrs )","ref":"Lightning.Jobs.Job.html#put_workflow/2"},{"type":"function","title":"Lightning.Jobs.Job.validate/1","doc":"","ref":"Lightning.Jobs.Job.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Job.t/0","doc":"","ref":"Lightning.Jobs.Job.html#t:t/0"},{"type":"module","title":"Lightning.Jobs.Query","doc":"Query module for finding Jobs.","ref":"Lightning.Jobs.Query.html"},{"type":"function","title":"Lightning.Jobs.Query.enabled_cron_jobs_by_edge/0","doc":"Returns active jobs with their cron triggers for use in the cron scheduling service.","ref":"Lightning.Jobs.Query.html#enabled_cron_jobs_by_edge/0"},{"type":"function","title":"Lightning.Jobs.Query.jobs_for/1","doc":"Returns all jobs accessible to a user, via their projects or all jobs in a given project.","ref":"Lightning.Jobs.Query.html#jobs_for/1"},{"type":"module","title":"Lightning.Jobs.Scheduler","doc":"The Scheduler is responsible for finding jobs that are ready to run based on their cron schedule, and then running them.","ref":"Lightning.Jobs.Scheduler.html"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/0","doc":"Find and start any cronjobs that are scheduled to run for a given time (defaults to the current time).","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/0"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/1","doc":"","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/1"},{"type":"module","title":"Lightning.Jobs.Trigger","doc":"Ecto model for Triggers. Triggers represent the criteria in which a Job might be invoked. Types Webhook (default) A webhook trigger allows a Job to invoked (via Lightning.Invocation ) when it's endpoint is called.","ref":"Lightning.Jobs.Trigger.html"},{"type":"function","title":"Lightning.Jobs.Trigger.new/1","doc":"","ref":"Lightning.Jobs.Trigger.html#new/1"},{"type":"function","title":"Lightning.Jobs.Trigger.validate/1","doc":"","ref":"Lightning.Jobs.Trigger.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Trigger.t/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:t/0"},{"type":"type","title":"Lightning.Jobs.Trigger.trigger_type/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:trigger_type/0"},{"type":"module","title":"Lightning.Projects","doc":"The Projects context.","ref":"Lightning.Projects.html"},{"type":"function","title":"Lightning.Projects.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Projects.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Projects.change_project/2","doc":"Returns an %Ecto.Changeset{} for tracking project changes. Examples iex> change_project ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#change_project/2"},{"type":"function","title":"Lightning.Projects.create_project/1","doc":"Creates a project. Examples iex> create_project ( %{ field : value } ) { :ok , % Project { } } iex> create_project ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#create_project/1"},{"type":"function","title":"Lightning.Projects.delete_project/1","doc":"Deletes a project and its related data, including workflows, work orders, runs, jobs, attempts, triggers, project users, project credentials, and dataclips Examples iex> delete_project ( project ) { :ok , % Project { } } iex> delete_project ( project ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#delete_project/1"},{"type":"function","title":"Lightning.Projects.export_project/2","doc":"Exports a project as yaml. Examples iex> export_project ( :yaml , project_id ) { :ok , string }","ref":"Lightning.Projects.html#export_project/2"},{"type":"function","title":"Lightning.Projects.get_project/1","doc":"","ref":"Lightning.Projects.html#get_project/1"},{"type":"function","title":"Lightning.Projects.get_project!/1","doc":"Gets a single project. Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project!/1"},{"type":"function","title":"Lightning.Projects.get_project_credential/2","doc":"","ref":"Lightning.Projects.html#get_project_credential/2"},{"type":"function","title":"Lightning.Projects.get_project_user/1","doc":"","ref":"Lightning.Projects.html#get_project_user/1"},{"type":"function","title":"Lightning.Projects.get_project_user/2","doc":"","ref":"Lightning.Projects.html#get_project_user/2"},{"type":"function","title":"Lightning.Projects.get_project_user!/1","doc":"Gets a single project_user. Raises Ecto.NoResultsError if the ProjectUser does not exist. Examples iex> get_project_user! ( 123 ) % ProjectUser { } iex> get_project_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_user!/1"},{"type":"function","title":"Lightning.Projects.get_project_user_role/2","doc":"Returns the role of a user in a project. Possible roles are :admin, :viewer, :editor, and :owner Examples iex> get_project_user_role ( user , project ) :admin iex> get_project_user_role ( user , project ) :viewer iex> get_project_user_role ( user , project ) :editor iex> get_project_user_role ( user , project ) :owner","ref":"Lightning.Projects.html#get_project_user_role/2"},{"type":"function","title":"Lightning.Projects.get_project_with_users!/1","doc":"Gets a single project with it's members via project_users . Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_with_users!/1"},{"type":"function","title":"Lightning.Projects.get_projects_for_user/1","doc":"","ref":"Lightning.Projects.html#get_projects_for_user/1"},{"type":"function","title":"Lightning.Projects.is_member_of?/2","doc":"","ref":"Lightning.Projects.html#is_member_of?/2"},{"type":"function","title":"Lightning.Projects.list_project_credentials/1","doc":"","ref":"Lightning.Projects.html#list_project_credentials/1"},{"type":"function","title":"Lightning.Projects.list_projects/0","doc":"Returns the list of projects. Examples iex> list_projects ( ) [ % Project { } , ... ]","ref":"Lightning.Projects.html#list_projects/0"},{"type":"function","title":"Lightning.Projects.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find projects that are ready for permanent deletion and purge them.","ref":"Lightning.Projects.html#perform/1"},{"type":"function","title":"Lightning.Projects.project_attempt_run_query/1","doc":"","ref":"Lightning.Projects.html#project_attempt_run_query/1"},{"type":"function","title":"Lightning.Projects.project_attempts_query/1","doc":"","ref":"Lightning.Projects.html#project_attempts_query/1"},{"type":"function","title":"Lightning.Projects.project_credentials_query/1","doc":"","ref":"Lightning.Projects.html#project_credentials_query/1"},{"type":"function","title":"Lightning.Projects.project_dataclip_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_dataclip_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_dataclips_query/1","doc":"","ref":"Lightning.Projects.html#project_dataclips_query/1"},{"type":"function","title":"Lightning.Projects.project_jobs_query/1","doc":"","ref":"Lightning.Projects.html#project_jobs_query/1"},{"type":"function","title":"Lightning.Projects.project_run_invocation_reasons/1","doc":"","ref":"Lightning.Projects.html#project_run_invocation_reasons/1"},{"type":"function","title":"Lightning.Projects.project_runs_query/1","doc":"","ref":"Lightning.Projects.html#project_runs_query/1"},{"type":"function","title":"Lightning.Projects.project_trigger_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_trigger_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_triggers_query/1","doc":"","ref":"Lightning.Projects.html#project_triggers_query/1"},{"type":"function","title":"Lightning.Projects.project_user_role_query/2","doc":"","ref":"Lightning.Projects.html#project_user_role_query/2"},{"type":"function","title":"Lightning.Projects.project_users_query/1","doc":"","ref":"Lightning.Projects.html#project_users_query/1"},{"type":"function","title":"Lightning.Projects.project_workflows_query/1","doc":"","ref":"Lightning.Projects.html#project_workflows_query/1"},{"type":"function","title":"Lightning.Projects.project_workorders_query/1","doc":"","ref":"Lightning.Projects.html#project_workorders_query/1"},{"type":"function","title":"Lightning.Projects.projects_for_user_query/1","doc":"","ref":"Lightning.Projects.html#projects_for_user_query/1"},{"type":"function","title":"Lightning.Projects.schedule_project_deletion/1","doc":"Given a project, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic project purge cronjob will never run. (Note that subsequent logins will be blocked for projects pending deletion.)","ref":"Lightning.Projects.html#schedule_project_deletion/1"},{"type":"function","title":"Lightning.Projects.select_first_project_for_user/1","doc":"","ref":"Lightning.Projects.html#select_first_project_for_user/1"},{"type":"function","title":"Lightning.Projects.update_project/2","doc":"Updates a project. Examples iex> update_project ( project , %{ field : new_value } ) { :ok , % Project { } } iex> update_project ( project , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project/2"},{"type":"function","title":"Lightning.Projects.update_project_user/2","doc":"Updates a project user. Examples iex> update_project_user ( project_user , %{ field : new_value } ) { :ok , % ProjectUser { } } iex> update_project_user ( projectUser , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project_user/2"},{"type":"function","title":"Lightning.Projects.url_safe_project_name/1","doc":"","ref":"Lightning.Projects.html#url_safe_project_name/1"},{"type":"function","title":"Lightning.Projects.validate_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the project scheduled_deletion. Examples iex> validate_for_deletion ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#validate_for_deletion/2"},{"type":"module","title":"Lightning.Projects.Project","doc":"Project model","ref":"Lightning.Projects.Project.html"},{"type":"function","title":"Lightning.Projects.Project.deletion_changeset/2","doc":"Changeset to validate a project deletion request, the user must enter the projects name to confirm.","ref":"Lightning.Projects.Project.html#deletion_changeset/2"},{"type":"function","title":"Lightning.Projects.Project.validate/1","doc":"","ref":"Lightning.Projects.Project.html#validate/1"},{"type":"type","title":"Lightning.Projects.Project.t/0","doc":"","ref":"Lightning.Projects.Project.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectCredential","doc":"Join table to assign credentials to a project","ref":"Lightning.Projects.ProjectCredential.html"},{"type":"type","title":"Lightning.Projects.ProjectCredential.t/0","doc":"","ref":"Lightning.Projects.ProjectCredential.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser","doc":"Join table to assign users to a project","ref":"Lightning.Projects.ProjectUser.html"},{"type":"type","title":"Lightning.Projects.ProjectUser.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.DigestEnum","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.DigestEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.RolesEnum","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.RolesEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.Provisioner","doc":"Provides functions for importing projects. This module is used by the provisioning HTTP API. When providing a project to import, all records must have an id field. It's up to the caller to ensure that the id is unique and generated ahead of time in the case of new records.","ref":"Lightning.Projects.Provisioner.html"},{"type":"function","title":"Lightning.Projects.Provisioner.import_document/3","doc":"Import a project.","ref":"Lightning.Projects.Provisioner.html#import_document/3"},{"type":"function","title":"Lightning.Projects.Provisioner.load_project/1","doc":"Load a project by ID, including all workflows and their associated jobs, triggers and edges. Returns nil if the project does not exist.","ref":"Lightning.Projects.Provisioner.html#load_project/1"},{"type":"function","title":"Lightning.Projects.Provisioner.parse_document/2","doc":"","ref":"Lightning.Projects.Provisioner.html#parse_document/2"},{"type":"function","title":"Lightning.Projects.Provisioner.validate_extraneous_params/1","doc":"Validate that there are no extraneous parameters in the changeset. For all params in the changeset, ensure that the param is in the list of known fields in the schema.","ref":"Lightning.Projects.Provisioner.html#validate_extraneous_params/1"},{"type":"module","title":"Lightning.Runtime.ChildProcess","doc":"Provides an interface between a RunSpec and the shell. Internally it calls node , and more specifically the OpenFn core CLI.","ref":"Lightning.Runtime.ChildProcess.html"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_command/1","doc":"Builds up a string for shell execution based on the RunSpec","ref":"Lightning.Runtime.ChildProcess.html#build_command/1"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_env/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#build_env/2"},{"type":"function","title":"Lightning.Runtime.ChildProcess.run/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#run/2"},{"type":"behaviour","title":"Lightning.Runtime.Handler","doc":"A strategy for executing things via ChildProcess. This module handles the dirty bits, setting up processes and coordinating results (and logs) as they arrive. Since it is a macro, see Lightning.Pipeline.Runner.Handler for a usage example.","ref":"Lightning.Runtime.Handler.html"},{"type":"function","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.log_callback/3","doc":"","ref":"Lightning.Runtime.Handler.html#c:log_callback/3"},{"type":"function","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_finish/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_finish/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_log_emit/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_log_emit/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_start/1","doc":"","ref":"Lightning.Runtime.Handler.html#on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.on_start/1","doc":"Called with context, if any - when the Run has been started.","ref":"Lightning.Runtime.Handler.html#c:on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.start/2","doc":"The entrypoint for executing a run.","ref":"Lightning.Runtime.Handler.html#c:start/2"},{"type":"type","title":"Lightning.Runtime.Handler.t/0","doc":"","ref":"Lightning.Runtime.Handler.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.LogAgent","doc":"Agent facility to consume STDOUT/STDERR byte by byte. Since it works on a byte by byte basis, you will need to perform line-splitting yourself. Usage: { :ok , log } = LogAgent . start_link ( ) "foo" = LogAgent . process_chunk ( log , { :stdout , "foo" } ) "foobar" = LogAgent . process_chunk ( log , { :stdout , "bar" } )","ref":"Lightning.Runtime.LogAgent.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Runtime.LogAgent.html#child_spec/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.start_link/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#start_link/1"},{"type":"type","title":"Lightning.Runtime.LogAgent.logline/0","doc":"","ref":"Lightning.Runtime.LogAgent.html#t:logline/0"},{"type":"module","title":"Lightning.Runtime.LogAgent.StringBuffer","doc":"Internal datastructure to hold and process new bytes for a list of characters. By checking the if the buffer is a complete grapheme, emitting the buffer once valid and returning nil otherwise. In the case of emojis and other language character sets, a character (in UTF-8) can be between 1-4 bytes - when streaming logs for example it's quite easy to receive less than the whole character which can result in crashes or corrupt text.","ref":"Lightning.Runtime.LogAgent.StringBuffer.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.new/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#new/0"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.reduce_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#reduce_chunk/2"},{"type":"type","title":"Lightning.Runtime.LogAgent.StringBuffer.t/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.Result","doc":"Data structure used to represent the result of a Run executed by Lightning.Runtime.ChildProcess .","ref":"Lightning.Runtime.Result.html"},{"type":"function","title":"Lightning.Runtime.Result.new/1","doc":"","ref":"Lightning.Runtime.Result.html#new/1"},{"type":"type","title":"Lightning.Runtime.Result.t/0","doc":"","ref":"Lightning.Runtime.Result.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.RunSpec","doc":"A struct containing all the parameters required to execute a Job.","ref":"Lightning.Runtime.RunSpec.html"},{"type":"function","title":"Lightning.Runtime.RunSpec.new/1","doc":"","ref":"Lightning.Runtime.RunSpec.html#new/1"},{"type":"type","title":"Lightning.Runtime.RunSpec.t/0","doc":"","ref":"Lightning.Runtime.RunSpec.html#t:t/0"},{"type":"exception","title":"Lightning.MetadataService.Error","doc":"","ref":"Lightning.MetadataService.Error.html"},{"type":"function","title":"Lightning.MetadataService.Error.message/1","doc":"Callback implementation for Exception.message/1 .","ref":"Lightning.MetadataService.Error.html#message/1"},{"type":"function","title":"Lightning.MetadataService.Error.new/1","doc":"","ref":"Lightning.MetadataService.Error.html#new/1"},{"type":"type","title":"Lightning.MetadataService.Error.t/0","doc":"","ref":"Lightning.MetadataService.Error.html#t:t/0"},{"type":"extras","title":"Lightning","doc":"OpenFn/Lightning is a fully open source, workflow automation platform that's used to automate critical business processes and integrate information systems. From last-mile services to national-level reporting, it boosts efficiency & effectiveness while enabling secure, stable, scalable interoperability at all levels. Read more about Lightning on OpenFn/Docs . Lightning is the latest "v2" of OpenFn, the Digital Public Good for workflow automation . Use Lightning to visually build, execute and manage workflows. The latest version of the OpenFn technology - first launched in 2014, now tried and tested by NGOs in over 40 countries Fully open source (no premium features or community edition, you get the same product whether using SaaS or self-hosted) Recognised as a Digital Public Good by the DPGA and a Global Good for Health by Digital Square","ref":"readme.html"},{"type":"extras","title":"Lightning - Contents","doc":"Demo Sign up for the BETA Features Getting started Run Lightning via Docker Deploy Lightning on Docker or Kubernetes Run Lightning on your local machine (contributors) Troubleshooting Contribute Project Configuration Quickstart Generate the documentation Security and standards Server specs for self-hosting Questions or feedback?","ref":"readme.html#contents"},{"type":"extras","title":"Lightning - Demo","doc":"Watch a quick demo here: https://www.youtube.com/watch?v=BNaxlHAWb5I Explore our demo app* with username: demo@openfn.org , password: welcome123 , or read through the features section to view screenshots of the app. *Note that the demo app refreshes daily, so do not configure workflows you want to save.","ref":"readme.html#demo"},{"type":"extras","title":"Lightning - Register for a Beta account","doc":"Register for a Beta account at app.openfn.org and go through the quick-start guide to get familiar with the app.","ref":"readme.html#register-for-a-beta-account"},{"type":"extras","title":"Lightning - Features","doc":"Build Plan and build workflows using Lightning's visual interface to quickly define when, where and what you want your automation to do. Use our CLI to quickly build, edit and deploy projects from the comfort of your own code editor. Monitor Monitor all workflow activity in one place. Filter and search runs to identify issues that need addressing and follow how a specific request has been processed Configure alerts to be notified on run failures Receive a project digest for a daily/weekly/monthly summary of your project activity Manage Manage users and access by project Roles and permissions Authorization is a central part of Lightning. As such, users are given different roles which determine what level of access they have for resources in the application. For more details about roles and permissions in Lightning, please refer to our documentation . Roadmap View our public roadmap here .","ref":"readme.html#features"},{"type":"extras","title":"Lightning - Getting Started","doc":"If you only want to RUN Lightning on your own server, we recommend using Docker. If you want to DEPLOY Lightning, we recommend Docker builds and Kubernetes. If you want to CONTRIBUTE to the project, we recommend running Lightning on your local machine .","ref":"readme.html#getting-started"},{"type":"extras","title":"Lightning - Run via Docker","doc":"Install the latest version of Docker Clone this repo using git Copy the .env.example file to .env Run docker compose run --rm web mix ecto.migrate By default the application will be running at localhost:4000 . You can then rebuild and run with docker compose build and docker compose up . See "Problems with Docker" for additional troubleshooting help. Note that you can also create your own docker-compose.yml file, configuring a postgres database and using a pre-built image from Dockerhub.","ref":"readme.html#run-via-docker"},{"type":"extras","title":"Lightning - Deploy on external infrastructure","doc":"See Deployment for more detailed information.","ref":"readme.html#deploy-on-external-infrastructure"},{"type":"extras","title":"Lightning - Run Lightning locally","doc":"Clone the repo and optionally set ENVs git clone git@github.com:OpenFn/Lightning.git # or from YOUR fork! cd Lightning cp .env.example .env # and adjust as necessary! Take note of database names and ports in particular—they've got to match across your Postgres setup and your ENVs. You can run lightning without any ENVs assuming a vanilla postgres setup (see below), but you may want to make adjustments. Database Setup If you're already using Postgres locally, create a new database called lightning_dev , for example. If you'd rather use Docker to set up a Postgres DB, create a new volume and image: docker volume create lightning-postgres-data docker create \\ --name lightning-postgres \\ --mount source=lightning-postgres-data,target=/var/lib/postgresql/data \\ --publish 5432:5432 \\ -e POSTGRES_PASSWORD=postgres \\ postgres:15.3-alpine docker start lightning-postgres Elixir & Ecto Setup We use asdf to configure our local environments. Included in the repo is a .tool-versions file that is read by asdf in order to dynamically make the specified versions of Elixir and Erlang available. You'll need asdf plugins for Erlang , NodeJs Elixir and k6 . asdf install # Install language versions mix local.hex mix deps.get mix local.rebar --force mix ecto.create # Create a development database in Postgres mix ecto.migrate [[ $(uname -m) == 'arm64' ]] && mix compile.rambo # Force compile rambo if on M1 mix lightning.install_runtime mix lightning.install_schemas npm install --prefix assets Run the app Lightning is a web app. To run it in interactive Elixir mode, start the development server by running with your environment variables by running: iex -S mix phx.server or if you have set up custom environment variables, run: env $(cat .env | grep -v "#" | xargs ) iex -S mix phx.server Once the server has started, head to localhost:4000 in your browser. Run the tests Before the first time running the tests, you need a test database setup. MIX_ENV=test mix ecto.create And then after that run the tests using: MIX_ENV=test mix test We also have test.watch installed which can be used to rerun the tests on file changes.","ref":"readme.html#run-lightning-locally"},{"type":"extras","title":"Lightning - Benchmarking","doc":"We are using k6 to benchmark Lightning. Under benchmarking folder you can find a script for benchmarking Webhook Workflows. See Benchmarking for more detailed information.","ref":"readme.html#benchmarking"},{"type":"extras","title":"Lightning - Troubleshooting","doc":"Trouble with environment variables For troubleshooting custom environment variable configuration it's important to know how an Elixir app loads and modifies configuration. The order is as follows: Stuff in config.exs is loaded. That is then modified (think: overwritten ) by stuff your ENV-specific config: dev.exs , prod.exs or test.exs . That is then modified by runtime.exs which is where you are allowed to use System.env() Finally init/2 (if present in a child application) gets called (which takes the config which has been set in steps 1-3) when that child application is started during the parent app startup defined in application.ex . Problems with Postgres If you're having connecting issues with Postgres, check the database section of your .env to ensure the DB url is correctly set for your environment — note that composing a DB url out of other, earlier declared variables, does not work while using xargs . Problems with Debian If you're getting this error on debian == > earmark_parser Compiling 1 file ( . yrl ) / usr / lib / erlang / lib / parsetools - 2.3 . 1 / include / yeccpre . hrl : no such file or directory could not compile dependency :earmark_parser , "mix compile" failed . You can recompile this dependency with "mix deps.compile earmark_parser" , update it with "mix deps.update earmark_parser" or clean it with "mix deps.clean earmark_parser" You need to install erlang development environment sudo apt install erlang-dev refer to this issue Problems with Docker Versions The build may not work on old versions of Docker and Docker compose. It has been tested against: Docker version 20.10 . 17 , build 100 c701 Docker Compose version v2 . 6.0 Problems with rambo When running mix compile.rambo on Apple Silicon (an Apple M1/M2, macarm , aarch64-apple-darwin ) and encountering the following error: ** (RuntimeError) Rambo does not ship with binaries for your environment. aarch64 - apple - darwin22 . 3.0 detected Install the Rust compiler so a binary can be prepared for you . lib / mix / tasks / compile . rambo . ex : 89 : Mix.Tasks.Compile.Rambo . compile! / 0 lib / mix / tasks / compile . rambo . ex : 51 : Mix.Tasks.Compile.Rambo . run / 1 ( mix 1.14 . 2 ) lib / mix / task . ex : 421 : anonymous fn / 3 in Mix.Task . run_task / 4 ( mix 1.14 . 2 ) lib / mix / cli . ex : 84 : Mix.CLI . run_task / 2 You can resolve this error by installing the Rust compiler using Homebrew. Run the following command in your terminal: brew install rust If you have already compiled Rambo explicitly via mix compile.rambo , and you are still seeing the following error: sh : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : No such file or directory sh : line 0 : exec : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : cannot execute : No such file or directory You can try renaming deps/rambo/priv/rambo-mac to deps/rambo/priv/rambo . If neither of the approaches above work, please raise an issue. Starting from scratch If you're actively working with docker, you start experiencing issues, and you would like to start from scratch you can clean up everything and start over like this: # To remove any ignored files and reset your .env to it's example git clean -fdx && cp .env.example .env # You can skip the line below if you want to keep your database docker compose down --rmi all --volumes docker compose build --no-cache web && \\ docker compose create --force-recreate docker compose run --rm web mix ecto.migrate docker compose up","ref":"readme.html#troubleshooting"},{"type":"extras","title":"Lightning - Project Configuration Quickstart","doc":"For help getting started with your OpenFn/Lightning Project and Workflows configuration, check out the Lightning Quickstart Guidance on OpenFn/Docs.","ref":"readme.html#project-configuration-quickstart"},{"type":"extras","title":"Lightning - Contribute to this project","doc":"First, thanks for being here! You're contributing to a digital public good that will always be free and open source and aimed at serving innovative NGOs, governments, and social impact organizations the world over! You rock. ❤️ FYI, Lightning is built in Elixir , harnessing the Phoenix Framework . Currently, the only unbundled dependency is a PostgreSQL database. If you'd like to contribute to this projects, follow the steps below: Assign yourself to an issue Read through the existing issues , assign yourself to the issue you have chosen. Leave a comment on the issue to let us know you'll be working on it, and if you have any questions of clarifications that would help you get started ask them there - we will get back to you as soon as possible. If there isn't already an issue for the feature you would like to contribute, please start a discussion in our community forum . Open a pull request Clone the Lightning repository, then fork it . Run through setting up your environment and make your changes. Make sure you have written your tests and updated /CHANGELOG.md (in the 'Unreleased' section, add a short description of the changes you are making, along with a link to your issue). Open a draft pull request by clicking "Contribute > Open Pull Request" from your forked repository. Fill out the pull request template (this will be added automatically for you), then make sure to self-review your code and go through the 'Review checklist'. Don't worry about the QA checkbox, our product manager Amber will tick that once she has reviewed your PR. You can leave any notes for the reviewer in a comment. Once you're ready to submit a pull request, you can mark your draft PR as 'Ready for review' and assign @stuartc or @taylordowns2000.","ref":"readme.html#contribute-to-this-project"},{"type":"extras","title":"Lightning - Generating Documentation","doc":"You can generate the HTML and EPUB documentation locally using: mix docs and opening doc/index.html in your browser.","ref":"readme.html#generating-documentation"},{"type":"extras","title":"Lightning - Security and Standards","doc":"We use a host of common Elixir static analysis tools to help us avoid common pitfalls and make sure we keep everything clean and consistent. In addition to our test suite, you can run the following commands: mix format --check-formatted Code formatting checker, run again without the --check-formatted flag to have your code automatically changed. mix dialyzer Static analysis for type mismatches and other common warnings. See dialyxir . mix credo Static analysis for consistency, and coding standards. See Credo . mix sobelow Check for commonly known security exploits. See Sobelow . MIX_ENV=test mix coveralls Test coverage reporter. This command also runs the test suite, and can be used in place of mix test when checking everything before pushing your code. See excoveralls . For convenience there is a verify mix task that runs all of the above and defaults the MIX_ENV to test . For more guidance on security best practices for workflow automation implementations, check out OpenFn Docs: docs.openfn.org/documentation/getting-started/security","ref":"readme.html#security-and-standards"},{"type":"extras","title":"Lightning - Server Specs for Self-Hosting","doc":"For recommend server specifications for self-hosting of Lightning, check out this Community topic: community.openfn.org/t/specs-for-self-hosting-lightning/292","ref":"readme.html#server-specs-for-self-hosting"},{"type":"extras","title":"Lightning - Support","doc":"If you have any questions, feedback, or issues, please: Post on the OpenFn Community at community.openfn.org Open an issue directly on this Github Repo: github.com/OpenFn/Lightning/issues","ref":"readme.html#support"},{"type":"extras","title":"Deployment","doc":"","ref":"deployment.html"},{"type":"extras","title":"Deployment - Encryption","doc":"Lightning enforces encryption at rest for Credentials, for which an encryption key must be provided when running in production. The key is expected to be a randomized set of bytes, 32 long; and Base64 encoded when setting the environment variable. There is a mix task that can generate keys in the correct shape for use as an environment variable: mix lightning.gen_encryption_key 0bJ9w+hn4ebQrsCaWXuA9JY49fP9kbHmywGd5K7k+/s= Copy your key (NOT THIS ONE) and set it as PRIMARY_ENCRYPTION_KEY in your environment.","ref":"deployment.html#encryption"},{"type":"extras","title":"Deployment - Environment Variables","doc":"Note that for secure deployments, it's recommended to use a combination of secrets and configMaps to generate secure environment variables. ADAPTORS_PATH - where you store your locally installed adaptors DISABLE_DB_SSL - in production the use of an SSL conntection to Postgres is required by default, setting this to "true" allows unencrypted connections to the database. This is strongly discouraged in real production environment. K8S_HEADLESS_SERVICE - this environment variable is automatically set if you're running on GKE and it is used to establish an Erlang node cluster. Note that if you're not using Kubernetes, the "gossip" strategy is used for establish clusters. LISTEN_ADDRESS " - the address the web server should bind to, defaults to 127.0.0.1 to block access from other machines. LOG_LEVEL - how noisy you want the logs to be (e.g. debug , info ) MAX_RUN_DURATION - the maximum time (in milliseconds) that jobs are allowed to run (keep this below your termination_grace_period if using kubernetes) MIX_ENV - your mix env, likely prod for deployment NODE_ENV - node env, likely production for deployment ORIGINS - the allowed origins for web traffic to the backend PORT - the port your Phoenix app runs on PRIMARY_ENCRYPTION_KEY - a base64 encoded 32 character long string. See Encryption . SCHEMAS_PATH - path to the credential schemas that provide forms for different adaptors SECRET_KEY_BASE - a secret key used as a base to generate secrets for encrypting and signing data. SENTRY_DSN - if using Sentry for error monitoring, your DSN URL_HOST - the host, used for writing urls (e.g., demo.openfn.org ) URL_PORT - the port, usually 443 for production URL_SCHEME - the scheme for writing urls, (e.g., https ) Google Using your Google Cloud account, provision a new OAuth 2.0 Client with the 'Web application' type. Set the callback url to: https://<ENDPOINT DOMAIN>/authenticate/callback . Replacing ENDPOINT DOMAIN with the host name of your instance. Once the client has been created, get/download the OAuth client JSON and set the following environment variables: GOOGLE_CLIENT_ID - Which is client_id from the client details. GOOGLE_CLIENT_SECRET - client_secret from the client details.","ref":"deployment.html#environment-variables"},{"type":"extras","title":"Benchmarking","doc":"Execute the following steps to run a benchmark on Lightning: Make sure you have k6 installed locally. If you're using asdf you can run asdf install in the project root. Spin up your Lightning local instance Run the demo setup script: mix run --no-start priv/repo/demo.exs The webhookURL is already set to default to the webhook created in the demo data In another terminal (do not stop the Lightning server) run the benchmarking/script.js file using the following command k6 run benchmarking/script.js If the script exits succesfully, this means the app met the defined performance thresholds. To collect the benchmarking data in a CSV file, run the previous command with the --out filename option. k6 run --out csv=test_results.csv benchmarking/script.js See results output for other available output formats.","ref":"benchmarking.html"},{"type":"extras","title":"Provisioning","doc":"Lightning offers the ability to configure projects via the HTTP API. By providing a JSON document with the desired configuration, the project can be configured to your liking.","ref":"provisioning.html"},{"type":"extras","title":"Provisioning - Using the API","doc":"The API is available at /api/provision , and expects an application/json Content-Type. Authentication The API requires a valid auth token to be provided in the Authorization header. Example Request curl -X POST \\ -d @project.json \\ -H "Authorization: Bearer $TOKEN" \\ -H "Content-Type: application/json" \\ $ENDPOINT/api/provision","ref":"provisioning.html#using-the-api"},{"type":"extras","title":"Provisioning - Document Structure","doc":"The provisioning document is a JSON document with the project at the root. All entities must have an id field, which is a UUIDv4 string. In the case of new entities, this must be generated by the client. The API is idempotent, and the distinction between creating and updating is determined by the presence of the id field. { "id": "<<project-id>>", "name": "<<project-name>>", "workflows": [ { "id": "<<workflow-id>>", "name": "<<workflow-name>>", "jobs": [ { "id": "<<job-id>>", "name": "<<job-name>>", "body": "<<job-body>>", "adaptor": "<<adaptor-name>>", "enabled": true } // ... more jobs ], "triggers": [ { "id": "<<trigger-id>>", "name": "<<trigger-name>>", "type": "webhook" } // ... more triggers ], "edges": [ { "id": "<<edge-id>>", "source_trigger_id": "<<trigger-id>>", "target_job_id": "<<job-id>>" } // ... more edges ] } // ... more workflows ] }","ref":"provisioning.html#document-structure"},{"type":"extras","title":"Provisioning - API Behaviour","doc":"The API expects all existing entities to be provided in the provisioning document. If the document provided is out of date (e.g. a new job was added on the server), a new reference document should be fetched and the changes applied to it. Deleting Entities Entities can be deleted by setting the disabled key to true . Example: { "id": "<<project-id>>", "workflows": [ { "id": "<<workflow-id>>", "jobs": [ { "id": "<<job-id>>", "delete": true // <== delete this job } ] } ] }","ref":"provisioning.html#api-behaviour"},{"type":"extras","title":"Provisioning - Relationship with Projects as Code","doc":"The Projects as Code spec is a superset of the provisioning API. Projects as Code allows for the user to specify a key for each entity, which makes it easier to manage the project in the future. For example: name: my-project workflows: workflow-one: jobs: job-one: body: | console.log("Hello World"); adaptor: '@openfn/language-common' enabled: true triggers: trigger-one: type: webhook edges: - source_trigger: trigger-one target_job: job-one The above YAML document illustrates the use of keys being used to identify entities. Allowing the user to provision the same project to multiple environments. The API is unaware of 'keys', and expects IDs to be provided by the client. In order to convert the above YAML document to a provisioning document, the CLI uses a local state file (if available) to map the keys to UUIDs. Using the example above a state file might look like this: { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": {}, "triggers": {} } } } The state file shows that the project and workflow already exist, but the job, trigger and edge do not. In order to create these new entities, IDs will be applied them. On a successful application of the provisioning document, the state file will be updated to reflect the new IDs and entities. { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": { "job-one": { "id": "18ed71de-caf8-4822-aefc-5b19351f4016" } }, "triggers": { "trigger-one": { "id": "e0b9f357-9cf9-4206-9924-4d5674aad830" } }, "edges": [ { "id": "c239d994-6662-4637-90f8-0293c924b461", "source_trigger_id": "e0b9f357-9cf9-4206-9924-4d5674aad830", "target_job_id": "18ed71de-caf8-4822-aefc-5b19351f4016" } ] } } }","ref":"provisioning.html#relationship-with-projects-as-code"},{"type":"extras","title":"Changelog","doc":"All notable changes to this project will be documented in this file. The format is based on Keep a Changelog , and this project adheres to Semantic Versioning .","ref":"changelog.html"},{"type":"extras","title":"Changelog - [Unreleased]","doc":"Added Changed Moved to Elixir 1.15 and Erlang 26.0.2 to sort our an annoying ElixirLS issue that was slowing down our engineers. Update Debian base to use bookworm (Debian 12) for our Docker images Change new credential modal to take up less space on the screen #931 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","ref":"changelog.html#unreleased"},{"type":"extras","title":"Changelog - [v0.7.0-pre5] - 2023-07-28","doc":"Added Changed Unless otherwise specified, only show workorders with activity in last 14 days #968 Fixed","ref":"changelog.html#v0-7-0-pre5-2023-07-28"},{"type":"extras","title":"Changelog - [v0.7.0-pre4] - 2023-07-27","doc":"Added Changed Don't add cast fragments if the search_term is nil #968 Fixed","ref":"changelog.html#v0-7-0-pre4-2023-07-27"},{"type":"extras","title":"Changelog - [v0.7.0-pre3] - 2023-07-26","doc":"Added Changed Fixed Fixed an issue with newly created edges that prevented downstream jobs 977","ref":"changelog.html#v0-7-0-pre3-2023-07-26"},{"type":"extras","title":"Changelog - [v0.7.0-pre2] - 2023-07-26","doc":"Note that this is a pre-release with a couple of known bugs that are tracked in the Nodes and Edges epic . Added Added ability for a user to enable MFA on their account; using 2FA apps like Authy, Google Authenticator etc #890 Write/run sql script to convert triggers #875 Export projects as .yaml via UI #249 Changed In v0.7.0 we change the underlying workflow building and execution infrastructure to align with a standard "nodes and edges" design for directed acyclic graphs (DAGs). Make sure to run the migrations! 793 Fixed Propagate url pushState/changes to Workflow Diagram selection #944 Fix issue when deleting nodes from the workflow editor #830 Fix issue when clicking a trigger on a new/unsaved workflow #954","ref":"changelog.html#v0-7-0-pre2-2023-07-26"},{"type":"extras","title":"Changelog - [0.6.7] - 2023-07-13","doc":"Added Add feature to bulk rerun work orders from a specific step in their workflow; e.g., "rerun these 50 work orders, starting each at step 4." #906 Changed Fixed Oban exception: "value too long" when log lines are longer than 255 chars #929","ref":"changelog.html#0-6-7-2023-07-13"},{"type":"extras","title":"Changelog - [0.6.6] - 2023-06-30","doc":"Added Add public API token to the demo site setup script Check and renew OAuth credentials when running a job #646 Fixed Remove google sheets from adaptors list until supporting oauth flow #792 Remove duplicate google sheets adaptor display on credential type picklist #663 Fix demo setup script for calling from outside the app on Kubernetes deployments #917","ref":"changelog.html#0-6-6-2023-06-30"},{"type":"extras","title":"Changelog - [0.6.5] - 2023-06-22","doc":"Added Ability to rerun workorders from start by selecting one of more of them from the History page and clicking the "Rerun" button. #659 Fixed Example runs for demo incorrect #856","ref":"changelog.html#0-6-5-2023-06-22"},{"type":"extras","title":"Changelog - [0.6.3] - 2023-06-15","doc":"Fixed Prevent saving null log lines to the database, fix issue with run display #866","ref":"changelog.html#0-6-3-2023-06-15"},{"type":"extras","title":"Changelog - [0.6.2] - 2023-06-09","doc":"Fixed Fixed viewer permissions for delete workflow Fixed bug with workflow cards #859","ref":"changelog.html#0-6-2-2023-06-09"},{"type":"extras","title":"Changelog - [0.6.1] - 2023-06-08","doc":"Fixed Fixed bug with run logs #864 Correctly stagger demo runs to maintain order #856 Remove Timex use from SetupUtils in favor of DateTime to fix issue when calling it in escript.","ref":"changelog.html#0-6-1-2023-06-08"},{"type":"extras","title":"Changelog - [0.6.0]- 2023-04-12","doc":"Added Create sample runs when generating sample workflow #821 Added a provisioning api for creating and updating projects and their workflows See: PROVISIONING.md #641 Add ability for a superuser to schedule deletion, cancel deletion, and delete projects #757 Add ability for a project owner to schedule deletion, cancel deletion, and delete projects #746 Changed Ability to store run log lines as rows in a separate table #514 Fixed Incorrect project digest queries #768 ] Fix issue when purging deleted users #747 Generate a random name for Workflows when creating one via the UI. #828 Handle error when deleting a job with runs. #814","ref":"changelog.html#0-6-0-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.2]","doc":"Added Add workflow_edges table in preparation for new workflow editor implementation #794 Stamped credential_id on run directly for easier auditing of the history interface. Admins can now see which credential was used to run a run. #800 Better errors when using magic functions: "no magic yet" and "check credential" #812 Changed The delete-project function now delete all associated activities #759 Fixed","ref":"changelog.html#0-5-2"},{"type":"extras","title":"Changelog - [0.5.1] - 2023-04-12","doc":"Added Added ability to create and revoke personal API tokens #147 Add last-used at to API tokens #722 Improved "save" for job builder; users can now press Ctrl + S or ⌘ + S to save new or updated jobs job panel will not close. (Click elsewhere in the canvas or click the "Close" button to close.) #568 Add filtered search params to the history page URL #660 Changed The secret scrubber now ignores booleans 690 Fixed The secret scrubber now properly handles integer secrets from credentials 690 Updated describe-package dependency, fixing sparkles in adaptor-docs 657 Clicks on the workflow canvas were not lining up with the nodes users clicked on; they are now 733 Job panel behaves better when collapsed 774","ref":"changelog.html#0-5-1-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.0] - 2023-04-03","doc":"Added Magic functions that fetch real metadata from connected systems via credentials and suggest completions in the job builder (e.g., pressing control-space when setting the orgUnit attribute for a DHIS2 create operation will pull the actual list of orgUnits with human readable labels and fill in their orgUnit codes upon enter.) 670 A "metadata explorer" to browse actual system metadata for connected instances. 658 Resizable job builder panel for the main canvas/workflow view. 681 Changed Display timezone for cron schedule—it is always UTC. #716 Instance administrators can now configure the interval between when a project owner or user requests deletion and when these records are purged from the database. It defaults to 7, but by providing a PURGE_DELETED_AFTER_DAYS environment variable the grace period can be altered. Note that setting this variable to 0 will make automatic purging never occur but will still make "deleted" projects and users unavailable. This has been requested by certain organizations that must retain audit logs in a Lightning instance. 758 Fixed Locked CLI version to @openfn/cli@0.0.35 . #761","ref":"changelog.html#0-5-0-2023-04-03"},{"type":"extras","title":"Changelog - [0.4.8] - 2023-03-29","doc":"Added Added a test harness for monitoring critical parts of the app using Telemetry #654 Changed Set log level to info for runs. Most of the debug logging is useful for the CLI, but not for Lightning. In the future the log level will be configurable at instance > project > job level by the superuser and any project admin . Renamed license file so that automagic github icon is less confusing Fixed Broken links in failure alert email #732 Registration Submission on app.openfn.org shows internal server error in browser #686 Run the correct runtime install mix task in Dockerfile-dev #541 Users not disabled when scheduled for deletion #719","ref":"changelog.html#0-4-8-2023-03-29"},{"type":"extras","title":"Changelog - [0.4.6] - 2023-03-23","doc":"Added Implement roles and permissions across entire app #645 Fix webhook URL ( https://<<HOST_URL>>/i/cae544ab-03dc-4ccc-a09c-fb4edb255d7a ) for the OpenHIE demo workflow 448 Phoenix Storybook for improved component development Load test for webhook endpoint performance #645 Notify user via email when they're added to a project #306 Added notify user via email when their account is created #307 Changed Improved errors when decoding encryption keys for use with Cloak. #684 Allow users to run ANY job with a custom input. #629 Fixed Ensure JSON schema form inputs are in the same order as they are written in the schema #685","ref":"changelog.html#0-4-6-2023-03-23"},{"type":"extras","title":"Changelog - [0.4.4] - 2023-03-10","doc":"Added Users can receive a digest email reporting on a specified project. #638 #585 Changed Fixed","ref":"changelog.html#0-4-4-2023-03-10"},{"type":"extras","title":"Changelog - [0.4.3] - 2023-03-06","doc":"Added Tooltips on Job Builder panel #650 Changed Upgraded to Phoenix 1.7 (3945856) Fixed Issue with FailureAlerter configuration missing in prod mode.","ref":"changelog.html#0-4-3-2023-03-06"},{"type":"extras","title":"Changelog - [0.4.2] - 2023-02-24","doc":"Added A user can change their own email #247 Added a SCHEMAS_PATH environment variable to override the default folder location for credential schemas #604 Added the ability to configure Google Sheets credentials #536 Function to import a project #574 Changed Users cannot register if they have not selected the terms and conditions #531 Fixed Jobs panel slow for first open after restart #567","ref":"changelog.html#0-4-2-2023-02-24"},{"type":"extras","title":"Changelog - [0.4.0] - 2023-02-08","doc":"Added Added a Delete job button in Inspector Filter workflow runs by text/value in run logs or input body Drop "configuration" key from Run output dataclips after completion Ability to 'rerun' a run from the Run list Attempts and Runs update themselves in the Runs list Configure a project and workflow for a new registering user Run a job with a custom input Added plausible analytics Allow user to click on Webhook Trigger Node to copy webhook URL on workflow diagram Allow any user to delete a credential that they own Create any credential through a form except for OAuth Refit all diagram nodes on browser and container resize Enable distributed Erlang, allowing any number of redundant Lightning nodes to communicate with each other. Users can set up realtime alerts for a project Changed Better code-assist and intelliense in the Job Editor Updated @openfn/workflow-diagram to 0.4.0 Make plus button part of job nodes in Workflow Diagram Updated @openfn/adaptor-docs to 0.0.5 Updated @openfn/describe-package to 0.0.10 Create an follow a manual Run from the Job Inspector View all workflows in a project on the workflows index page Move @openfn/workflow-diagram into the application, the NPM module is now deprecated. Remove workflow name from first node Move the used parts of @openfn/engine into the application. [BREAKING CHANGE] Ported mix openfn.install.runtime into application, use mix lightning.install_runtime . [BREAKING CHANGE] Introduced @openfn/cli as the new runtime for Jobs Rename a workflow through the page heading Hide the dataclips tab for beta Make adaptor default to common@latest Remove jobs list page Better error handling in the docs panel Disable credential ownership transfer in dev and prod environments Add project settings page Change Workorder filters to apply to the aggregate state of the workorder and not the run directly Enable jobs by default Set log level to info Add Beta checkbox to register page User roles and permissions Fixed Don't consider disabled jobs when calculating subsequent runs Fixed overflow on Job Editor Tooltips Fixed auto-scroll when adding a new snippet in the Job Editor Fixed common operation typings in Job Editor","ref":"changelog.html#0-4-0-2023-02-08"},{"type":"extras","title":"Changelog - [0.3.1] - 2022-11-22","doc":"Fixed Fixed bug that attempted to execute HTML scripts in dataclips Fixed bug that prevented workorders from displaying in the order of their last run, descending. Remove alerts after set timeout or close","ref":"changelog.html#0-3-1-2022-11-22"},{"type":"extras","title":"Changelog - [0.3.0] - 2022-11-21","doc":"Added Add seed data for demo site Create adaptor credentials through a form Configure cron expressions through a form View runs grouped by workorders and attempts Run an existing Job with any dataclip uuid from the Job form Changed Redirect users to projects list page when they click on Admin Settings menu Move job, project, input and output Dataclips to Run table Reverse the relationship between Jobs and Triggers. Triggers now can exist on their own; setting the stage for branching and merging workflows Updated Elixir and frontend dependencies [BREAKING CHANGE] Pipeline now uses WorkOrders, previous data is not compatible. Runs, Dataclips and Attempts now all correctly use usec resolution timestamps. Upgraded LiveView to 0.18.0 Upgraded Elixir to 1.14.1 and OTP 25 Workflow Job editor now behaves like a panel Split JobLive.InspectorFormComponent into different plug-able subcomponents Ensure new jobs with cron triggers receive a default frequency Webhooks are now referenced by the trigger id instead of job id. Filter runs by status Filter runs by workflow Filter runs by date View a job run from the runs history View latest matching inputs to run a job with","ref":"changelog.html#0-3-0-2022-11-21"},{"type":"extras","title":"Changelog - [0.2.0] - 2022-09-12","doc":"Changed [BREAKING CHANGE] Add Workflow model, Jobs now belong to a Workflow This is a breaking change to the schema. Use Node.js 18, soon to be in LTS. Visualize success/fail triggers in workflow diagram. Move WorkflowDiagram related actions from DashboardLive into WorkflowLive Move WorkflowDiagram component into liveview, so that we can subscribe to channels (i.e. updating of the diagram when someone changes something). Integrate @openfn/workflow-diagram@0.0.8 and use the new Store interface for updating it. Remove component_mounted event from WorkflowDiagram hook, using a MutationObserver and a Base64 encoded JSON payload. Fixed an issue where the compiler component would try and load a 'nothing adaptor', added a condition to check an adaptor is actually selected. Removed previous Workflow CTE queries, replaced by the introduction of the Workflow model, see ( https://github.com/OpenFn/Lightning/blob/53da6883483e7d8d078783f348da327d1dd72d20/lib/lightning/workflows.ex#L111-L119 ).","ref":"changelog.html#0-2-0-2022-09-12"},{"type":"extras","title":"Changelog - [0.1.13] - 2022-08-29","doc":"Added Allow administrators to configure OIDC providers for authentication (note that this is just for authenticating, not yet for creating new accounts via OIDC) Add Monaco editor to the step/job panel Allow users to delete their own accounts. Schedule their user and credentials data for deletion when they do. Allow superusers to delete a user account. Schedule the user's credentials and user data for deletion when they do. If a user is scheduled for deletion, disable their account and prevent them from logging in. The 'User profile' and 'Credentials' page now have a sidebar menu Changed Project users now have one of the following roles: viewer, editor, admin, owner Users only have the following roles: user, superuser","ref":"changelog.html#0-1-13-2022-08-29"},{"type":"extras","title":"Changelog - [0.1.12] - 2022-08-15","doc":"Added Transfer credential ownership to another user. Create credentials via a form interface* Show "projects with access" in credentials list view. Show job in runs list and run view. Added roles and permissions to workflows and history page #645 *The form is defined by a JSON schema provided by an adaptor, in most cases: e.g., language-dhis2 provides a single schema which defines the required attributes for state.configuration , while language-common provides multiple credential schemas like "oauth" or "basic auth" which define attributes for state.configuration and which might be used by lots of different jobs.) Fixed User menu (top right) appears on top of all other components. User profile screen integrated with the rest of the liveview app.","ref":"changelog.html#0-1-12-2022-08-15"},{"type":"extras","title":"Changelog - [0.1.11] - 2022-08-05","doc":"Fixed Fixed logging in Runner when :debug log level used; note that this caused crashes in Oban","ref":"changelog.html#0-1-11-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.10] - 2022-08-05","doc":"Added Credential auditing Build/version information display for easier debugging Fixed Fixed a bug that enqueued cron-triggered jobs even when they were disabled","ref":"changelog.html#0-1-10-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.9] - 2022-07-27","doc":"Added Navigate to user profile or credentials page and log out through the user icon dropdown Create and edit dataclips Add a production tag to credentials View a dropdown of operations and their description for the language-common v2.0.0-rc2 adaptor (this pattern to be rolled out across adaptors) Changed Navigate between projects through a project picker on the navbar Fixed Run Lightning with docker Security Sensitive credential values are scrubbed from run logs All credentials are encrypted at REST","ref":"changelog.html#0-1-9-2022-07-27"},{"type":"extras","title":"Changelog - [0.1.7] - 2022-06-24","doc":"Added Run a job with a cron trigger Queue jobs via Oban/Postgres Edit jobs via the workflow canvas","ref":"changelog.html#0-1-7-2022-06-24"},{"type":"extras","title":"Changelog - [0.1.6] - 2022-06-07","doc":"Added Register, log in and log out of an account Allow superusers and admin users to create projects Allow admin users to create or disable a user’s account Allow superusers for local deployments to create users and give them access to project spaces Create and edit a job with a webhook, flow/fail or cron trigger Create and edit credentials for a job Copy a job's webhook URL View all workflows in a project visually Deploy lightning locally with Docker Enable a job to automatically process incoming requests Run a job with a webhook or flow/fail trigger View job runs along with their logs, exit code, start and end time View data clips that have initiated job runs (http requests for webhooks, run results) Changed - Removed -","ref":"changelog.html#0-1-6-2022-06-07"}] \ No newline at end of file +searchNodes=[{"type":"module","title":"Lightning","doc":"Lightning keeps the contexts that define your domain and business logic. Contexts are also responsible for managing your data, regardless if it comes from the database, an external API or others.","ref":"Lightning.html"},{"type":"module","title":"Lightning.AdaptorRegistry","doc":"Registry process to query and maintain a list of adaptors available for writing jobs. Currently it queries NPM for all modules in the @openfn organization and filters out modules that are known not to be adaptors. Usage # Starting the process AdaptorRegistry . start_link ( ) # Getting a list of all adaptors Lightning.AdaptorRegistry.AdaptorRegistry . all ( ) Caching By default the results are cached to disk, and will be reused every start. In order to disable or configure caching pass see: start_link/1 . The process uses :continue to return before the adaptors have been queried. This does mean that the first call to the process will be delayed until the handle_continue/2 has finished. Timeouts There is a 'general' timeout of 30s, this is used for GenServer calls like all/1 and also internally when the modules are being queried. NPM can be extremely fast to respond if the package is cached on their side, but can take a couple of seconds if not cached.","ref":"Lightning.AdaptorRegistry.html"},{"type":"function","title":"Lightning.AdaptorRegistry.all/1","doc":"Get the current in-process list of adaptors. This call will wait behind the :continue message when the process starts up, so it may take a while the first time it is called (and the list hasn't been fetched yet).","ref":"Lightning.AdaptorRegistry.html#all/1"},{"type":"function","title":"Lightning.AdaptorRegistry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorRegistry.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorRegistry.fetch/0","doc":"Fetch a list of packages for the @openfn organisation","ref":"Lightning.AdaptorRegistry.html#fetch/0"},{"type":"function","title":"Lightning.AdaptorRegistry.latest_for/2","doc":"Get a latest version for a given module.","ref":"Lightning.AdaptorRegistry.html#latest_for/2"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_adaptor/1","doc":"","ref":"Lightning.AdaptorRegistry.html#resolve_adaptor/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name/1","doc":"Destructures an NPM style package name into module name and version. Example iex> resolve_package_name ( "@openfn/language-salesforce@1.2.3" ) { "@openfn/language-salesforce" , "1.2.3" } iex> resolve_package_name ( "@openfn/language-salesforce" ) { "@openfn/language-salesforce" , nil }","ref":"Lightning.AdaptorRegistry.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorRegistry.resolve_package_name!/1","doc":"Same as resolve_package_name/1 except will throw an exception if a package name cannot be matched.","ref":"Lightning.AdaptorRegistry.html#resolve_package_name!/1"},{"type":"function","title":"Lightning.AdaptorRegistry.start_link/1","doc":"Starts the AdaptorRegistry Options :use_cache (defaults to false) - stores the last set of results on disk and uses the cached file for every subsequent start. It can either be a boolean, or a string - the latter being a file path to set where the cache file is located. :name (defaults to AdaptorRegistry) - the name of the process, useful for testing and/or running multiple versions of the registry","ref":"Lightning.AdaptorRegistry.html#start_link/1"},{"type":"function","title":"Lightning.AdaptorRegistry.versions_for/2","doc":"Get a list of versions for a given module.","ref":"Lightning.AdaptorRegistry.html#versions_for/2"},{"type":"module","title":"Lightning.AdaptorRegistry.Npm","doc":"NPM API functions","ref":"Lightning.AdaptorRegistry.Npm.html"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#delete!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#get!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#head!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#options!/3"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.package_detail/1","doc":"Retrieve all details for an NPM package","ref":"Lightning.AdaptorRegistry.Npm.html#package_detail/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#patch!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#post!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_body/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_options/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_request_params/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_headers/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_status_code/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AdaptorRegistry.Npm.html#process_url/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#put!/4"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AdaptorRegistry.Npm.html#request/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AdaptorRegistry.Npm.html#request/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AdaptorRegistry.Npm.html#request!/5"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AdaptorRegistry.Npm.html#start/0"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AdaptorRegistry.Npm.html#stream_next/1"},{"type":"function","title":"Lightning.AdaptorRegistry.Npm.user_packages/1","doc":"Retrieve all packages for a given user or organization. Return empty list if application cannot connect to NPM. (E.g., because it's started offline.)","ref":"Lightning.AdaptorRegistry.Npm.html#user_packages/1"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.body/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:body/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.headers/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:headers/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.method/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:method/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.options/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:options/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.params/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:params/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.request/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:request/0"},{"type":"type","title":"Lightning.AdaptorRegistry.Npm.url/0","doc":"","ref":"Lightning.AdaptorRegistry.Npm.html#t:url/0"},{"type":"module","title":"Lightning.AdaptorService","doc":"The Adaptor Service is use to query and install adaptors in order to run jobs. On startup, it queries the filesystem for package.json files and builds up a list of available adaptors. Configuration The service requires at least :adaptors_path , which is used to both query which adaptors are installed and when to install new adaptors. Another optional setting is: :repo , which must point at a module that will be used to do the querying and installing. Installing Adaptors Using the install/2 function an adaptor can be installed, which will also add it to the list of available adaptors. The adaptor is marked as :installing , to allow for conditional behaviour elsewhere such as delaying or rejecting processing until the adaptor becomes available. Looking up adaptors The module leans on Elixir's built-in Version module to provide version lookups. When looking up an adaptor, either a string or a tuple can be used. In the case of requesting the latest version, any one of these will return the latest version the service is aware of. @openfn/language-http @openfn/language-http@latest {"@openfn/language-http", nil} {"@openfn/language-http", "latest"} {~r/language-http/, "latest"} You can also request a specific version, or use a range specification: @openfn/language-http@1.2.3 {"@openfn/language-http", "~> 1.2.0"} {"@openfn/language-http", "< 2.0.0"} NOTE More complex npm style install strings like: ">=0.1.0 <0.2.0" are not supported. Generally the tuple style is preferred when using range specifications as the npm style strings have a simplistic regex splitter. See Version for more details on matching versions.","ref":"Lightning.AdaptorService.html"},{"type":"function","title":"Lightning.AdaptorService.build_aliased_name/1","doc":"Turns a package name and version into a string for NPM. Since multiple versions of the same package can be installed, it's important to rely on npms built-in package aliasing. E.g. @openfn/language-http@1.2.8 turns into: ` @openfn / language - http - 1.2 . 8 @npm : @openfn / language - http @ 1.2 . 8 ` Which is pretty long winded but necessary for the reason above. If using this module as a base, it's likely you would need to adaptor this to suit your particular naming strategy.","ref":"Lightning.AdaptorService.html#build_aliased_name/1"},{"type":"function","title":"Lightning.AdaptorService.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AdaptorService.html#child_spec/1"},{"type":"function","title":"Lightning.AdaptorService.find_adaptor/2","doc":"","ref":"Lightning.AdaptorService.html#find_adaptor/2"},{"type":"function","title":"Lightning.AdaptorService.get_adaptors/1","doc":"","ref":"Lightning.AdaptorService.html#get_adaptors/1"},{"type":"function","title":"Lightning.AdaptorService.install/2","doc":"","ref":"Lightning.AdaptorService.html#install/2"},{"type":"function","title":"Lightning.AdaptorService.install!/2","doc":"","ref":"Lightning.AdaptorService.html#install!/2"},{"type":"function","title":"Lightning.AdaptorService.installed?/2","doc":"","ref":"Lightning.AdaptorService.html#installed?/2"},{"type":"function","title":"Lightning.AdaptorService.resolve_package_name/1","doc":"","ref":"Lightning.AdaptorService.html#resolve_package_name/1"},{"type":"function","title":"Lightning.AdaptorService.start_link/1","doc":"","ref":"Lightning.AdaptorService.html#start_link/1"},{"type":"type","title":"Lightning.AdaptorService.package_spec/0","doc":"","ref":"Lightning.AdaptorService.html#t:package_spec/0"},{"type":"module","title":"Lightning.Attempt","doc":"Ecto model for Attempts.","ref":"Lightning.Attempt.html"},{"type":"function","title":"Lightning.Attempt.new/1","doc":"","ref":"Lightning.Attempt.html#new/1"},{"type":"type","title":"Lightning.Attempt.t/0","doc":"","ref":"Lightning.Attempt.html#t:t/0"},{"type":"module","title":"Lightning.AttemptRun","doc":"Ecto model for an Attempts Runs.","ref":"Lightning.AttemptRun.html"},{"type":"function","title":"Lightning.AttemptRun.new/1","doc":"","ref":"Lightning.AttemptRun.html#new/1"},{"type":"function","title":"Lightning.AttemptRun.new/2","doc":"","ref":"Lightning.AttemptRun.html#new/2"},{"type":"type","title":"Lightning.AttemptRun.t/0","doc":"","ref":"Lightning.AttemptRun.html#t:t/0"},{"type":"module","title":"Lightning.AttemptService","doc":"The Attempts context.","ref":"Lightning.AttemptService.html"},{"type":"function","title":"Lightning.AttemptService.append/2","doc":"Adds an Attempt to an unsaved Run When given an Attempt, it simply adds the Run to a new AttemptRun. However when given an AttemptRun, the Run (from the AttemptRun) is set as the previous Run for the new unsaved Run.","ref":"Lightning.AttemptService.html#append/2"},{"type":"function","title":"Lightning.AttemptService.build_attempt/2","doc":"","ref":"Lightning.AttemptService.html#build_attempt/2"},{"type":"function","title":"Lightning.AttemptService.calculate_runs/3","doc":"","ref":"Lightning.AttemptService.html#calculate_runs/3"},{"type":"function","title":"Lightning.AttemptService.create_attempt/3","doc":"Create an attempt Examples iex> create_attempt ( %{ field : value } ) { :ok , % Attempt { } } iex> create_attempt ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.AttemptService.html#create_attempt/3"},{"type":"function","title":"Lightning.AttemptService.get_for_rerun/2","doc":"","ref":"Lightning.AttemptService.html#get_for_rerun/2"},{"type":"function","title":"Lightning.AttemptService.get_last_attempt_for/1","doc":"Get the latest attempt associated to a given run","ref":"Lightning.AttemptService.html#get_last_attempt_for/1"},{"type":"function","title":"Lightning.AttemptService.get_workflow_for/1","doc":"","ref":"Lightning.AttemptService.html#get_workflow_for/1"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_job/2","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of workorder ids that are associated to the given Job","ref":"Lightning.AttemptService.html#list_for_rerun_from_job/2"},{"type":"function","title":"Lightning.AttemptService.list_for_rerun_from_start/1","doc":"Returns a list of AttemptRun structs that should be rerun for the given list of work order ids.","ref":"Lightning.AttemptService.html#list_for_rerun_from_start/1"},{"type":"function","title":"Lightning.AttemptService.retry/3","doc":"Creates a new Attempt starting from a given run. All upstream/prior Runs that were performed on that attempt are associated with the new Attempt, where as the specified run is used to create a new one and is added to the Attempt. Any runs downstream from the Run given are ignored.","ref":"Lightning.AttemptService.html#retry/3"},{"type":"function","title":"Lightning.AttemptService.retry_many/2","doc":"Creates new Attempts for each pair of corresponding AttemptRun and InvocationReason.","ref":"Lightning.AttemptService.html#retry_many/2"},{"type":"module","title":"Lightning.Auditing","doc":"Context for working with Audit records.","ref":"Lightning.Auditing.html"},{"type":"function","title":"Lightning.Auditing.list_all/1","doc":"","ref":"Lightning.Auditing.html#list_all/1"},{"type":"module","title":"Lightning.Auditing.Model","doc":"Macro module to add common model behaviour to a given Ecto model","ref":"Lightning.Auditing.Model.html"},{"type":"function","title":"Lightning.Auditing.Model.event/5","doc":"Creates a schema changeset for the event identified by row_id and caused by actor_id . The given metadata can be either nil , Ecto.Changeset , struct or map. It returns :no_changes in case of an Ecto.Changeset metadata that changed nothing or an Ecto.Changeset with the event ready to be inserted.","ref":"Lightning.Auditing.Model.html#event/5"},{"type":"function","title":"Lightning.Auditing.Model.save/2","doc":"Saves the event to the Repo . In case of nothing changes, do nothing. It returns {:ok, :no_changes} if nothing changed, {:ok, struct} if the log has been successfully saved or {:error, changeset} in case of error.","ref":"Lightning.Auditing.Model.html#save/2"},{"type":"module","title":"Lightning.AuthProviders","doc":"Context module for dealing with external Auth Providers.","ref":"Lightning.AuthProviders.html"},{"type":"function","title":"Lightning.AuthProviders.build_handler/2","doc":"","ref":"Lightning.AuthProviders.html#build_handler/2"},{"type":"function","title":"Lightning.AuthProviders.create/1","doc":"","ref":"Lightning.AuthProviders.html#create/1"},{"type":"function","title":"Lightning.AuthProviders.create_handler/1","doc":"","ref":"Lightning.AuthProviders.html#create_handler/1"},{"type":"function","title":"Lightning.AuthProviders.delete!/1","doc":"","ref":"Lightning.AuthProviders.html#delete!/1"},{"type":"function","title":"Lightning.AuthProviders.get_authorize_url/1","doc":"Retrieve the authorization url for a given handler or handler name.","ref":"Lightning.AuthProviders.html#get_authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.get_existing/0","doc":"","ref":"Lightning.AuthProviders.html#get_existing/0"},{"type":"function","title":"Lightning.AuthProviders.get_existing/1","doc":"","ref":"Lightning.AuthProviders.html#get_existing/1"},{"type":"function","title":"Lightning.AuthProviders.get_handler/1","doc":"","ref":"Lightning.AuthProviders.html#get_handler/1"},{"type":"function","title":"Lightning.AuthProviders.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.new/0","doc":"","ref":"Lightning.AuthProviders.html#new/0"},{"type":"function","title":"Lightning.AuthProviders.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.html#remove_handler/1"},{"type":"function","title":"Lightning.AuthProviders.update/2","doc":"","ref":"Lightning.AuthProviders.html#update/2"},{"type":"module","title":"Lightning.AuthProviders.AuthConfig","doc":"AuthProvider model","ref":"Lightning.AuthProviders.AuthConfig.html"},{"type":"type","title":"Lightning.AuthProviders.AuthConfig.t/0","doc":"","ref":"Lightning.AuthProviders.AuthConfig.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.CacheWarmer","doc":"Dummy warmer which caches database rows every 30s.","ref":"Lightning.AuthProviders.CacheWarmer.html"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.AuthProviders.CacheWarmer.html#child_spec/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.execute/1","doc":"Executes this cache warmer with a connection.","ref":"Lightning.AuthProviders.CacheWarmer.html#execute/1"},{"type":"function","title":"Lightning.AuthProviders.CacheWarmer.interval/0","doc":"Returns the interval for this warmer.","ref":"Lightning.AuthProviders.CacheWarmer.html#interval/0"},{"type":"module","title":"Lightning.AuthProviders.Google","doc":"","ref":"Lightning.AuthProviders.Google.html"},{"type":"function","title":"Lightning.AuthProviders.Google.authorize_url/2","doc":"","ref":"Lightning.AuthProviders.Google.html#authorize_url/2"},{"type":"function","title":"Lightning.AuthProviders.Google.build_client/1","doc":"Builds a new client","ref":"Lightning.AuthProviders.Google.html#build_client/1"},{"type":"function","title":"Lightning.AuthProviders.Google.get_config/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_config/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Google.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown/0"},{"type":"function","title":"Lightning.AuthProviders.Google.get_wellknown!/0","doc":"","ref":"Lightning.AuthProviders.Google.html#get_wellknown!/0"},{"type":"function","title":"Lightning.AuthProviders.Google.refresh_token/2","doc":"","ref":"Lightning.AuthProviders.Google.html#refresh_token/2"},{"type":"module","title":"Lightning.AuthProviders.Handler","doc":"Module which wraps Oauth configuration and a WellKnown document into a convenient struct that can be used to authenticate users against any OIDC compliant provider.","ref":"Lightning.AuthProviders.Handler.html"},{"type":"function","title":"Lightning.AuthProviders.Handler.authorize_url/1","doc":"","ref":"Lightning.AuthProviders.Handler.html#authorize_url/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.from_model/1","doc":"Returns a Handler from a given AuthConfig","ref":"Lightning.AuthProviders.Handler.html#from_model/1"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_token/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_token/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.get_userinfo/2","doc":"","ref":"Lightning.AuthProviders.Handler.html#get_userinfo/2"},{"type":"function","title":"Lightning.AuthProviders.Handler.new/2","doc":"Create a new Provider struct, expects a name and opts: :client_id - The providers issued id :client_secret - Secret for the client :redirect_uri - The URI for redirecting after authentication, usually the callback url in the router. :wellknown - A AuthProviders.WellKnown struct with the providers .well-known/openid-configuration .","ref":"Lightning.AuthProviders.Handler.html#new/2"},{"type":"type","title":"Lightning.AuthProviders.Handler.opts/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:opts/0"},{"type":"type","title":"Lightning.AuthProviders.Handler.t/0","doc":"","ref":"Lightning.AuthProviders.Handler.html#t:t/0"},{"type":"module","title":"Lightning.AuthProviders.Store","doc":"Store module for caching Handlers. Since Handlers often have to fetch their .well-known files when being initialized we cache these in order to avoid repeatedly making HTTP requests to a providers API.","ref":"Lightning.AuthProviders.Store.html"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.get_handlers/0","doc":"","ref":"Lightning.AuthProviders.Store.html#get_handlers/0"},{"type":"function","title":"Lightning.AuthProviders.Store.put_handler/2","doc":"","ref":"Lightning.AuthProviders.Store.html#put_handler/2"},{"type":"function","title":"Lightning.AuthProviders.Store.remove_handler/1","doc":"","ref":"Lightning.AuthProviders.Store.html#remove_handler/1"},{"type":"type","title":"Lightning.AuthProviders.Store.finder/0","doc":"","ref":"Lightning.AuthProviders.Store.html#t:finder/0"},{"type":"module","title":"Lightning.AuthProviders.WellKnown","doc":"A datastructure to fetch and hold information about a given OIDC/OAuth provider","ref":"Lightning.AuthProviders.WellKnown.html"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#delete!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.fetch!/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#fetch!/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#get!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#head!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.new/1","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#new/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#options!/3"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#patch!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#post!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_options/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_params/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_request_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_body/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_chunk/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_headers/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_response_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_status_code/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Lightning.AuthProviders.WellKnown.html#process_url/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#put!/4"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Lightning.AuthProviders.WellKnown.html#request/1"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Lightning.AuthProviders.WellKnown.html#request/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Lightning.AuthProviders.WellKnown.html#request!/5"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Lightning.AuthProviders.WellKnown.html#start/0"},{"type":"function","title":"Lightning.AuthProviders.WellKnown.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Lightning.AuthProviders.WellKnown.html#stream_next/1"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.body/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:body/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.headers/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:headers/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.method/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:method/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.options/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:options/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.params/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:params/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.request/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:request/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.t/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:t/0"},{"type":"type","title":"Lightning.AuthProviders.WellKnown.url/0","doc":"","ref":"Lightning.AuthProviders.WellKnown.html#t:url/0"},{"type":"module","title":"Lightning.CLI","doc":"Module providing facilities to make calls to the OpenFn CLI. See @openfn/cli","ref":"Lightning.CLI.html"},{"type":"function","title":"Lightning.CLI.execute/1","doc":"Execute a command in a child process and parse the results.","ref":"Lightning.CLI.html#execute/1"},{"type":"function","title":"Lightning.CLI.metadata/2","doc":"Retrieve metadata for a given adaptor and configuration.","ref":"Lightning.CLI.html#metadata/2"},{"type":"module","title":"Lightning.CLI.Result","doc":"Struct that wraps the output of an OpenFn CLI call. Containing the keys: start_time end_time status logs Logs The OpenFn CLI returns JSON formatted log lines, which are decoded and added to a Result struct. There are two kinds of output: { "level" :"<<level>>" , "name" :"<<module>>" , "message" :"..." ] , "time" :< < timestamp >> } These are usually for general logging, and debugging. { "message" : [ "<<message|filepath|output>>" ] } The above is the equivalent of the output of a command","ref":"Lightning.CLI.Result.html"},{"type":"function","title":"Lightning.CLI.Result.get_messages/1","doc":"Returns message type log lines from a Result .","ref":"Lightning.CLI.Result.html#get_messages/1"},{"type":"function","title":"Lightning.CLI.Result.new/1","doc":"","ref":"Lightning.CLI.Result.html#new/1"},{"type":"function","title":"Lightning.CLI.Result.parse/2","doc":"","ref":"Lightning.CLI.Result.html#parse/2"},{"type":"type","title":"Lightning.CLI.Result.t/0","doc":"","ref":"Lightning.CLI.Result.html#t:t/0"},{"type":"module","title":"Lightning.Demo","doc":"Demo encapsulates logic for setting up a demonstration site.","ref":"Lightning.Demo.html"},{"type":"function","title":"Lightning.Demo.reset_demo/0","doc":"Deletes everything in the database including the superuser and creates a set of publicly available users for a demo site via a command that can be run on Kubernetes-deployed systems.","ref":"Lightning.Demo.html#reset_demo/0"},{"type":"module","title":"Lightning.ExportUtils","doc":"Module that expose a function generating a complete and valid yaml string from a project and its workflows.","ref":"Lightning.ExportUtils.html"},{"type":"function","title":"Lightning.ExportUtils.build_yaml_tree/2","doc":"","ref":"Lightning.ExportUtils.html#build_yaml_tree/2"},{"type":"function","title":"Lightning.ExportUtils.generate_new_yaml/1","doc":"","ref":"Lightning.ExportUtils.html#generate_new_yaml/1"},{"type":"module","title":"Lightning.FailureNotifierView","doc":"","ref":"Lightning.FailureNotifierView.html"},{"type":"function","title":"Lightning.FailureNotifierView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"Lightning.FailureNotifierView.html#__resource__/0"},{"type":"function","title":"Lightning.FailureNotifierView.failure_alert.html/1","doc":"","ref":"Lightning.FailureNotifierView.html#failure_alert.html/1"},{"type":"function","title":"Lightning.FailureNotifierView.render/2","doc":"Renders the given template locally.","ref":"Lightning.FailureNotifierView.html#render/2"},{"type":"function","title":"Lightning.FailureNotifierView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"Lightning.FailureNotifierView.html#template_not_found/2"},{"type":"module","title":"Lightning.Helpers","doc":"Common functions for the context","ref":"Lightning.Helpers.html"},{"type":"function","title":"Lightning.Helpers.coerce_json_field/2","doc":"Changes a given maps field from a json string to a map. If it cannot be converted, it leaves the original value","ref":"Lightning.Helpers.html#coerce_json_field/2"},{"type":"function","title":"Lightning.Helpers.indefinite_article/1","doc":"","ref":"Lightning.Helpers.html#indefinite_article/1"},{"type":"function","title":"Lightning.Helpers.json_safe/1","doc":"Recursively ensures a given map is safe to convert to JSON, where all keys are strings and all values are json safe (primitive values).","ref":"Lightning.Helpers.html#json_safe/1"},{"type":"function","title":"Lightning.Helpers.ms_to_human/1","doc":"Converts milliseconds (integer) to a human duration, such as "1 minute" or "45 years, 6 months, 5 days, 21 hours, 12 minutes, 34 seconds" using Timex.Format.Duration.Formatters.Humanized.format() .","ref":"Lightning.Helpers.html#ms_to_human/1"},{"type":"module","title":"Lightning.MetadataService","doc":"Retrieves metadata for a given credential and adaptor using the OpenFn CLI.","ref":"Lightning.MetadataService.html"},{"type":"function","title":"Lightning.MetadataService.fetch/2","doc":"Retrieve metadata for a given adaptor and credential. The adaptor must be an npm specification.","ref":"Lightning.MetadataService.html#fetch/2"},{"type":"module","title":"Lightning.Name","doc":"Generates a random names.","ref":"Lightning.Name.html"},{"type":"function","title":"Lightning.Name.generate/1","doc":"","ref":"Lightning.Name.html#generate/1"},{"type":"module","title":"Lightning.ObanManager","doc":"The Oban Manager","ref":"Lightning.ObanManager.html"},{"type":"function","title":"Lightning.ObanManager.handle_event/4","doc":"","ref":"Lightning.ObanManager.html#handle_event/4"},{"type":"module","title":"Lightning.Policies.Permissions","doc":"This module defines a unique interface managing authorizations in Lightning. Users in Lightning have instance-wide and project-wide roles which determine their level of access to resources in the application. Fo rmore details see the documentation . These authorizations policies are all implemented under the lib/lightning/policies folder. In that folder you can find 3 files: The users.ex file has all the policies for the instances wide access levels The project_users.ex file has all the policies for the project wide access levels The permissions.ex file defines the Lightning.Policies.Permissions.can/4 interface. Which is a wrapper around the Bodyguard.permit/4 function. We use that interface to be able to harmonize the use of policies accross the entire app. All the policies are tested in the test/lightning/policies folder. And the test are written in a way that allows the reader to quickly who can do what in the app. We have two variants of the Lightning.Policies.Permissions.can/4 interface: Lightning.Policies.Permissions.can(policy, action, actor, resource) returns :ok if the actor can perform the action on the resource and {:error, :unauthorized} otherwise. Lightning.Policies.Permissions.can?(policy, action, actor, resource) returns true if the actor can perform the action on the resource and false otherwise. Here is an example of how we the Lightning.Policies.Permissions.can/4 interface to check if the a user can edit a job or not can_edit_job = Lightning.Policies.ProjectUsers |> Lightning.Policies.Permissions . can? ( :edit_job , socket . assigns . current_user , socket . assigns . project ) if can_edit_job do # allow user to edit the job else # quick user out end","ref":"Lightning.Policies.Permissions.html"},{"type":"function","title":"Lightning.Policies.Permissions.can/4","doc":"checks if user has the permissions to apply action using some policy module Returns :ok if user can apply action and {:error, :unauthorized} otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) :ok iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) { :error , :unauthorized }","ref":"Lightning.Policies.Permissions.html#can/4"},{"type":"function","title":"Lightning.Policies.Permissions.can?/4","doc":"same as can/4 but returns true if user can apply action and false otherwise Examples iex> can ( Lightning.Policies.Users , :create_workflow , user , project ) true iex> can ( Lightning.Policies.Users , :create_project , user , %{ } ) false","ref":"Lightning.Policies.Permissions.html#can?/4"},{"type":"module","title":"Lightning.Policies.ProjectUsers","doc":"The Bodyguard Policy module for projects members roles.","ref":"Lightning.Policies.ProjectUsers.html"},{"type":"function","title":"Lightning.Policies.ProjectUsers.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action in that project and false if they cannot. Note that permissions are grouped by action, rather than by user role. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.ProjectUsers.html#authorize/3"},{"type":"type","title":"Lightning.Policies.ProjectUsers.actions/0","doc":"","ref":"Lightning.Policies.ProjectUsers.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Provisioning","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Provisioning.html"},{"type":"function","title":"Lightning.Policies.Provisioning.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied. Only a superuser can provision a new project. Owners and admins can update existing projects.","ref":"Lightning.Policies.Provisioning.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Provisioning.actions/0","doc":"","ref":"Lightning.Policies.Provisioning.html#t:actions/0"},{"type":"module","title":"Lightning.Policies.Users","doc":"The Bodyguard Policy module for users roles.","ref":"Lightning.Policies.Users.html"},{"type":"function","title":"Lightning.Policies.Users.authorize/3","doc":"authorize/3 takes an action, a user, and a project. It checks the user's role for this project and returns true if the user can perform the action and false if they cannot. Note that permissions are grouped by action. We deny by default, so if a user's role is not added to the allow roles list for a particular action they are denied.","ref":"Lightning.Policies.Users.html#authorize/3"},{"type":"type","title":"Lightning.Policies.Users.actions/0","doc":"","ref":"Lightning.Policies.Users.html#t:actions/0"},{"type":"module","title":"Lightning.Release","doc":"Used for executing DB release tasks when run in production without Mix installed.","ref":"Lightning.Release.html"},{"type":"function","title":"Lightning.Release.create_db/0","doc":"","ref":"Lightning.Release.html#create_db/0"},{"type":"function","title":"Lightning.Release.load_app/0","doc":"","ref":"Lightning.Release.html#load_app/0"},{"type":"function","title":"Lightning.Release.migrate/0","doc":"","ref":"Lightning.Release.html#migrate/0"},{"type":"function","title":"Lightning.Release.rollback/2","doc":"","ref":"Lightning.Release.html#rollback/2"},{"type":"module","title":"Lightning.Repo","doc":"","ref":"Lightning.Repo.html"},{"type":"function","title":"Lightning.Repo.aggregate/3","doc":"Callback implementation for Ecto.Repo.aggregate/3 .","ref":"Lightning.Repo.html#aggregate/3"},{"type":"function","title":"Lightning.Repo.aggregate/4","doc":"Callback implementation for Ecto.Repo.aggregate/4 .","ref":"Lightning.Repo.html#aggregate/4"},{"type":"function","title":"Lightning.Repo.all/2","doc":"Callback implementation for Ecto.Repo.all/2 .","ref":"Lightning.Repo.html#all/2"},{"type":"function","title":"Lightning.Repo.checked_out?/0","doc":"Callback implementation for Ecto.Repo.checked_out?/0 .","ref":"Lightning.Repo.html#checked_out?/0"},{"type":"function","title":"Lightning.Repo.checkout/2","doc":"Callback implementation for Ecto.Repo.checkout/2 .","ref":"Lightning.Repo.html#checkout/2"},{"type":"function","title":"Lightning.Repo.child_spec/1","doc":"","ref":"Lightning.Repo.html#child_spec/1"},{"type":"function","title":"Lightning.Repo.config/0","doc":"Callback implementation for Ecto.Repo.config/0 .","ref":"Lightning.Repo.html#config/0"},{"type":"function","title":"Lightning.Repo.default_options/1","doc":"Callback implementation for Ecto.Repo.default_options/1 .","ref":"Lightning.Repo.html#default_options/1"},{"type":"function","title":"Lightning.Repo.delete/2","doc":"Callback implementation for Ecto.Repo.delete/2 .","ref":"Lightning.Repo.html#delete/2"},{"type":"function","title":"Lightning.Repo.delete!/2","doc":"Callback implementation for Ecto.Repo.delete!/2 .","ref":"Lightning.Repo.html#delete!/2"},{"type":"function","title":"Lightning.Repo.delete_all/2","doc":"Callback implementation for Ecto.Repo.delete_all/2 .","ref":"Lightning.Repo.html#delete_all/2"},{"type":"function","title":"Lightning.Repo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the pool to disconnect within the given interval. See Ecto.Adapters.SQL.disconnect_all/3 for more information.","ref":"Lightning.Repo.html#disconnect_all/2"},{"type":"function","title":"Lightning.Repo.exists?/2","doc":"Callback implementation for Ecto.Repo.exists?/2 .","ref":"Lightning.Repo.html#exists?/2"},{"type":"function","title":"Lightning.Repo.explain/3","doc":"A convenience function for SQL-based repositories that executes an EXPLAIN statement or similar depending on the adapter to obtain statistics for the given query. See Ecto.Adapters.SQL.explain/4 for more information.","ref":"Lightning.Repo.html#explain/3"},{"type":"function","title":"Lightning.Repo.get/3","doc":"Callback implementation for Ecto.Repo.get/3 .","ref":"Lightning.Repo.html#get/3"},{"type":"function","title":"Lightning.Repo.get!/3","doc":"Callback implementation for Ecto.Repo.get!/3 .","ref":"Lightning.Repo.html#get!/3"},{"type":"function","title":"Lightning.Repo.get_by/3","doc":"Callback implementation for Ecto.Repo.get_by/3 .","ref":"Lightning.Repo.html#get_by/3"},{"type":"function","title":"Lightning.Repo.get_by!/3","doc":"Callback implementation for Ecto.Repo.get_by!/3 .","ref":"Lightning.Repo.html#get_by!/3"},{"type":"function","title":"Lightning.Repo.get_dynamic_repo/0","doc":"Callback implementation for Ecto.Repo.get_dynamic_repo/0 .","ref":"Lightning.Repo.html#get_dynamic_repo/0"},{"type":"function","title":"Lightning.Repo.in_transaction?/0","doc":"Callback implementation for Ecto.Repo.in_transaction?/0 .","ref":"Lightning.Repo.html#in_transaction?/0"},{"type":"function","title":"Lightning.Repo.insert/2","doc":"Callback implementation for Ecto.Repo.insert/2 .","ref":"Lightning.Repo.html#insert/2"},{"type":"function","title":"Lightning.Repo.insert!/2","doc":"Callback implementation for Ecto.Repo.insert!/2 .","ref":"Lightning.Repo.html#insert!/2"},{"type":"function","title":"Lightning.Repo.insert_all/3","doc":"Callback implementation for Ecto.Repo.insert_all/3 .","ref":"Lightning.Repo.html#insert_all/3"},{"type":"function","title":"Lightning.Repo.insert_or_update/2","doc":"Callback implementation for Ecto.Repo.insert_or_update/2 .","ref":"Lightning.Repo.html#insert_or_update/2"},{"type":"function","title":"Lightning.Repo.insert_or_update!/2","doc":"Callback implementation for Ecto.Repo.insert_or_update!/2 .","ref":"Lightning.Repo.html#insert_or_update!/2"},{"type":"function","title":"Lightning.Repo.load/2","doc":"Callback implementation for Ecto.Repo.load/2 .","ref":"Lightning.Repo.html#load/2"},{"type":"function","title":"Lightning.Repo.one/2","doc":"Callback implementation for Ecto.Repo.one/2 .","ref":"Lightning.Repo.html#one/2"},{"type":"function","title":"Lightning.Repo.one!/2","doc":"Callback implementation for Ecto.Repo.one!/2 .","ref":"Lightning.Repo.html#one!/2"},{"type":"function","title":"Lightning.Repo.paginate/2","doc":"","ref":"Lightning.Repo.html#paginate/2"},{"type":"function","title":"Lightning.Repo.preload/3","doc":"Callback implementation for Ecto.Repo.preload/3 .","ref":"Lightning.Repo.html#preload/3"},{"type":"function","title":"Lightning.Repo.prepare_query/3","doc":"Callback implementation for Ecto.Repo.prepare_query/3 .","ref":"Lightning.Repo.html#prepare_query/3"},{"type":"function","title":"Lightning.Repo.put_dynamic_repo/1","doc":"Callback implementation for Ecto.Repo.put_dynamic_repo/1 .","ref":"Lightning.Repo.html#put_dynamic_repo/1"},{"type":"function","title":"Lightning.Repo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query/4 for more information.","ref":"Lightning.Repo.html#query/3"},{"type":"function","title":"Lightning.Repo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query. See Ecto.Adapters.SQL.query!/4 for more information.","ref":"Lightning.Repo.html#query!/3"},{"type":"function","title":"Lightning.Repo.query_many/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many/4 for more information.","ref":"Lightning.Repo.html#query_many/3"},{"type":"function","title":"Lightning.Repo.query_many!/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query. See Ecto.Adapters.SQL.query_many!/4 for more information.","ref":"Lightning.Repo.html#query_many!/3"},{"type":"function","title":"Lightning.Repo.reload/2","doc":"Callback implementation for Ecto.Repo.reload/2 .","ref":"Lightning.Repo.html#reload/2"},{"type":"function","title":"Lightning.Repo.reload!/2","doc":"Callback implementation for Ecto.Repo.reload!/2 .","ref":"Lightning.Repo.html#reload!/2"},{"type":"function","title":"Lightning.Repo.rollback/1","doc":"Callback implementation for Ecto.Repo.rollback/1 .","ref":"Lightning.Repo.html#rollback/1"},{"type":"function","title":"Lightning.Repo.scrivener_defaults/0","doc":"","ref":"Lightning.Repo.html#scrivener_defaults/0"},{"type":"function","title":"Lightning.Repo.start_link/1","doc":"Callback implementation for Ecto.Repo.start_link/1 .","ref":"Lightning.Repo.html#start_link/1"},{"type":"function","title":"Lightning.Repo.stop/1","doc":"Callback implementation for Ecto.Repo.stop/1 .","ref":"Lightning.Repo.html#stop/1"},{"type":"function","title":"Lightning.Repo.stream/2","doc":"Callback implementation for Ecto.Repo.stream/2 .","ref":"Lightning.Repo.html#stream/2"},{"type":"function","title":"Lightning.Repo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL. See Ecto.Adapters.SQL.to_sql/3 for more information.","ref":"Lightning.Repo.html#to_sql/2"},{"type":"function","title":"Lightning.Repo.transaction/2","doc":"Callback implementation for Ecto.Repo.transaction/2 .","ref":"Lightning.Repo.html#transaction/2"},{"type":"function","title":"Lightning.Repo.update/2","doc":"Callback implementation for Ecto.Repo.update/2 .","ref":"Lightning.Repo.html#update/2"},{"type":"function","title":"Lightning.Repo.update!/2","doc":"Callback implementation for Ecto.Repo.update!/2 .","ref":"Lightning.Repo.html#update!/2"},{"type":"function","title":"Lightning.Repo.update_all/3","doc":"Callback implementation for Ecto.Repo.update_all/3 .","ref":"Lightning.Repo.html#update_all/3"},{"type":"module","title":"Lightning.RunSearchForm","doc":"Run filtering search form.","ref":"Lightning.RunSearchForm.html"},{"type":"module","title":"Lightning.SafetyString","doc":"Utilities for securely encoding serializable structs, lists and strings into URL-safe strings. In order to pass the state around in a URL, in a manner that protects secrets from leaking - and allows us to avoid persistance we take a set of parameters and: Encode into a URI query string gzip it to save characters encrypt the string base64 encode it for URI encoding safety","ref":"Lightning.SafetyString.html"},{"type":"function","title":"Lightning.SafetyString.decode/1","doc":"","ref":"Lightning.SafetyString.html#decode/1"},{"type":"function","title":"Lightning.SafetyString.encode/1","doc":"","ref":"Lightning.SafetyString.html#encode/1"},{"type":"module","title":"Lightning.Scrubber","doc":"Process used to scrub strings of sensitive information. Can be started via start_link/1 . { :ok , scrubber } = Lightning.Scrubber . start_link ( samples : Lightning.Credentials . sensitive_values_for ( credential ) ) Takes an optional :name key, in case you need to name the process.","ref":"Lightning.Scrubber.html"},{"type":"function","title":"Lightning.Scrubber.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Scrubber.html#child_spec/1"},{"type":"function","title":"Lightning.Scrubber.encode_samples/1","doc":"Prepare a list of sensitive samples (strings) into a potentially bigger list composed of variations a sample may appear.","ref":"Lightning.Scrubber.html#encode_samples/1"},{"type":"function","title":"Lightning.Scrubber.samples/1","doc":"","ref":"Lightning.Scrubber.html#samples/1"},{"type":"function","title":"Lightning.Scrubber.scrub/2","doc":"","ref":"Lightning.Scrubber.html#scrub/2"},{"type":"function","title":"Lightning.Scrubber.start_link/1","doc":"","ref":"Lightning.Scrubber.html#start_link/1"},{"type":"module","title":"Lightning.SetupUtils","doc":"SetupUtils encapsulates logic for setting up initial data for various sites.","ref":"Lightning.SetupUtils.html"},{"type":"function","title":"Lightning.SetupUtils.add_and_update_runs/3","doc":"","ref":"Lightning.SetupUtils.html#add_and_update_runs/3"},{"type":"function","title":"Lightning.SetupUtils.create_dhis2_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_dhis2_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_openhie_project/1","doc":"","ref":"Lightning.SetupUtils.html#create_openhie_project/1"},{"type":"function","title":"Lightning.SetupUtils.create_starter_project/2","doc":"","ref":"Lightning.SetupUtils.html#create_starter_project/2"},{"type":"function","title":"Lightning.SetupUtils.setup_demo/1","doc":"Creates initial data and returns the created records.","ref":"Lightning.SetupUtils.html#setup_demo/1"},{"type":"function","title":"Lightning.SetupUtils.tear_down/1","doc":"","ref":"Lightning.SetupUtils.html#tear_down/1"},{"type":"module","title":"Lightning.TaskWorker","doc":"A TaskWorker with concurrency limits. A simple concurrency limiter that wraps Task.Supervisor , which already does have the ability to specify max_children ; it throws an error when that limit is exceeded. To use it, start it like any other process; ideally in your supervision tree. ... , { Lightning.TaskWorker , name : :cli_task_worker , max_tasks : 4 } Options :max_tasks Defaults to the number of system schedulers available to the vm.","ref":"Lightning.TaskWorker.html"},{"type":"function","title":"Lightning.TaskWorker.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.TaskWorker.html#child_spec/1"},{"type":"function","title":"Lightning.TaskWorker.get_status/1","doc":"","ref":"Lightning.TaskWorker.html#get_status/1"},{"type":"function","title":"Lightning.TaskWorker.start_link/1","doc":"","ref":"Lightning.TaskWorker.html#start_link/1"},{"type":"function","title":"Lightning.TaskWorker.start_task/2","doc":"","ref":"Lightning.TaskWorker.html#start_task/2"},{"type":"module","title":"Lightning.Validators","doc":"Extra validators for Ecto.Changeset.","ref":"Lightning.Validators.html"},{"type":"function","title":"Lightning.Validators.validate_exclusive/3","doc":"Validate that only one of the fields is set at a time. Example: changeset |> validate_exclusive ( [ :source_job_id , :source_trigger_id ] , "source_job_id and source_trigger_id are mutually exclusive" )","ref":"Lightning.Validators.html#validate_exclusive/3"},{"type":"function","title":"Lightning.Validators.validate_one_required/3","doc":"Validate that at least one of the fields is set.","ref":"Lightning.Validators.html#validate_one_required/3"},{"type":"module","title":"Lightning.Vault","doc":"Module for handling the encryption and decryption of database fields.","ref":"Lightning.Vault.html"},{"type":"function","title":"Lightning.Vault.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Vault.html#child_spec/1"},{"type":"function","title":"Lightning.Vault.start_link/1","doc":"","ref":"Lightning.Vault.html#start_link/1"},{"type":"module","title":"Lightning.WorkOrder","doc":"Ecto model for Workorders.","ref":"Lightning.WorkOrder.html"},{"type":"function","title":"Lightning.WorkOrder.new/0","doc":"","ref":"Lightning.WorkOrder.html#new/0"},{"type":"type","title":"Lightning.WorkOrder.t/0","doc":"","ref":"Lightning.WorkOrder.html#t:t/0"},{"type":"module","title":"Lightning.WorkOrderService","doc":"The WorkOrderService.","ref":"Lightning.WorkOrderService.html"},{"type":"function","title":"Lightning.WorkOrderService.attempt_updated/1","doc":"","ref":"Lightning.WorkOrderService.html#attempt_updated/1"},{"type":"function","title":"Lightning.WorkOrderService.build/2","doc":"","ref":"Lightning.WorkOrderService.html#build/2"},{"type":"function","title":"Lightning.WorkOrderService.create_manual_workorder/3","doc":"","ref":"Lightning.WorkOrderService.html#create_manual_workorder/3"},{"type":"function","title":"Lightning.WorkOrderService.create_webhook_workorder/2","doc":"","ref":"Lightning.WorkOrderService.html#create_webhook_workorder/2"},{"type":"function","title":"Lightning.WorkOrderService.create_work_order/1","doc":"Creates a work_order. Examples iex> create_work_order ( %{ field : value } ) { :ok , % WorkOrder { } } iex> create_work_order ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.WorkOrderService.html#create_work_order/1"},{"type":"function","title":"Lightning.WorkOrderService.multi_for/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for/3"},{"type":"function","title":"Lightning.WorkOrderService.multi_for_manual/3","doc":"","ref":"Lightning.WorkOrderService.html#multi_for_manual/3"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_run/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_run/2"},{"type":"function","title":"Lightning.WorkOrderService.retry_attempt_runs/2","doc":"","ref":"Lightning.WorkOrderService.html#retry_attempt_runs/2"},{"type":"function","title":"Lightning.WorkOrderService.subscribe/1","doc":"","ref":"Lightning.WorkOrderService.html#subscribe/1"},{"type":"module","title":"Lightning.Workflows","doc":"The Workflows context.","ref":"Lightning.Workflows.html"},{"type":"function","title":"Lightning.Workflows.build_trigger/1","doc":"Builds a Trigger","ref":"Lightning.Workflows.html#build_trigger/1"},{"type":"function","title":"Lightning.Workflows.change_workflow/2","doc":"Returns an %Ecto.Changeset{} for tracking workflow changes. Examples iex> change_workflow ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#change_workflow/2"},{"type":"function","title":"Lightning.Workflows.create_edge/1","doc":"Creates an edge","ref":"Lightning.Workflows.html#create_edge/1"},{"type":"function","title":"Lightning.Workflows.create_workflow/1","doc":"Creates a workflow. Examples iex> create_workflow ( %{ field : value } ) { :ok , % Workflow { } } iex> create_workflow ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#create_workflow/1"},{"type":"function","title":"Lightning.Workflows.delete_workflow/1","doc":"Deletes a workflow. Examples iex> delete_workflow ( workflow ) { :ok , % Workflow { } } iex> delete_workflow ( workflow ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#delete_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_edge_by_webhook/1","doc":"Gets a Single Edge by it's webhook trigger.","ref":"Lightning.Workflows.html#get_edge_by_webhook/1"},{"type":"function","title":"Lightning.Workflows.get_edges_for_cron_execution/1","doc":"Returns a list of edges with jobs to execute, given a current timestamp in Unix. This is used by the scheduler, which calls this function once every minute.","ref":"Lightning.Workflows.html#get_edges_for_cron_execution/1"},{"type":"function","title":"Lightning.Workflows.get_workflow/1","doc":"","ref":"Lightning.Workflows.html#get_workflow/1"},{"type":"function","title":"Lightning.Workflows.get_workflow!/1","doc":"Gets a single workflow. Raises Ecto.NoResultsError if the Workflow does not exist. Examples iex> get_workflow! ( 123 ) % Workflow { } iex> get_workflow! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Workflows.html#get_workflow!/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for/1","doc":"Retrieves a list of Workflows with their jobs and triggers preloaded.","ref":"Lightning.Workflows.html#get_workflows_for/1"},{"type":"function","title":"Lightning.Workflows.get_workflows_for_query/1","doc":"","ref":"Lightning.Workflows.html#get_workflows_for_query/1"},{"type":"function","title":"Lightning.Workflows.list_workflows/0","doc":"Returns the list of workflows. Examples iex> list_workflows ( ) [ % Workflow { } , ... ]","ref":"Lightning.Workflows.html#list_workflows/0"},{"type":"function","title":"Lightning.Workflows.mark_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the workflow request_deletion. Examples iex> change_request_deletion ( workflow ) % Ecto.Changeset { data : % Workflow { } }","ref":"Lightning.Workflows.html#mark_for_deletion/2"},{"type":"function","title":"Lightning.Workflows.to_project_space/1","doc":"","ref":"Lightning.Workflows.html#to_project_space/1"},{"type":"function","title":"Lightning.Workflows.update_trigger/2","doc":"Updates a trigger","ref":"Lightning.Workflows.html#update_trigger/2"},{"type":"function","title":"Lightning.Workflows.update_workflow/2","doc":"Updates a workflow. Examples iex> update_workflow ( workflow , %{ field : new_value } ) { :ok , % Workflow { } } iex> update_workflow ( workflow , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Workflows.html#update_workflow/2"},{"type":"module","title":"Lightning.Workflows.Edge","doc":"Ecto model for Workflow Edges. A Workflow Edge represents a connection between two jobs (or a trigger and a job) in a workflow. The source of the edge is either a job or a trigger. The target of the edge is always a job.","ref":"Lightning.Workflows.Edge.html"},{"type":"function","title":"Lightning.Workflows.Edge.changeset/2","doc":"","ref":"Lightning.Workflows.Edge.html#changeset/2"},{"type":"function","title":"Lightning.Workflows.Edge.new/1","doc":"","ref":"Lightning.Workflows.Edge.html#new/1"},{"type":"function","title":"Lightning.Workflows.Edge.validate/1","doc":"","ref":"Lightning.Workflows.Edge.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Edge.edge_condition/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:edge_condition/0"},{"type":"type","title":"Lightning.Workflows.Edge.t/0","doc":"","ref":"Lightning.Workflows.Edge.html#t:t/0"},{"type":"module","title":"Lightning.Workflows.Graph","doc":"Utility to construct and manipulate a graph/plan made out of Jobs","ref":"Lightning.Workflows.Graph.html"},{"type":"function","title":"Lightning.Workflows.Graph.new/1","doc":"","ref":"Lightning.Workflows.Graph.html#new/1"},{"type":"function","title":"Lightning.Workflows.Graph.remove/2","doc":"","ref":"Lightning.Workflows.Graph.html#remove/2"},{"type":"function","title":"Lightning.Workflows.Graph.vertices/1","doc":"","ref":"Lightning.Workflows.Graph.html#vertices/1"},{"type":"type","title":"Lightning.Workflows.Graph.t/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:t/0"},{"type":"type","title":"Lightning.Workflows.Graph.vertex/0","doc":"","ref":"Lightning.Workflows.Graph.html#t:vertex/0"},{"type":"module","title":"Lightning.Workflows.Workflow","doc":"Ecto model for Workflows. A Workflow contains the fields for defining a workflow. name A plain text identifier","ref":"Lightning.Workflows.Workflow.html"},{"type":"function","title":"Lightning.Workflows.Workflow.request_deletion_changeset/2","doc":"","ref":"Lightning.Workflows.Workflow.html#request_deletion_changeset/2"},{"type":"function","title":"Lightning.Workflows.Workflow.validate/1","doc":"","ref":"Lightning.Workflows.Workflow.html#validate/1"},{"type":"type","title":"Lightning.Workflows.Workflow.t/0","doc":"","ref":"Lightning.Workflows.Workflow.html#t:t/0"},{"type":"module","title":"Lightning.Workorders.SearchParams","doc":"This module is used to parse search parameters for workorders and provide a query to the database.","ref":"Lightning.Workorders.SearchParams.html"},{"type":"function","title":"Lightning.Workorders.SearchParams.new/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#new/1"},{"type":"function","title":"Lightning.Workorders.SearchParams.to_uri_params/1","doc":"","ref":"Lightning.Workorders.SearchParams.html#to_uri_params/1"},{"type":"type","title":"Lightning.Workorders.SearchParams.t/0","doc":"","ref":"Lightning.Workorders.SearchParams.html#t:t/0"},{"type":"module","title":"LightningWeb","doc":"The entrypoint for defining your web interface, such as controllers, views, channels and so on. This can be used in your application as: use LightningWeb , :controller use LightningWeb , :view The definitions below will be executed for every view, controller, etc, so keep them short and clean, focused on imports, uses and aliases. Do NOT define functions inside the quoted expressions below. Instead, define any helper function in modules and import those modules here.","ref":"LightningWeb.html"},{"type":"macro","title":"LightningWeb.__using__/1","doc":"When used, dispatch to the appropriate controller/view/etc.","ref":"LightningWeb.html#__using__/1"},{"type":"function","title":"LightningWeb.channel/0","doc":"","ref":"LightningWeb.html#channel/0"},{"type":"function","title":"LightningWeb.component/0","doc":"","ref":"LightningWeb.html#component/0"},{"type":"function","title":"LightningWeb.controller/0","doc":"","ref":"LightningWeb.html#controller/0"},{"type":"function","title":"LightningWeb.html/0","doc":"","ref":"LightningWeb.html#html/0"},{"type":"function","title":"LightningWeb.live_component/0","doc":"","ref":"LightningWeb.html#live_component/0"},{"type":"function","title":"LightningWeb.live_view/1","doc":"","ref":"LightningWeb.html#live_view/1"},{"type":"function","title":"LightningWeb.router/0","doc":"","ref":"LightningWeb.html#router/0"},{"type":"function","title":"LightningWeb.static_paths/0","doc":"","ref":"LightningWeb.html#static_paths/0"},{"type":"function","title":"LightningWeb.verified_routes/0","doc":"","ref":"LightningWeb.html#verified_routes/0"},{"type":"function","title":"LightningWeb.view/0","doc":"","ref":"LightningWeb.html#view/0"},{"type":"module","title":"LightningWeb.API.Helpers","doc":"Helpers for the API views","ref":"LightningWeb.API.Helpers.html"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_link/3","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_link/3"},{"type":"function","title":"LightningWeb.API.Helpers.pagination_links/2","doc":"","ref":"LightningWeb.API.Helpers.html#pagination_links/2"},{"type":"function","title":"LightningWeb.API.Helpers.url_for/2","doc":"","ref":"LightningWeb.API.Helpers.html#url_for/2"},{"type":"module","title":"LightningWeb.API.JobController","doc":"","ref":"LightningWeb.API.JobController.html"},{"type":"function","title":"LightningWeb.API.JobController.index/2","doc":"","ref":"LightningWeb.API.JobController.html#index/2"},{"type":"function","title":"LightningWeb.API.JobController.show/2","doc":"","ref":"LightningWeb.API.JobController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProjectController","doc":"","ref":"LightningWeb.API.ProjectController.html"},{"type":"function","title":"LightningWeb.API.ProjectController.index/2","doc":"","ref":"LightningWeb.API.ProjectController.html#index/2"},{"type":"function","title":"LightningWeb.API.ProjectController.show/2","doc":"","ref":"LightningWeb.API.ProjectController.html#show/2"},{"type":"module","title":"LightningWeb.API.ProvisioningController","doc":"","ref":"LightningWeb.API.ProvisioningController.html"},{"type":"function","title":"LightningWeb.API.ProvisioningController.create/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#create/2"},{"type":"function","title":"LightningWeb.API.ProvisioningController.show/2","doc":"","ref":"LightningWeb.API.ProvisioningController.html#show/2"},{"type":"module","title":"LightningWeb.API.RunController","doc":"","ref":"LightningWeb.API.RunController.html"},{"type":"function","title":"LightningWeb.API.RunController.index/2","doc":"","ref":"LightningWeb.API.RunController.html#index/2"},{"type":"function","title":"LightningWeb.API.RunController.show/2","doc":"","ref":"LightningWeb.API.RunController.html#show/2"},{"type":"module","title":"LightningWeb.AuditLive.Index","doc":"LiveView for listing Audit events","ref":"LightningWeb.AuditLive.Index.html"},{"type":"function","title":"LightningWeb.AuditLive.Index.diff/1","doc":"","ref":"LightningWeb.AuditLive.Index.html#diff/1"},{"type":"function","title":"LightningWeb.AuditLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.AuditLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.AuthProvidersLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.AuthProvidersLive.FormComponent.html"},{"type":"module","title":"LightningWeb.AuthProvidersLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.AuthProvidersLive.Index.html"},{"type":"module","title":"LightningWeb.ChangesetJSON","doc":"Renders changesets as JSON.","ref":"LightningWeb.ChangesetJSON.html"},{"type":"function","title":"LightningWeb.ChangesetJSON.error/1","doc":"","ref":"LightningWeb.ChangesetJSON.html#error/1"},{"type":"module","title":"LightningWeb.ChangesetView","doc":"","ref":"LightningWeb.ChangesetView.html"},{"type":"function","title":"LightningWeb.ChangesetView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ChangesetView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ChangesetView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ChangesetView.html#render/2"},{"type":"function","title":"LightningWeb.ChangesetView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ChangesetView.html#template_not_found/2"},{"type":"function","title":"LightningWeb.ChangesetView.translate_errors/1","doc":"Traverses and translates changeset errors. See Ecto.Changeset.traverse_errors/2 and LightningWeb.ErrorHelpers.translate_error/1 for more details.","ref":"LightningWeb.ChangesetView.html#translate_errors/1"},{"type":"module","title":"LightningWeb.Components.Modal","doc":"A modal component that can be used to display a modal on the page. This currently isn't used anywhere but should be used in the future to replace the existing modal implementations.","ref":"LightningWeb.Components.Modal.html"},{"type":"module","title":"LightningWeb.CredentialLive.Edit","doc":"LiveView for editing a single Credential, which inturn uses LightningWeb.CredentialLive.FormComponent for common functionality.","ref":"LightningWeb.CredentialLive.Edit.html"},{"type":"function","title":"LightningWeb.CredentialLive.Edit.handle_info/2","doc":"A generic handler for forwarding updates from PubSub","ref":"LightningWeb.CredentialLive.Edit.html#handle_info/2"},{"type":"module","title":"LightningWeb.CredentialLive.FormComponent","doc":"Form Component for working with a single Credential","ref":"LightningWeb.CredentialLive.FormComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.credential_transfer/1","doc":"Attributes users ( :list ) (required) form ( :map ) (required)","ref":"LightningWeb.CredentialLive.FormComponent.html#credential_transfer/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.form_component/1","doc":"Switcher components for different types of credentials. Attributes type ( :string ) (required) form ( :map ) (required) update_body ( :any ) Slots inner_block","ref":"LightningWeb.CredentialLive.FormComponent.html#form_component/1"},{"type":"function","title":"LightningWeb.CredentialLive.FormComponent.project_credentials/1","doc":"Attributes form ( :map ) (required) projects ( :list ) (required) selected ( :map ) (required) phx_target ( :any ) - Defaults to nil .","ref":"LightningWeb.CredentialLive.FormComponent.html#project_credentials/1"},{"type":"module","title":"LightningWeb.CredentialLive.GoogleSheetsComponent","doc":"Form component to setup a Google Sheets component. This component has several moving parts: Subscribes to a PubSub topic specially link to the component id See: LightningWeb.OauthCredentialHelper . Uses the Lightning.Google module to set up an OAuth client for generating urls, exchanging the code and requesting a new access_token . The flow for creating a new token is: Generate an authorization link which contains: The authorization url from the Google client with the applications callback_url A state string that is an encrypted set of data with the components module and id in it Once the user authorizes the client the callback is requested with a code The LightningWeb.OidcController decodes the state returned to it and does a 'broadcast_forward' which is simply a message expected to be received by a LiveView and applied to Phoenix.LiveView.send_update/3 . The component receives the code and requests a token. Any changes to the token (Credential body) are still handled by the parent component and so a update_body function is passed in to send params changes back up to update the form.","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.disabled_authorize_button/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#disabled_authorize_button/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.error_block/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#error_block/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.fieldset/1","doc":"Attributes form ( :map ) (required) update_body ( :any ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.GoogleSheetsComponent.userinfo/1","doc":"","ref":"LightningWeb.CredentialLive.GoogleSheetsComponent.html#userinfo/1"},{"type":"module","title":"LightningWeb.CredentialLive.Index","doc":"LiveView for listing and managing credentials","ref":"LightningWeb.CredentialLive.Index.html"},{"type":"function","title":"LightningWeb.CredentialLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.CredentialLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#fieldset/1"},{"type":"function","title":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.schema_input/1","doc":"Attributes form ( :map ) (required) schema ( :map ) (required) field ( :any ) (required)","ref":"LightningWeb.CredentialLive.JsonSchemaBodyComponent.html#schema_input/1"},{"type":"module","title":"LightningWeb.CredentialLive.RawBodyComponent","doc":"","ref":"LightningWeb.CredentialLive.RawBodyComponent.html"},{"type":"function","title":"LightningWeb.CredentialLive.RawBodyComponent.fieldset/1","doc":"Attributes form ( :map ) (required) Slots inner_block","ref":"LightningWeb.CredentialLive.RawBodyComponent.html#fieldset/1"},{"type":"module","title":"LightningWeb.CredentialLive.TypePicker","doc":"","ref":"LightningWeb.CredentialLive.TypePicker.html"},{"type":"module","title":"LightningWeb.DataclipLive.Edit","doc":"LiveView for editing a single dataclip.","ref":"LightningWeb.DataclipLive.Edit.html"},{"type":"function","title":"LightningWeb.DataclipLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.FormComponent","doc":"Form Component for working with a single dataclip","ref":"LightningWeb.DataclipLive.FormComponent.html"},{"type":"function","title":"LightningWeb.DataclipLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.DataclipLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.DataclipLive.Index","doc":"LiveView for listing and working with a list of Dataclips","ref":"LightningWeb.DataclipLive.Index.html"},{"type":"function","title":"LightningWeb.DataclipLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.DataclipLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.DownloadsController","doc":"","ref":"LightningWeb.DownloadsController.html"},{"type":"function","title":"LightningWeb.DownloadsController.download_project_yaml/2","doc":"","ref":"LightningWeb.DownloadsController.html#download_project_yaml/2"},{"type":"module","title":"LightningWeb.Endpoint","doc":"","ref":"LightningWeb.Endpoint.html"},{"type":"function","title":"LightningWeb.Endpoint.broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast/3 .","ref":"LightningWeb.Endpoint.html#broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast!/3","doc":"Callback implementation for Phoenix.Endpoint.broadcast!/3 .","ref":"LightningWeb.Endpoint.html#broadcast!/3"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.broadcast_from!/4","doc":"Callback implementation for Phoenix.Endpoint.broadcast_from!/4 .","ref":"LightningWeb.Endpoint.html#broadcast_from!/4"},{"type":"function","title":"LightningWeb.Endpoint.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.Endpoint.html#call/2"},{"type":"function","title":"LightningWeb.Endpoint.child_spec/1","doc":"Returns the child specification to start the endpoint under a supervision tree.","ref":"LightningWeb.Endpoint.html#child_spec/1"},{"type":"function","title":"LightningWeb.Endpoint.config/2","doc":"Returns the endpoint configuration for key Returns default if the key does not exist.","ref":"LightningWeb.Endpoint.html#config/2"},{"type":"function","title":"LightningWeb.Endpoint.config_change/2","doc":"Reloads the configuration given the application environment changes.","ref":"LightningWeb.Endpoint.html#config_change/2"},{"type":"function","title":"LightningWeb.Endpoint.host/0","doc":"Returns the host for the given endpoint.","ref":"LightningWeb.Endpoint.html#host/0"},{"type":"function","title":"LightningWeb.Endpoint.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.Endpoint.html#init/1"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast/3","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast/3 .","ref":"LightningWeb.Endpoint.html#local_broadcast/3"},{"type":"function","title":"LightningWeb.Endpoint.local_broadcast_from/4","doc":"Callback implementation for Phoenix.Endpoint.local_broadcast_from/4 .","ref":"LightningWeb.Endpoint.html#local_broadcast_from/4"},{"type":"function","title":"LightningWeb.Endpoint.path/1","doc":"Generates the path information when routing to this endpoint.","ref":"LightningWeb.Endpoint.html#path/1"},{"type":"function","title":"LightningWeb.Endpoint.script_name/0","doc":"Generates the script name.","ref":"LightningWeb.Endpoint.html#script_name/0"},{"type":"function","title":"LightningWeb.Endpoint.start_link/1","doc":"Starts the endpoint supervision tree. All other options are merged into the endpoint configuration.","ref":"LightningWeb.Endpoint.html#start_link/1"},{"type":"function","title":"LightningWeb.Endpoint.static_integrity/1","doc":"Generates a base64-encoded cryptographic hash (sha512) to a static file in priv/static . Meant to be used for Subresource Integrity with CDNs.","ref":"LightningWeb.Endpoint.html#static_integrity/1"},{"type":"function","title":"LightningWeb.Endpoint.static_lookup/1","doc":"Returns a two item tuple with the first item being the static_path and the second item being the static_integrity .","ref":"LightningWeb.Endpoint.html#static_lookup/1"},{"type":"function","title":"LightningWeb.Endpoint.static_path/1","doc":"Generates a route to a static file in priv/static .","ref":"LightningWeb.Endpoint.html#static_path/1"},{"type":"function","title":"LightningWeb.Endpoint.static_url/0","doc":"Generates the static URL without any path information. It uses the configuration under :static_url to generate such. It falls back to :url if :static_url is not set.","ref":"LightningWeb.Endpoint.html#static_url/0"},{"type":"function","title":"LightningWeb.Endpoint.struct_url/0","doc":"Generates the endpoint base URL but as a URI struct. It uses the configuration under :url to generate such. Useful for manipulating the URL data and passing it to URL helpers.","ref":"LightningWeb.Endpoint.html#struct_url/0"},{"type":"function","title":"LightningWeb.Endpoint.subscribe/2","doc":"Callback implementation for Phoenix.Endpoint.subscribe/2 .","ref":"LightningWeb.Endpoint.html#subscribe/2"},{"type":"function","title":"LightningWeb.Endpoint.unsubscribe/1","doc":"Callback implementation for Phoenix.Endpoint.unsubscribe/1 .","ref":"LightningWeb.Endpoint.html#unsubscribe/1"},{"type":"function","title":"LightningWeb.Endpoint.url/0","doc":"Generates the endpoint base URL without any path information. It uses the configuration under :url to generate such.","ref":"LightningWeb.Endpoint.html#url/0"},{"type":"module","title":"LightningWeb.ErrorView","doc":"","ref":"LightningWeb.ErrorView.html"},{"type":"function","title":"LightningWeb.ErrorView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.ErrorView.html#__resource__/0"},{"type":"function","title":"LightningWeb.ErrorView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.ErrorView.html#render/2"},{"type":"function","title":"LightningWeb.ErrorView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.ErrorView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.FallbackController","doc":"Translates controller action results into valid Plug.Conn responses. See Phoenix.Controller.action_fallback/1 for more details.","ref":"LightningWeb.FallbackController.html"},{"type":"module","title":"LightningWeb.FirstSetupLive.Superuser","doc":"Superuser setup liveview Allows the creation on the first user in the system. It has only one action: :show","ref":"LightningWeb.FirstSetupLive.Superuser.html"},{"type":"function","title":"LightningWeb.FirstSetupLive.Superuser.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.FirstSetupLive.Superuser.html#render/1"},{"type":"module","title":"LightningWeb.FormHelpers","doc":"Conveniences for building forms.","ref":"LightningWeb.FormHelpers.html"},{"type":"module","title":"LightningWeb.Gettext","doc":"A module providing Internationalization with a gettext-based API. By using Gettext , your module gains a set of macros for translations, for example: import LightningWeb.Gettext # Simple translation gettext ( "Here is the string to translate" ) # Plural translation ngettext ( "Here is the string to translate" , "Here are the strings to translate" , 3 ) # Domain-based translation dgettext ( "errors" , "Here is the error message to translate" ) See the Gettext Docs for detailed usage.","ref":"LightningWeb.Gettext.html"},{"type":"macro","title":"LightningWeb.Gettext.dgettext/3","doc":"Callback implementation for Gettext.Backend.dgettext/3 .","ref":"LightningWeb.Gettext.html#dgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.dgettext_noop/2","doc":"Callback implementation for Gettext.Backend.dgettext_noop/2 .","ref":"LightningWeb.Gettext.html#dgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.dngettext/5","doc":"Callback implementation for Gettext.Backend.dngettext/5 .","ref":"LightningWeb.Gettext.html#dngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.dngettext_noop/3","doc":"Callback implementation for Gettext.Backend.dngettext_noop/3 .","ref":"LightningWeb.Gettext.html#dngettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext/4","doc":"Callback implementation for Gettext.Backend.dpgettext/4 .","ref":"LightningWeb.Gettext.html#dpgettext/4"},{"type":"macro","title":"LightningWeb.Gettext.dpgettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#dpgettext_noop/3"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext/6","doc":"Callback implementation for Gettext.Backend.dpngettext/6 .","ref":"LightningWeb.Gettext.html#dpngettext/6"},{"type":"macro","title":"LightningWeb.Gettext.dpngettext_noop/4","doc":"","ref":"LightningWeb.Gettext.html#dpngettext_noop/4"},{"type":"macro","title":"LightningWeb.Gettext.gettext/2","doc":"Callback implementation for Gettext.Backend.gettext/2 .","ref":"LightningWeb.Gettext.html#gettext/2"},{"type":"macro","title":"LightningWeb.Gettext.gettext_comment/1","doc":"Callback implementation for Gettext.Backend.gettext_comment/1 .","ref":"LightningWeb.Gettext.html#gettext_comment/1"},{"type":"macro","title":"LightningWeb.Gettext.gettext_noop/1","doc":"Callback implementation for Gettext.Backend.gettext_noop/1 .","ref":"LightningWeb.Gettext.html#gettext_noop/1"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_bindings/2","doc":"Callback implementation for Gettext.Backend.handle_missing_bindings/2 .","ref":"LightningWeb.Gettext.html#handle_missing_bindings/2"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_plural_translation/7","doc":"Callback implementation for Gettext.Backend.handle_missing_plural_translation/7 .","ref":"LightningWeb.Gettext.html#handle_missing_plural_translation/7"},{"type":"function","title":"LightningWeb.Gettext.handle_missing_translation/5","doc":"Callback implementation for Gettext.Backend.handle_missing_translation/5 .","ref":"LightningWeb.Gettext.html#handle_missing_translation/5"},{"type":"function","title":"LightningWeb.Gettext.lgettext/5","doc":"","ref":"LightningWeb.Gettext.html#lgettext/5"},{"type":"function","title":"LightningWeb.Gettext.lngettext/7","doc":"","ref":"LightningWeb.Gettext.html#lngettext/7"},{"type":"macro","title":"LightningWeb.Gettext.ngettext/4","doc":"Callback implementation for Gettext.Backend.ngettext/4 .","ref":"LightningWeb.Gettext.html#ngettext/4"},{"type":"macro","title":"LightningWeb.Gettext.ngettext_noop/2","doc":"Callback implementation for Gettext.Backend.ngettext_noop/2 .","ref":"LightningWeb.Gettext.html#ngettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pgettext/3","doc":"Callback implementation for Gettext.Backend.pgettext/3 .","ref":"LightningWeb.Gettext.html#pgettext/3"},{"type":"macro","title":"LightningWeb.Gettext.pgettext_noop/2","doc":"","ref":"LightningWeb.Gettext.html#pgettext_noop/2"},{"type":"macro","title":"LightningWeb.Gettext.pngettext/5","doc":"Callback implementation for Gettext.Backend.pngettext/5 .","ref":"LightningWeb.Gettext.html#pngettext/5"},{"type":"macro","title":"LightningWeb.Gettext.pngettext_noop/3","doc":"","ref":"LightningWeb.Gettext.html#pngettext_noop/3"},{"type":"module","title":"LightningWeb.HealthCheck","doc":"","ref":"LightningWeb.HealthCheck.html"},{"type":"function","title":"LightningWeb.HealthCheck.call/2","doc":"Callback implementation for Plug.call/2 .","ref":"LightningWeb.HealthCheck.html#call/2"},{"type":"function","title":"LightningWeb.HealthCheck.init/1","doc":"Callback implementation for Plug.init/1 .","ref":"LightningWeb.HealthCheck.html#init/1"},{"type":"module","title":"LightningWeb.Hooks","doc":"LiveView Hooks","ref":"LightningWeb.Hooks.html"},{"type":"function","title":"LightningWeb.Hooks.on_mount/4","doc":"Finds and assigns a project to the socket, if a user doesn't have access they are redirected and shown a 'No Access' screen via a :nav flash message. There is a fallthru function, when there is no project_id in the params - this is for liveviews that may or may not have a project_id depending on usage - like DashboardLive .","ref":"LightningWeb.Hooks.html#on_mount/4"},{"type":"module","title":"LightningWeb.InitAssigns","doc":"Ensures common assigns are applied to all LiveViews attaching this hook.","ref":"LightningWeb.InitAssigns.html"},{"type":"function","title":"LightningWeb.InitAssigns.on_mount/4","doc":"","ref":"LightningWeb.InitAssigns.html#on_mount/4"},{"type":"module","title":"LightningWeb.JobLive.AdaptorPicker","doc":"Component allowing selecting an adaptor and it's version","ref":"LightningWeb.JobLive.AdaptorPicker.html"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.display_name_for_adaptor/1","doc":"Converts standard adaptor names into "label","value" lists and returns non-standard names as merely "value"; both can be passed directly into a select option list.","ref":"LightningWeb.JobLive.AdaptorPicker.html#display_name_for_adaptor/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.get_adaptor_version_options/1","doc":"","ref":"LightningWeb.JobLive.AdaptorPicker.html#get_adaptor_version_options/1"},{"type":"function","title":"LightningWeb.JobLive.AdaptorPicker.render/1","doc":"Attributes form ( :map ) (required) on_change ( :any ) - Defaults to nil . disabled ( :boolean ) - Defaults to false .","ref":"LightningWeb.JobLive.AdaptorPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CredentialPicker","doc":"Component allowing selecting a credential or creating a new one via a modal.","ref":"LightningWeb.JobLive.CredentialPicker.html"},{"type":"function","title":"LightningWeb.JobLive.CredentialPicker.render/1","doc":"Attributes form ( :map ) (required) disabled ( :boolean ) - Defaults to false . credentials ( :list ) (required) on_change ( :any ) - Defaults to nil .","ref":"LightningWeb.JobLive.CredentialPicker.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.CronSetupComponent","doc":"A live component for managing cron setup in a form. The CronSetupComponent provides an interactive form for configuring cron settings. It includes fields for specifying the frequency, minute, hour, weekday, monthday, and cron expression. Usage Include the CronSetupComponent in your live view or template. Pass the necessary assigns to the component, such as form , on_change , and disabled . Handle the cron_expression_change event to capture changes in the form inputs. Use the updated cron_expression in your application logic.","ref":"LightningWeb.JobLive.CronSetupComponent.html"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.build_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#build_cron_expression/2"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.frequency_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#frequency_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.get_cron_data/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#get_cron_data/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.hour_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#hour_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.minute_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#minute_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.monthday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#monthday_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.render/1","doc":"Attributes id ( :string ) (required) form ( :map ) (required) on_change ( :any ) (required) disabled ( :boolean ) (required)","ref":"LightningWeb.JobLive.CronSetupComponent.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.time_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#time_field/1"},{"type":"function","title":"LightningWeb.JobLive.CronSetupComponent.weekday_field/1","doc":"","ref":"LightningWeb.JobLive.CronSetupComponent.html#weekday_field/1"},{"type":"module","title":"LightningWeb.JobLive.JobBuilder","doc":"Job Builder Panel","ref":"LightningWeb.JobLive.JobBuilder.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.delete_title/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#delete_title/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.follow_run/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#follow_run/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.render/1","doc":"Attributes return_to ( :string ) (required) params ( :map ) - Defaults to %{} . can_edit_job ( :boolean ) (required) can_run_job ( :boolean ) (required)","ref":"LightningWeb.JobLive.JobBuilder.html#render/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_adaptor/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_adaptor/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.send_credential/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#send_credential/2"},{"type":"function","title":"LightningWeb.JobLive.JobBuilder.update_cron_expression/2","doc":"","ref":"LightningWeb.JobLive.JobBuilder.html#update_cron_expression/2"},{"type":"module","title":"LightningWeb.JobLive.JobBuilderComponents","doc":"","ref":"LightningWeb.JobLive.JobBuilderComponents.html"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.job_editor_component/1","doc":"Attributes adaptor ( :string ) (required) disabled ( :boolean ) - Defaults to false . source ( :string ) (required) change_event ( :string ) - Defaults to "job_body_changed" . Global attributes are accepted.","ref":"LightningWeb.JobLive.JobBuilderComponents.html#job_editor_component/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.trigger_picker/1","doc":"Attributes form ( :map ) (required) upstream_jobs ( :list ) (required) on_cron_change ( :any ) (required) disabled ( :boolean ) - Defaults to true .","ref":"LightningWeb.JobLive.JobBuilderComponents.html#trigger_picker/1"},{"type":"function","title":"LightningWeb.JobLive.JobBuilderComponents.when_invalid/1","doc":"Attributes changeset ( :map ) (required) field ( :atom ) (required) Slots inner_block (required)","ref":"LightningWeb.JobLive.JobBuilderComponents.html#when_invalid/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.render/1","doc":"Attributes job ( :map ) (required) on_run ( :any ) (required) user ( :map ) (required) selected_dataclip_id ( :string ) (required)","ref":"LightningWeb.JobLive.ManualRunComponent.html#render/1"},{"type":"module","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html"},{"type":"function","title":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.changeset/2","doc":"","ref":"LightningWeb.JobLive.ManualRunComponent.ManualWorkorder.html#changeset/2"},{"type":"module","title":"LightningWeb.LiveHelpers","doc":"General purpose LiveView helper functions","ref":"LightningWeb.LiveHelpers.html"},{"type":"function","title":"LightningWeb.LiveHelpers.display_short_uuid/1","doc":"","ref":"LightningWeb.LiveHelpers.html#display_short_uuid/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_in/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_in/1"},{"type":"function","title":"LightningWeb.LiveHelpers.fade_out/1","doc":"","ref":"LightningWeb.LiveHelpers.html#fade_out/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_error_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_error_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_info_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_info_block/1"},{"type":"function","title":"LightningWeb.LiveHelpers.live_nav_block/1","doc":"","ref":"LightningWeb.LiveHelpers.html#live_nav_block/1"},{"type":"module","title":"LightningWeb.ModalPortal","doc":"Component for rendering content inside layout without full DOM patch.","ref":"LightningWeb.ModalPortal.html"},{"type":"function","title":"LightningWeb.ModalPortal.close_modal/0","doc":"","ref":"LightningWeb.ModalPortal.html#close_modal/0"},{"type":"function","title":"LightningWeb.ModalPortal.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.ModalPortal.html#handle_event/3"},{"type":"function","title":"LightningWeb.ModalPortal.on_hide/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_hide/2"},{"type":"function","title":"LightningWeb.ModalPortal.on_show/2","doc":"","ref":"LightningWeb.ModalPortal.html#on_show/2"},{"type":"function","title":"LightningWeb.ModalPortal.open_modal/2","doc":"","ref":"LightningWeb.ModalPortal.html#open_modal/2"},{"type":"function","title":"LightningWeb.ModalPortal.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ModalPortal.html#render/1"},{"type":"function","title":"LightningWeb.ModalPortal.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.ModalPortal.html#update/2"},{"type":"module","title":"LightningWeb.OauthCredentialHelper","doc":"A set of helper functions to encodes state and coordinate OAuth callbacks back to a LiveView component.","ref":"LightningWeb.OauthCredentialHelper.html"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast/2","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#broadcast/2"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.broadcast_forward/3","doc":"Broadcast a message specifically for forwarding a message to a component. It expects a subscription_id , the module of the component and opts being a keyword list containing an :id key of the specific component. See: Phoenix.LiveView.send_update/3 for more info. A corresponding LiveView (that is subscribed) is expected to have a matching handle_info/2 that looks like this: def handle_info ( { :forward , mod , opts } , socket ) do send_update ( mod , opts ) { :noreply , socket } end","ref":"LightningWeb.OauthCredentialHelper.html#broadcast_forward/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.build_state/3","doc":"Encode and encrypt the callback data which will be sent so a provider as the state key in the request. The values are: subscription_id The same ID used to subscribe. The component module The LiveView component that is going to receive update The component id The ID of the component","ref":"LightningWeb.OauthCredentialHelper.html#build_state/3"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.decode_state/1","doc":"","ref":"LightningWeb.OauthCredentialHelper.html#decode_state/1"},{"type":"function","title":"LightningWeb.OauthCredentialHelper.subscribe/1","doc":"Subscribe to the oauth_credential topic. It expects the a unique ID for the topic, usually the LiveView's socket.id .","ref":"LightningWeb.OauthCredentialHelper.html#subscribe/1"},{"type":"module","title":"LightningWeb.OidcController","doc":"","ref":"LightningWeb.OidcController.html"},{"type":"function","title":"LightningWeb.OidcController.new/2","doc":"Once the user has completed the authorization flow from above, they are returned here, and the authorization code is used to log them in.","ref":"LightningWeb.OidcController.html#new/2"},{"type":"function","title":"LightningWeb.OidcController.show/2","doc":"Given a known provider, redirect them to the authorize url on the provider","ref":"LightningWeb.OidcController.html#show/2"},{"type":"module","title":"LightningWeb.PageView","doc":"","ref":"LightningWeb.PageView.html"},{"type":"function","title":"LightningWeb.PageView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"LightningWeb.PageView.html#__resource__/0"},{"type":"function","title":"LightningWeb.PageView.render/2","doc":"Renders the given template locally.","ref":"LightningWeb.PageView.html#render/2"},{"type":"function","title":"LightningWeb.PageView.template_not_found/2","doc":"Callback invoked when no template is found. By default it raises but can be customized to render a particular template.","ref":"LightningWeb.PageView.html#template_not_found/2"},{"type":"module","title":"LightningWeb.Pagination","doc":"Pagination Components This has been extracted and adapted from scrivener_html . See: https://github.com/mgwidmann/scrivener_html","ref":"LightningWeb.Pagination.html"},{"type":"function","title":"LightningWeb.Pagination.page_link/1","doc":"","ref":"LightningWeb.Pagination.html#page_link/1"},{"type":"function","title":"LightningWeb.Pagination.pagination_bar/1","doc":"","ref":"LightningWeb.Pagination.html#pagination_bar/1"},{"type":"function","title":"LightningWeb.Pagination.raw_pagination_links/2","doc":"Returns the raw data in order to generate the proper HTML for pagination links. Data is returned in a {text, page_number} format where text is intended to be the text of the link and page_number is the page it should go to. Defaults are already supplied and they are as follows: [ distance : 5 , next : :next , previous : :previous , first : true , last : true , ellipsis : :ellipsis ] distance must be a positive non-zero integer or an exception is raised. next and previous should be strings but can be anything you want as long as it is truthy, falsey values will remove them from the output. first and last are only booleans, and they just include/remove their respective link from output. An example of the data returned: iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 10 , page_number : 5 } ) [ { "<<" , 4 } , { 1 , 1 } , { 2 , 2 } , { 3 , 3 } , { 4 , 4 } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { ">>" , 6 } ] iex> Scrivener.HTML . raw_pagination_links ( %{ total_pages : 20 , page_number : 10 } , first : [ "←" ] , last : [ "→" ] ) [ { "<<" , 9 } , { [ "←" ] , 1 } , { :ellipsis , { :safe , "&hellip;" } } , { 5 , 5 } , { 6 , 6 } , { 7 , 7 } , { 8 , 8 } , { 9 , 9 } , { 10 , 10 } , { 11 , 11 } , { 12 , 12 } , { 13 , 13 } , { 14 , 14 } , { 15 , 15 } , { :ellipsis , { :safe , "&hellip;" } } , { [ "→" ] , 20 } , { ">>" , 11 } ] Simply loop and pattern match over each item and transform it to your custom HTML.","ref":"LightningWeb.Pagination.html#raw_pagination_links/2"},{"type":"module","title":"LightningWeb.Plugs.FirstSetup","doc":"Plug to redirect HTTP requests to /first_setup if there are no superusers in the system yet.","ref":"LightningWeb.Plugs.FirstSetup.html"},{"type":"module","title":"LightningWeb.ProfileLive.Edit","doc":"LiveView for user profile page.","ref":"LightningWeb.ProfileLive.Edit.html"},{"type":"function","title":"LightningWeb.ProfileLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProfileLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.FormComponent","doc":"Form component update profile email and password","ref":"LightningWeb.ProfileLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProfileLive.MfaComponent","doc":"Component to enable MFA on a User's account","ref":"LightningWeb.ProfileLive.MfaComponent.html"},{"type":"function","title":"LightningWeb.ProfileLive.MfaComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProfileLive.MfaComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.FormComponent","doc":"Form Component for working with a single Job A Job's adaptor field is a combination of the module name and the version. It's formatted as an NPM style string. The form allows the user to select a module by name and then it's version, while the version dropdown itself references adaptor directly. Meaning the adaptor_name dropdown and assigns value is not persisted.","ref":"LightningWeb.ProjectLive.FormComponent.html"},{"type":"function","title":"LightningWeb.ProjectLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.ProjectLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Index","doc":"LiveView for listing and managing Projects","ref":"LightningWeb.ProjectLive.Index.html"},{"type":"function","title":"LightningWeb.ProjectLive.Index.delete_action/1","doc":"","ref":"LightningWeb.ProjectLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.ProjectLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.ProjectLive.Settings","doc":"Index Liveview for Runs","ref":"LightningWeb.ProjectLive.Settings.html"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.digest/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#digest/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.failure_alert/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#failure_alert/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.ProjectLive.Settings.html#render/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.role/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#role/1"},{"type":"function","title":"LightningWeb.ProjectLive.Settings.user/1","doc":"","ref":"LightningWeb.ProjectLive.Settings.html#user/1"},{"type":"module","title":"LightningWeb.RouteHelpers","doc":"Convenience functions for generating paths.","ref":"LightningWeb.RouteHelpers.html"},{"type":"function","title":"LightningWeb.RouteHelpers.oidc_callback_url/0","doc":"","ref":"LightningWeb.RouteHelpers.html#oidc_callback_url/0"},{"type":"function","title":"LightningWeb.RouteHelpers.show_run_url/2","doc":"","ref":"LightningWeb.RouteHelpers.html#show_run_url/2"},{"type":"module","title":"LightningWeb.Router","doc":"","ref":"LightningWeb.Router.html"},{"type":"function","title":"LightningWeb.Router.api/2","doc":"","ref":"LightningWeb.Router.html#api/2"},{"type":"function","title":"LightningWeb.Router.browser/2","doc":"","ref":"LightningWeb.Router.html#browser/2"},{"type":"function","title":"LightningWeb.Router.call/2","doc":"Callback invoked by Plug on every request.","ref":"LightningWeb.Router.html#call/2"},{"type":"function","title":"LightningWeb.Router.init/1","doc":"Callback required by Plug that initializes the router for serving web requests.","ref":"LightningWeb.Router.html#init/1"},{"type":"function","title":"LightningWeb.Router.storybook_assets/2","doc":"","ref":"LightningWeb.Router.html#storybook_assets/2"},{"type":"function","title":"LightningWeb.Router.storybook_browser/2","doc":"","ref":"LightningWeb.Router.html#storybook_browser/2"},{"type":"module","title":"LightningWeb.RunLive.Index","doc":"Index Liveview for Runs","ref":"LightningWeb.RunLive.Index.html"},{"type":"function","title":"LightningWeb.RunLive.Index.checked/2","doc":"","ref":"LightningWeb.RunLive.Index.html#checked/2"},{"type":"function","title":"LightningWeb.RunLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.RunLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.RunLive.RerunJobComponent","doc":"Rerun job component","ref":"LightningWeb.RunLive.RerunJobComponent.html"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.handle_event/3","doc":"Callback implementation for Phoenix.LiveComponent.handle_event/3 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#handle_event/3"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#render/1"},{"type":"function","title":"LightningWeb.RunLive.RerunJobComponent.update/2","doc":"Callback implementation for Phoenix.LiveComponent.update/2 .","ref":"LightningWeb.RunLive.RerunJobComponent.html#update/2"},{"type":"module","title":"LightningWeb.RunLive.RunViewerLive","doc":"","ref":"LightningWeb.RunLive.RunViewerLive.html"},{"type":"function","title":"LightningWeb.RunLive.RunViewerLive.handle_info/2","doc":"Reload the run when any update messages arrive.","ref":"LightningWeb.RunLive.RunViewerLive.html#handle_info/2"},{"type":"module","title":"LightningWeb.RunLive.Show","doc":"Show page for individual runs.","ref":"LightningWeb.RunLive.Show.html"},{"type":"function","title":"LightningWeb.RunLive.Show.apply_action/3","doc":"","ref":"LightningWeb.RunLive.Show.html#apply_action/3"},{"type":"module","title":"LightningWeb.RunLive.WorkOrderComponent","doc":"Workorder component","ref":"LightningWeb.RunLive.WorkOrderComponent.html"},{"type":"function","title":"LightningWeb.RunLive.WorkOrderComponent.render/1","doc":"Attributes show_details ( :boolean ) - Defaults to false . entry_selected ( :boolean ) - Defaults to false .","ref":"LightningWeb.RunLive.WorkOrderComponent.html#render/1"},{"type":"module","title":"LightningWeb.Telemetry","doc":"Assorted metrics to collect during runtime. See https://hexdocs.pm/phoenix/telemetry.html","ref":"LightningWeb.Telemetry.html"},{"type":"function","title":"LightningWeb.Telemetry.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"LightningWeb.Telemetry.html#child_spec/1"},{"type":"function","title":"LightningWeb.Telemetry.metrics/0","doc":"","ref":"LightningWeb.Telemetry.html#metrics/0"},{"type":"function","title":"LightningWeb.Telemetry.start_link/1","doc":"","ref":"LightningWeb.Telemetry.html#start_link/1"},{"type":"module","title":"LightningWeb.TokensLive.Index","doc":"LiveView for listing and managing tokens","ref":"LightningWeb.TokensLive.Index.html"},{"type":"function","title":"LightningWeb.TokensLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.TokensLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserAuth","doc":"The UserAuth controller.","ref":"LightningWeb.UserAuth.html"},{"type":"function","title":"LightningWeb.UserAuth.authenticate_bearer/2","doc":"","ref":"LightningWeb.UserAuth.html#authenticate_bearer/2"},{"type":"function","title":"LightningWeb.UserAuth.fetch_current_user/2","doc":"Authenticates the user by looking into the session and remember me token.","ref":"LightningWeb.UserAuth.html#fetch_current_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_in_user/2","doc":"Logs the user in by creating a new session token.","ref":"LightningWeb.UserAuth.html#log_in_user/2"},{"type":"function","title":"LightningWeb.UserAuth.log_out_user/1","doc":"Logs the user out. It clears all session data for safety. See renew_session.","ref":"LightningWeb.UserAuth.html#log_out_user/1"},{"type":"function","title":"LightningWeb.UserAuth.mark_totp_pending/1","doc":"","ref":"LightningWeb.UserAuth.html#mark_totp_pending/1"},{"type":"function","title":"LightningWeb.UserAuth.new_session/2","doc":"Assigns the token to a new session. It renews the session ID and clears the whole session to avoid fixation attacks. See the renew_session function to customize this behaviour. It also sets a :live_socket_id key in the session, so LiveView sessions are identified and automatically disconnected on log out. The line can be safely removed if you are not using LiveView.","ref":"LightningWeb.UserAuth.html#new_session/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_if_user_is_authenticated/2","doc":"Used for routes that require the user to not be authenticated.","ref":"LightningWeb.UserAuth.html#redirect_if_user_is_authenticated/2"},{"type":"function","title":"LightningWeb.UserAuth.redirect_with_return_to/2","doc":"Returns to or redirects to the dashboard and potentially set remember_me token.","ref":"LightningWeb.UserAuth.html#redirect_with_return_to/2"},{"type":"function","title":"LightningWeb.UserAuth.require_authenticated_user/2","doc":"Used for routes that require the user to be authenticated. If you want to enforce the user email is confirmed before they use the application at all, here would be a good place.","ref":"LightningWeb.UserAuth.html#require_authenticated_user/2"},{"type":"function","title":"LightningWeb.UserAuth.totp_pending?/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_pending?/1"},{"type":"function","title":"LightningWeb.UserAuth.totp_validated/1","doc":"","ref":"LightningWeb.UserAuth.html#totp_validated/1"},{"type":"module","title":"LightningWeb.UserConfirmationController","doc":"","ref":"LightningWeb.UserConfirmationController.html"},{"type":"function","title":"LightningWeb.UserConfirmationController.confirm_email/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#confirm_email/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.create/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#create/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.edit/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#edit/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.new/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#new/2"},{"type":"function","title":"LightningWeb.UserConfirmationController.update/2","doc":"","ref":"LightningWeb.UserConfirmationController.html#update/2"},{"type":"module","title":"LightningWeb.UserLive.Edit","doc":"LiveView for editing a single job, which inturn uses LightningWeb.JobLive.BigFormComponent for common functionality.","ref":"LightningWeb.UserLive.Edit.html"},{"type":"function","title":"LightningWeb.UserLive.Edit.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Edit.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.FormComponent","doc":"Form component for creating and editing users","ref":"LightningWeb.UserLive.FormComponent.html"},{"type":"function","title":"LightningWeb.UserLive.FormComponent.render/1","doc":"Callback implementation for Phoenix.LiveComponent.render/1 .","ref":"LightningWeb.UserLive.FormComponent.html#render/1"},{"type":"module","title":"LightningWeb.UserLive.Index","doc":"Index page for listing users","ref":"LightningWeb.UserLive.Index.html"},{"type":"function","title":"LightningWeb.UserLive.Index.delete_action/1","doc":"","ref":"LightningWeb.UserLive.Index.html#delete_action/1"},{"type":"function","title":"LightningWeb.UserLive.Index.render/1","doc":"Callback implementation for Phoenix.LiveView.render/1 .","ref":"LightningWeb.UserLive.Index.html#render/1"},{"type":"module","title":"LightningWeb.UserRegistrationController","doc":"","ref":"LightningWeb.UserRegistrationController.html"},{"type":"function","title":"LightningWeb.UserRegistrationController.create/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#create/2"},{"type":"function","title":"LightningWeb.UserRegistrationController.new/2","doc":"","ref":"LightningWeb.UserRegistrationController.html#new/2"},{"type":"module","title":"LightningWeb.UserResetPasswordController","doc":"","ref":"LightningWeb.UserResetPasswordController.html"},{"type":"function","title":"LightningWeb.UserResetPasswordController.create/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#create/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.edit/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#edit/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.new/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#new/2"},{"type":"function","title":"LightningWeb.UserResetPasswordController.update/2","doc":"","ref":"LightningWeb.UserResetPasswordController.html#update/2"},{"type":"module","title":"LightningWeb.UserSessionController","doc":"","ref":"LightningWeb.UserSessionController.html"},{"type":"function","title":"LightningWeb.UserSessionController.auth_handler_url/0","doc":"","ref":"LightningWeb.UserSessionController.html#auth_handler_url/0"},{"type":"function","title":"LightningWeb.UserSessionController.create/2","doc":"","ref":"LightningWeb.UserSessionController.html#create/2"},{"type":"function","title":"LightningWeb.UserSessionController.delete/2","doc":"","ref":"LightningWeb.UserSessionController.html#delete/2"},{"type":"function","title":"LightningWeb.UserSessionController.exchange_token/2","doc":"","ref":"LightningWeb.UserSessionController.html#exchange_token/2"},{"type":"function","title":"LightningWeb.UserSessionController.new/2","doc":"","ref":"LightningWeb.UserSessionController.html#new/2"},{"type":"module","title":"LightningWeb.UserTOTPController","doc":"","ref":"LightningWeb.UserTOTPController.html"},{"type":"function","title":"LightningWeb.UserTOTPController.create/2","doc":"","ref":"LightningWeb.UserTOTPController.html#create/2"},{"type":"function","title":"LightningWeb.UserTOTPController.new/2","doc":"","ref":"LightningWeb.UserTOTPController.html#new/2"},{"type":"module","title":"LightningWeb.WebhooksController","doc":"","ref":"LightningWeb.WebhooksController.html"},{"type":"function","title":"LightningWeb.WebhooksController.create/2","doc":"","ref":"LightningWeb.WebhooksController.html#create/2"},{"type":"module","title":"LightningWeb.WorkflowLive.EditorPane","doc":"","ref":"LightningWeb.WorkflowLive.EditorPane.html"},{"type":"function","title":"LightningWeb.WorkflowLive.EditorPane.render/1","doc":"Attributes id ( :string ) (required) disabled ( :boolean ) - Defaults to false . class ( :string ) - Defaults to "" . on_change ( :any ) (required) adaptor ( :string ) (required) source ( :string ) (required) job_id ( :string ) (required)","ref":"LightningWeb.WorkflowLive.EditorPane.html#render/1"},{"type":"module","title":"LightningWeb.WorkflowLive.JobView","doc":"","ref":"LightningWeb.WorkflowLive.JobView.html"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.container/1","doc":"Attributes id ( :string ) (required) Slots top column - Accepts attributes: class ( :string ) - Extra CSS classes for the column. bottom","ref":"LightningWeb.WorkflowLive.JobView.html#container/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.input_pane/1","doc":"Attributes job ( :map ) (required) user ( :map ) (required) project ( :map ) (required) on_run ( :any ) (required) - Callback to run a job manually. can_run_job ( :boolean ) - Defaults to true .","ref":"LightningWeb.WorkflowLive.JobView.html#input_pane/1"},{"type":"function","title":"LightningWeb.WorkflowLive.JobView.job_edit_view/1","doc":"Attributes job ( :map ) (required) form ( :map ) (required) - A form built from a job. current_user ( :map ) (required) project ( :map ) (required) close_url ( :any ) (required) socket ( :any ) (required) on_run ( :any ) (required) - Callback to run a job manually. follow_run_id ( :any ) - Defaults to nil . Slots footer","ref":"LightningWeb.WorkflowLive.JobView.html#job_edit_view/1"},{"type":"module","title":"LightningWeb.WorkflowNewLive.WorkflowParams","doc":"Various function for reconciling changes to a workflow params map. The front end editor uses JSON patches to represent changes to the workflow.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_form_params/2","doc":"Produce a new set of params by applying the given form params to the current params.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_form_params/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.apply_patches/2","doc":"Produce a new set of params by applying the given patches to the current parms","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#apply_patches/2"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_map/1","doc":"Convert a changeset to a serializable map of workflow params, suitable for sending to the front end editor. It uses Lightning.Helpers.json_safe/1 to ensure that the map is safe to serialize to JSON. This is necessary because the underlying model may contain atom values.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_map/1"},{"type":"function","title":"LightningWeb.WorkflowNewLive.WorkflowParams.to_patches/2","doc":"Produce a set of patches that represent the difference between the initial params and the target params. This usually is used to produce a set of patches that represent the changes introduced by a changeset.","ref":"LightningWeb.WorkflowNewLive.WorkflowParams.html#to_patches/2"},{"type":"task","title":"Mix.Tasks.Lightning.GenEncryptionKey","doc":"Helper to generate a unique encryption key for Vault","ref":"Mix.Tasks.Lightning.GenEncryptionKey.html"},{"type":"task","title":"Mix.Tasks.Lightning.InstallRuntime","doc":"Installs the following NodeJS packages: core language-common","ref":"Mix.Tasks.Lightning.InstallRuntime.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.packages/0","doc":"","ref":"Mix.Tasks.Lightning.InstallRuntime.html#packages/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallRuntime.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallRuntime.html#run/1"},{"type":"task","title":"Mix.Tasks.Lightning.InstallSchemas","doc":"Install the credential json schemas Use --exclude language-package1, language-package2 to exclude specific packages","ref":"Mix.Tasks.Lightning.InstallSchemas.html"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete/3","doc":"Issues a DELETE request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.delete!/3","doc":"Issues a DELETE request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#delete!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.fetch_schemas/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#fetch_schemas/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get/3","doc":"Issues a GET request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.get!/3","doc":"Issues a GET request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#get!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head/3","doc":"Issues a HEAD request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.head!/3","doc":"Issues a HEAD request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#head!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options/3","doc":"Issues an OPTIONS request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.options!/3","doc":"Issues a OPTIONS request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#options!/3"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.parse_excluded/1","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#parse_excluded/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch/4","doc":"Issues a PATCH request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.patch!/4","doc":"Issues a PATCH request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#patch!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.persist_schema/2","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#persist_schema/2"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post/4","doc":"Issues a POST request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.post!/4","doc":"Issues a POST request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#post!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_headers/1","doc":"Callback implementation for HTTPoison.Base.process_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_body/1","doc":"Callback implementation for HTTPoison.Base.process_request_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_headers/1","doc":"Callback implementation for HTTPoison.Base.process_request_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_options/1","doc":"Callback implementation for HTTPoison.Base.process_request_options/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_options/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_params/1","doc":"Callback implementation for HTTPoison.Base.process_request_params/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_params/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_request_url/1","doc":"Callback implementation for HTTPoison.Base.process_request_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_request_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response/1","doc":"Callback implementation for HTTPoison.Base.process_response/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_body/1","doc":"Callback implementation for HTTPoison.Base.process_response_body/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_body/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_chunk/1","doc":"Callback implementation for HTTPoison.Base.process_response_chunk/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_chunk/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_headers/1","doc":"Callback implementation for HTTPoison.Base.process_response_headers/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_headers/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_response_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_response_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_response_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_status_code/1","doc":"Callback implementation for HTTPoison.Base.process_status_code/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_status_code/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.process_url/1","doc":"Callback implementation for HTTPoison.Base.process_url/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#process_url/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put/4","doc":"Issues a PUT request to the given url. Returns {:ok, response} if the request is successful, {:error, reason} otherwise. See request/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.put!/4","doc":"Issues a PUT request to the given url, raising an exception in case of failure. If the request does not fail, the response is returned. See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#put!/4"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/1","doc":"Issues an HTTP request using a Request struct. This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request = % HTTPoison.Request { method : :post , url : "https://my.website.com" , body : "{ \\" foo \\" : 3}" , headers : [ { "Accept" , "application/json" } ] } request ( request )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request/5","doc":"Issues an HTTP request with the given method to the given url. This function is usually used indirectly by get/3 , post/4 , put/4 , etc Args: method - HTTP method as an atom ( :get , :head , :post , :put , :delete , etc.) url - target url as a binary string or char list body - request body. See more below headers - HTTP headers as an orddict (e.g., [{"Accept", "application/json"}] ) options - Keyword list of options Body: see type HTTPoison.Request Options: see type HTTPoison.Request This function returns {:ok, response} , {:ok, async_response} , or {:ok, maybe_redirect} if the request is successful, {:error, reason} otherwise. Redirect handling If the option :follow_redirect is given, HTTP redirects are automatically follow if the method is set to :get or :head and the response's status_code is 301 , 302 or 307 . If the method is set to :post , then the only status_code that get's automatically followed is 303 . If any other method or status_code is returned, then this function returns a returns a {:ok, %HTTPoison.MaybeRedirect{}} containing the redirect_url for you to re-request with the method set to :get . Examples request ( :post , "https://my.website.com" , "{ \\" foo \\" : 3}" , [ { "Accept" , "application/json" } ] )","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.request!/5","doc":"Issues an HTTP request with the given method to the given url, raising an exception in case of failure. request!/5 works exactly like request/5 but it returns just the response in case of a successful request, raising an exception in case the request fails.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#request!/5"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.run/1","doc":"Callback implementation for Mix.Task.run/1 .","ref":"Mix.Tasks.Lightning.InstallSchemas.html#run/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.start/0","doc":"Starts HTTPoison and its dependencies.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#start/0"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.stream_next/1","doc":"Requests the next message to be streamed for a given HTTPoison.AsyncResponse . See request!/5 for more detailed information.","ref":"Mix.Tasks.Lightning.InstallSchemas.html#stream_next/1"},{"type":"function","title":"Mix.Tasks.Lightning.InstallSchemas.write_schema/3","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#write_schema/3"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.body/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:body/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.headers/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:headers/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.method/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:method/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.options/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:options/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.params/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:params/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.request/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:request/0"},{"type":"type","title":"Mix.Tasks.Lightning.InstallSchemas.url/0","doc":"","ref":"Mix.Tasks.Lightning.InstallSchemas.html#t:url/0"},{"type":"module","title":"ObanPruner","doc":"The Oban Pruner removes completed Oban jobs. It leaves everything else for manual inspection.","ref":"ObanPruner.html"},{"type":"function","title":"ObanPruner.perform/1","doc":"Deletes completed Oban jobs, leaving discarded for manual inspection.","ref":"ObanPruner.html#perform/1"},{"type":"module","title":"Storybook.Root","doc":"","ref":"Storybook.Root.html"},{"type":"module","title":"Lightning.Accounts","doc":"The Accounts context.","ref":"Lightning.Accounts.html"},{"type":"function","title":"Lightning.Accounts.apply_user_email/3","doc":"Emulates that the email will change without actually changing it in the database. Examples iex> apply_user_email ( user , "valid password" , %{ email : ... } ) { :ok , % User { } } role : :superuser iex> apply_user_email ( user , "invalid password" , %{ email : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#apply_user_email/3"},{"type":"function","title":"Lightning.Accounts.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Accounts.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Accounts.change_scheduled_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the user scheduled_deletion. Examples iex> change_scheduled_deletion ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_scheduled_deletion/2"},{"type":"function","title":"Lightning.Accounts.change_superuser_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking superuser changes. Examples iex> change_superuser_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_superuser_registration/1"},{"type":"function","title":"Lightning.Accounts.change_user_details/2","doc":"","ref":"Lightning.Accounts.html#change_user_details/2"},{"type":"function","title":"Lightning.Accounts.change_user_email/2","doc":"Returns an %Ecto.Changeset{} for changing the user email. Examples iex> change_user_email ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_email/2"},{"type":"function","title":"Lightning.Accounts.change_user_password/2","doc":"Returns an %Ecto.Changeset{} for changing the user password. Examples iex> change_user_password ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_password/2"},{"type":"function","title":"Lightning.Accounts.change_user_registration/1","doc":"Returns an %Ecto.Changeset{} for tracking user changes. Examples iex> change_user_registration ( user ) % Ecto.Changeset { data : % User { } }","ref":"Lightning.Accounts.html#change_user_registration/1"},{"type":"function","title":"Lightning.Accounts.confirm_user/1","doc":"Confirms a user by the given token. If the token matches, the user account is marked as confirmed and the token is deleted.","ref":"Lightning.Accounts.html#confirm_user/1"},{"type":"function","title":"Lightning.Accounts.delete_auth_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_auth_token/1"},{"type":"function","title":"Lightning.Accounts.delete_session_token/1","doc":"Deletes the signed token with the given context.","ref":"Lightning.Accounts.html#delete_session_token/1"},{"type":"function","title":"Lightning.Accounts.delete_token/1","doc":"Deletes a token. Examples iex> delete_token ( token ) { :ok , % UserToken { } } iex> delete_token ( token ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_token/1"},{"type":"function","title":"Lightning.Accounts.delete_user/1","doc":"Deletes a user. Examples iex> delete_user ( user ) { :ok , % User { } } iex> delete_user ( user ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#delete_user/1"},{"type":"function","title":"Lightning.Accounts.delete_user_totp/1","doc":"Deletes the given user's TOTP","ref":"Lightning.Accounts.html#delete_user_totp/1"},{"type":"function","title":"Lightning.Accounts.deliver_update_email_instructions/3","doc":"Delivers the update email instructions to the given user. Examples iex> deliver_update_email_instructions ( user , current_email , & Routes . user_update_email_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_update_email_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/2","doc":"Delivers the confirmation email instructions to the given user. Examples iex> deliver_user_confirmation_instructions ( user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } } iex> deliver_user_confirmation_instructions ( confirmed_user , & Routes . user_confirmation_url ( conn , :edit , &1 ) ) { :error , :already_confirmed }","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.deliver_user_confirmation_instructions/3","doc":"","ref":"Lightning.Accounts.html#deliver_user_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.deliver_user_reset_password_instructions/2","doc":"Delivers the reset password email to the given user. Examples iex> deliver_user_reset_password_instructions ( user , & Routes . user_reset_password_url ( conn , :edit , &1 ) ) { :ok , %{ to : ... , body : ... } }","ref":"Lightning.Accounts.html#deliver_user_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.exchange_auth_token/1","doc":"Exchanges an auth token for a session token. The auth token is removed from the database if successful.","ref":"Lightning.Accounts.html#exchange_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_api_token/1","doc":"Generates an API token for a user.","ref":"Lightning.Accounts.html#generate_api_token/1"},{"type":"function","title":"Lightning.Accounts.generate_auth_token/1","doc":"Generates an auth token.","ref":"Lightning.Accounts.html#generate_auth_token/1"},{"type":"function","title":"Lightning.Accounts.generate_user_session_token/1","doc":"Generates a session token.","ref":"Lightning.Accounts.html#generate_user_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_token!/1","doc":"Gets a single token. Raises Ecto.NoResultsError if the UserToken does not exist. Examples iex> get_token! ( 123 ) % UserToken { } iex> get_token! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_token!/1"},{"type":"function","title":"Lightning.Accounts.get_user!/1","doc":"Gets a single user. Raises Ecto.NoResultsError if the User does not exist. Examples iex> get_user! ( 123 ) % User { } iex> get_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Accounts.html#get_user!/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_api_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_api_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_auth_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_auth_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email/1","doc":"Gets a user by email. Examples iex> get_user_by_email ( "foo@example.com" ) % User { } iex> get_user_by_email ( "unknown@example.com" ) nil","ref":"Lightning.Accounts.html#get_user_by_email/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_email_and_password/2","doc":"Gets a user by email and password. Examples iex> get_user_by_email_and_password ( "foo@example.com" , "correct_password" ) % User { } iex> get_user_by_email_and_password ( "foo@example.com" , "invalid_password" ) nil","ref":"Lightning.Accounts.html#get_user_by_email_and_password/2"},{"type":"function","title":"Lightning.Accounts.get_user_by_reset_password_token/1","doc":"Gets the user by reset password token. Examples iex> get_user_by_reset_password_token ( "validtoken" ) % User { } iex> get_user_by_reset_password_token ( "invalidtoken" ) nil","ref":"Lightning.Accounts.html#get_user_by_reset_password_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_by_session_token/1","doc":"Gets the user with the given signed token.","ref":"Lightning.Accounts.html#get_user_by_session_token/1"},{"type":"function","title":"Lightning.Accounts.get_user_totp/1","doc":"Gets a single UserTOTP if any exists.","ref":"Lightning.Accounts.html#get_user_totp/1"},{"type":"function","title":"Lightning.Accounts.get_users_to_alert_for_project/1","doc":"Gets all users to alert of workflow failure for a project","ref":"Lightning.Accounts.html#get_users_to_alert_for_project/1"},{"type":"function","title":"Lightning.Accounts.has_activity_in_projects?/1","doc":"","ref":"Lightning.Accounts.html#has_activity_in_projects?/1"},{"type":"function","title":"Lightning.Accounts.has_one_superuser?/0","doc":"Used to determine if there is at least one Superuser in the system. This triggers the setup page on fresh installs.","ref":"Lightning.Accounts.html#has_one_superuser?/0"},{"type":"function","title":"Lightning.Accounts.list_api_tokens/1","doc":"Lists all user tokens","ref":"Lightning.Accounts.html#list_api_tokens/1"},{"type":"function","title":"Lightning.Accounts.list_users/0","doc":"Returns the list of users. Examples iex> list_users ( ) [ % User { } , ... ]","ref":"Lightning.Accounts.html#list_users/0"},{"type":"function","title":"Lightning.Accounts.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find users that are ready for permanent deletion and purge them.","ref":"Lightning.Accounts.html#perform/1"},{"type":"function","title":"Lightning.Accounts.purge_user/1","doc":"","ref":"Lightning.Accounts.html#purge_user/1"},{"type":"function","title":"Lightning.Accounts.register_superuser/1","doc":"Registers a superuser. Examples iex> register_superuser ( %{ field : value } ) { :ok , % User { } } iex> register_superuser ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_superuser/1"},{"type":"function","title":"Lightning.Accounts.register_user/1","doc":"Registers a user. Examples iex> register_user ( %{ field : value } ) { :ok , % User { } } iex> register_user ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#register_user/1"},{"type":"function","title":"Lightning.Accounts.reset_user_password/2","doc":"Resets the user password. Examples iex> reset_user_password ( user , %{ password : "new long password" , password_confirmation : "new long password" } ) { :ok , % User { } } iex> reset_user_password ( user , %{ password : "valid" , password_confirmation : "not the same" } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#reset_user_password/2"},{"type":"function","title":"Lightning.Accounts.schedule_user_deletion/2","doc":"Given a user and a confirmation email, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic user purge cronjob will never run. (Note that subsequent logins will be blocked for users pending deletion.)","ref":"Lightning.Accounts.html#schedule_user_deletion/2"},{"type":"function","title":"Lightning.Accounts.update_user_details/2","doc":"","ref":"Lightning.Accounts.html#update_user_details/2"},{"type":"function","title":"Lightning.Accounts.update_user_email/2","doc":"Updates the user email using the given token. If the token matches, the user email is updated and the token is deleted. The confirmed_at date is also updated to the current time.","ref":"Lightning.Accounts.html#update_user_email/2"},{"type":"function","title":"Lightning.Accounts.update_user_password/3","doc":"Updates the user password. Examples iex> update_user_password ( user , "valid password" , %{ password : ... } ) { :ok , % User { } } iex> update_user_password ( user , "invalid password" , %{ password : ... } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Accounts.html#update_user_password/3"},{"type":"function","title":"Lightning.Accounts.upsert_user_totp/2","doc":"Updates or Inserts the user's TOTP","ref":"Lightning.Accounts.html#upsert_user_totp/2"},{"type":"function","title":"Lightning.Accounts.valid_user_totp?/2","doc":"Validates if the given TOTP code is valid.","ref":"Lightning.Accounts.html#valid_user_totp?/2"},{"type":"function","title":"Lightning.Accounts.validate_change_user_email/2","doc":"","ref":"Lightning.Accounts.html#validate_change_user_email/2"},{"type":"module","title":"Lightning.Accounts.User","doc":"The User model.","ref":"Lightning.Accounts.User.html"},{"type":"function","title":"Lightning.Accounts.User.confirm_changeset/1","doc":"Confirms the account by setting confirmed_at .","ref":"Lightning.Accounts.User.html#confirm_changeset/1"},{"type":"function","title":"Lightning.Accounts.User.details_changeset/2","doc":"A user changeset for user details: email first_name last_name role","ref":"Lightning.Accounts.User.html#details_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.email_changeset/2","doc":"A user changeset for changing the email. It requires the email to change otherwise an error is added.","ref":"Lightning.Accounts.User.html#email_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.password_changeset/3","doc":"A user changeset for changing the password. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#password_changeset/3"},{"type":"function","title":"Lightning.Accounts.User.scheduled_deletion_changeset/2","doc":"A user changeset for changing the scheduled_deletion property.","ref":"Lightning.Accounts.User.html#scheduled_deletion_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.superuser_registration_changeset/2","doc":"A superuser changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#superuser_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.user_registration_changeset/2","doc":"A user changeset for registration. It is important to validate the length of both email and password. Otherwise databases may truncate the email without warnings, which could lead to unpredictable or insecure behaviour. Long passwords may also be very expensive to hash for certain algorithms. Options :hash_password - Hashes the password so it can be stored securely in the database and ensures the password field is cleared to prevent leaks in the logs. If password hashing is not needed and clearing the password field is not desired (like when using this changeset for validations on a LiveView form), this option can be set to false . Defaults to true .","ref":"Lightning.Accounts.User.html#user_registration_changeset/2"},{"type":"function","title":"Lightning.Accounts.User.valid_password?/2","doc":"Verifies the password. If there is no user or the user doesn't have a password, we call Bcrypt.no_user_verify/0 to avoid timing attacks.","ref":"Lightning.Accounts.User.html#valid_password?/2"},{"type":"function","title":"Lightning.Accounts.User.validate_current_password/2","doc":"Validates the current password otherwise adds an error to the changeset.","ref":"Lightning.Accounts.User.html#validate_current_password/2"},{"type":"type","title":"Lightning.Accounts.User.t/0","doc":"","ref":"Lightning.Accounts.User.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.User.RolesEnum","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.create_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Accounts.User.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Accounts.User.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schema/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Accounts.User.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Accounts.User.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Accounts.User.RolesEnum.t/0","doc":"","ref":"Lightning.Accounts.User.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserNotifier","doc":"The UserNotifier module.","ref":"Lightning.Accounts.UserNotifier.html"},{"type":"function","title":"Lightning.Accounts.UserNotifier.build_digest_url/3","doc":"","ref":"Lightning.Accounts.UserNotifier.html#build_digest_url/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/2","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_confirmation_instructions/3","doc":"Deliver instructions to confirm account.","ref":"Lightning.Accounts.UserNotifier.html#deliver_confirmation_instructions/3"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_addition_notification/2","doc":"Deliver email to notify user of his addition of a project.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_addition_notification/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_project_digest/2","doc":"Deliver a project digest of daily/weekly or monthly activity to a user.","ref":"Lightning.Accounts.UserNotifier.html#deliver_project_digest/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_reset_password_instructions/2","doc":"Deliver instructions to reset a user password.","ref":"Lightning.Accounts.UserNotifier.html#deliver_reset_password_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_instructions/2","doc":"Deliver instructions to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_instructions/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.deliver_update_email_warning/2","doc":"Deliver warning to update a user email.","ref":"Lightning.Accounts.UserNotifier.html#deliver_update_email_warning/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.notify_project_deletion/2","doc":"","ref":"Lightning.Accounts.UserNotifier.html#notify_project_deletion/2"},{"type":"function","title":"Lightning.Accounts.UserNotifier.send_deletion_notification_email/1","doc":"Deliver an email to notify the user about their account being deleted","ref":"Lightning.Accounts.UserNotifier.html#send_deletion_notification_email/1"},{"type":"module","title":"Lightning.Accounts.UserTOTP","doc":"User Time based OTPs schema","ref":"Lightning.Accounts.UserTOTP.html"},{"type":"function","title":"Lightning.Accounts.UserTOTP.changeset/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserTOTP.valid_totp?/2","doc":"","ref":"Lightning.Accounts.UserTOTP.html#valid_totp?/2"},{"type":"type","title":"Lightning.Accounts.UserTOTP.t/0","doc":"","ref":"Lightning.Accounts.UserTOTP.html#t:t/0"},{"type":"module","title":"Lightning.Accounts.UserToken","doc":"The UserToken model. The reason why we store session tokens in the database, even though Phoenix already provides a session cookie, is because Phoenix' default session cookies are not persisted, they are simply signed and potentially encrypted. This means they are valid indefinitely, unless you change the signing/encryption salt. Therefore, storing them allows individual user sessions to be expired. The token system can also be extended to store additional data, such as the device used for logging in. You could then use this information to display all valid sessions and devices in the UI and allow users to explicitly expire any session they deem invalid.","ref":"Lightning.Accounts.UserToken.html"},{"type":"function","title":"Lightning.Accounts.UserToken.build_email_token/3","doc":"Builds a token and its hash to be delivered to the user's email. The non-hashed token is sent to the user email while the hashed part is stored in the database. The original token cannot be reconstructed, which means anyone with read-only access to the database cannot directly use the token in the application to gain access. Furthermore, if the user changes their email in the system, the tokens sent to the previous email are no longer valid. Users can easily adapt the existing code to provide other types of delivery methods, for example, by phone numbers.","ref":"Lightning.Accounts.UserToken.html#build_email_token/3"},{"type":"function","title":"Lightning.Accounts.UserToken.build_token/2","doc":"Generates a token that will be stored in a signed place, such as session or cookie. As they are signed, those tokens do not need to be hashed.","ref":"Lightning.Accounts.UserToken.html#build_token/2"},{"type":"function","title":"Lightning.Accounts.UserToken.changeset/2","doc":"","ref":"Lightning.Accounts.UserToken.html#changeset/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign/2","doc":"Combines generate_claims/1 and encode_and_sign/2","ref":"Lightning.Accounts.UserToken.html#generate_and_sign/2"},{"type":"function","title":"Lightning.Accounts.UserToken.generate_and_sign!/2","doc":"Same as generate_and_sign/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#generate_and_sign!/2"},{"type":"function","title":"Lightning.Accounts.UserToken.last_used_changeset/1","doc":"Update when the api token was last used by setting last_used_at .","ref":"Lightning.Accounts.UserToken.html#last_used_changeset/1"},{"type":"function","title":"Lightning.Accounts.UserToken.token_and_context_query/2","doc":"Returns the token struct for the given token value and context.","ref":"Lightning.Accounts.UserToken.html#token_and_context_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.user_and_contexts_query/2","doc":"Gets all tokens for the given user for the given contexts.","ref":"Lightning.Accounts.UserToken.html#user_and_contexts_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate/3","doc":"Combines verify/2 and validate/2","ref":"Lightning.Accounts.UserToken.html#verify_and_validate/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_and_validate!/3","doc":"Same as verify_and_validate/2 but raises if error","ref":"Lightning.Accounts.UserToken.html#verify_and_validate!/3"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_change_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. This is used to validate requests to change the user email. It is different from verify_email_token_query/2 precisely because verify_email_token_query/2 validates the email has not changed, which is the starting point by this function. The given token is valid if it matches its hashed counterpart in the database and if it has not expired (after @change_email_validity_in_days). The context must always start with "change:".","ref":"Lightning.Accounts.UserToken.html#verify_change_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_email_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The given token is valid if it matches its hashed counterpart in the database and the user email has not changed. This function also checks if the token is being used within a certain period, depending on the context. The default contexts supported by this function are either "confirm", for account confirmation emails, and "reset_password", for resetting the password. For verifying requests to change the email, see verify_change_email_token_query/2 .","ref":"Lightning.Accounts.UserToken.html#verify_email_token_query/2"},{"type":"function","title":"Lightning.Accounts.UserToken.verify_token_query/2","doc":"Checks if the token is valid and returns its underlying lookup query. The query returns the user found by the token, if any. The token is valid if it matches the value in the database and it has not expired (after @auth_validity_in_seconds or @session_validity_in_days).","ref":"Lightning.Accounts.UserToken.html#verify_token_query/2"},{"type":"module","title":"Lightning.Credentials","doc":"The Credentials context.","ref":"Lightning.Credentials.html"},{"type":"function","title":"Lightning.Credentials.change_credential/2","doc":"Returns an %Ecto.Changeset{} for tracking credential changes. Examples iex> change_credential ( credential ) % Ecto.Changeset { data : % Credential { } }","ref":"Lightning.Credentials.html#change_credential/2"},{"type":"function","title":"Lightning.Credentials.create_credential/1","doc":"Creates a credential. Examples iex> create_credential ( %{ field : value } ) { :ok , % Credential { } } iex> create_credential ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#create_credential/1"},{"type":"function","title":"Lightning.Credentials.delete_credential/1","doc":"Deletes a credential. Examples iex> delete_credential ( credential ) { :ok , % Credential { } } iex> delete_credential ( credential ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#delete_credential/1"},{"type":"function","title":"Lightning.Credentials.get_credential!/1","doc":"Gets a single credential. Raises Ecto.NoResultsError if the Credential does not exist. Examples iex> get_credential! ( 123 ) % Credential { } iex> get_credential! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Credentials.html#get_credential!/1"},{"type":"function","title":"Lightning.Credentials.invalid_projects_for_user/2","doc":"Given a credential and a user, returns a list of invalid projects—i.e., those that the credential is shared with but that the user does not have access to. This is used to generate a validation error when a credential cannot be transferred. Examples iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ ] iex> can_credential_be_shared_to_user ( credential_id , user_id ) [ "52ea8758-6ce5-43d7-912f-6a1e1f11dc55" ]","ref":"Lightning.Credentials.html#invalid_projects_for_user/2"},{"type":"function","title":"Lightning.Credentials.list_credentials/0","doc":"Returns the list of credentials. Examples iex> list_credentials ( ) [ % Credential { } , ... ]","ref":"Lightning.Credentials.html#list_credentials/0"},{"type":"function","title":"Lightning.Credentials.list_credentials/1","doc":"","ref":"Lightning.Credentials.html#list_credentials/1"},{"type":"function","title":"Lightning.Credentials.list_credentials_for_user/1","doc":"Returns the list of credentials for a given user. Examples iex> list_credentials_for_user ( 123 ) [ % Credential { user_id : 123 } , % Credential { user_id : 123 } , ... ]","ref":"Lightning.Credentials.html#list_credentials_for_user/1"},{"type":"function","title":"Lightning.Credentials.maybe_refresh_token/1","doc":"","ref":"Lightning.Credentials.html#maybe_refresh_token/1"},{"type":"function","title":"Lightning.Credentials.sensitive_values_for/1","doc":"","ref":"Lightning.Credentials.html#sensitive_values_for/1"},{"type":"function","title":"Lightning.Credentials.update_credential/2","doc":"Updates a credential. Examples iex> update_credential ( credential , %{ field : new_value } ) { :ok , % Credential { } } iex> update_credential ( credential , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Credentials.html#update_credential/2"},{"type":"module","title":"Lightning.Credentials.Audit","doc":"Model for storing changes to Credentials","ref":"Lightning.Credentials.Audit.html"},{"type":"function","title":"Lightning.Credentials.Audit.event/4","doc":"","ref":"Lightning.Credentials.Audit.html#event/4"},{"type":"function","title":"Lightning.Credentials.Audit.save/1","doc":"","ref":"Lightning.Credentials.Audit.html#save/1"},{"type":"module","title":"Lightning.Credentials.Credential","doc":"The Credential model.","ref":"Lightning.Credentials.Credential.html"},{"type":"type","title":"Lightning.Credentials.Credential.t/0","doc":"","ref":"Lightning.Credentials.Credential.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.Schema","doc":"Structure that can parse JsonSchemas (using ExJsonSchema ) and validate changesets for a given schema.","ref":"Lightning.Credentials.Schema.html"},{"type":"function","title":"Lightning.Credentials.Schema.new/2","doc":"","ref":"Lightning.Credentials.Schema.html#new/2"},{"type":"function","title":"Lightning.Credentials.Schema.properties/2","doc":"","ref":"Lightning.Credentials.Schema.html#properties/2"},{"type":"function","title":"Lightning.Credentials.Schema.required?/2","doc":"","ref":"Lightning.Credentials.Schema.html#required?/2"},{"type":"function","title":"Lightning.Credentials.Schema.validate/2","doc":"","ref":"Lightning.Credentials.Schema.html#validate/2"},{"type":"type","title":"Lightning.Credentials.Schema.t/0","doc":"","ref":"Lightning.Credentials.Schema.html#t:t/0"},{"type":"module","title":"Lightning.Credentials.SchemaDocument","doc":"Provides facilities to dynamically create and validate a changeset for a given Schema","ref":"Lightning.Credentials.SchemaDocument.html"},{"type":"function","title":"Lightning.Credentials.SchemaDocument.changeset/3","doc":"","ref":"Lightning.Credentials.SchemaDocument.html#changeset/3"},{"type":"module","title":"Lightning.Credentials.SensitiveValues","doc":"Functions to pull out sensitive values inside a credential. These values are used to scrub logs for leaked secrets.","ref":"Lightning.Credentials.SensitiveValues.html"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.flatten_map/1","doc":"Keys that are not considered sensitive","ref":"Lightning.Credentials.SensitiveValues.html#flatten_map/1"},{"type":"function","title":"Lightning.Credentials.SensitiveValues.secret_values/1","doc":"Given a map, find all values allowed (via @safe_keys ) and return them as a list.","ref":"Lightning.Credentials.SensitiveValues.html#secret_values/1"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:pairs/0"},{"type":"type","title":"Lightning.Credentials.SensitiveValues.raw_pairs/0","doc":"","ref":"Lightning.Credentials.SensitiveValues.html#t:raw_pairs/0"},{"type":"module","title":"Lightning.Invocation","doc":"The Invocation context.","ref":"Lightning.Invocation.html"},{"type":"function","title":"Lightning.Invocation.change_dataclip/2","doc":"Returns an %Ecto.Changeset{} for tracking dataclip changes. Examples iex> change_dataclip ( dataclip ) % Ecto.Changeset { data : % Dataclip { } }","ref":"Lightning.Invocation.html#change_dataclip/2"},{"type":"function","title":"Lightning.Invocation.change_run/2","doc":"Returns an %Ecto.Changeset{} for tracking run changes. Examples iex> change_run ( run ) % Ecto.Changeset { data : % Run { } }","ref":"Lightning.Invocation.html#change_run/2"},{"type":"function","title":"Lightning.Invocation.create_dataclip/1","doc":"Creates a dataclip. Examples iex> create_dataclip ( %{ field : value } ) { :ok , % Dataclip { } } iex> create_dataclip ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_dataclip/1"},{"type":"function","title":"Lightning.Invocation.create_log_line/2","doc":"","ref":"Lightning.Invocation.html#create_log_line/2"},{"type":"function","title":"Lightning.Invocation.create_run/1","doc":"Creates a run. Examples iex> create_run ( %{ field : value } ) { :ok , % Run { } } iex> create_run ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#create_run/1"},{"type":"function","title":"Lightning.Invocation.delete_dataclip/1","doc":"Deletes a dataclip. Examples iex> delete_dataclip ( dataclip ) { :ok , % Dataclip { } } iex> delete_dataclip ( dataclip ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_dataclip/1"},{"type":"function","title":"Lightning.Invocation.delete_run/1","doc":"Deletes a run. Examples iex> delete_run ( run ) { :ok , % Run { } } iex> delete_run ( run ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#delete_run/1"},{"type":"function","title":"Lightning.Invocation.filter_run_body_and_logs_where/2","doc":"","ref":"Lightning.Invocation.html#filter_run_body_and_logs_where/2"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_finished_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_finished_before_where/1"},{"type":"function","title":"Lightning.Invocation.filter_run_status_where/1","doc":"","ref":"Lightning.Invocation.html#filter_run_status_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workflow_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workflow_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_after_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_after_where/1"},{"type":"function","title":"Lightning.Invocation.filter_workorder_insert_before_where/1","doc":"","ref":"Lightning.Invocation.html#filter_workorder_insert_before_where/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip/1","doc":"Gets a single dataclip given one of: a Dataclip uuid a Run model Returns nil if the Dataclip does not exist. Examples iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) % Dataclip { } iex> get_dataclip ( "27b73932-16c7-4a72-86a3-85d805ccff98" ) nil iex> get_dataclip ( % Run { id : "a uuid" } ) % Dataclip { }","ref":"Lightning.Invocation.html#get_dataclip/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip!/1","doc":"Gets a single dataclip. Raises Ecto.NoResultsError if the Dataclip does not exist. Examples iex> get_dataclip! ( 123 ) % Dataclip { } iex> get_dataclip! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_dataclip!/1"},{"type":"function","title":"Lightning.Invocation.get_dataclip_query/1","doc":"Query for retrieving the dataclip that a runs starting dataclip.","ref":"Lightning.Invocation.html#get_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_result_dataclip_query/1","doc":"Query for retrieving the dataclip that was the result of a successful run.","ref":"Lightning.Invocation.html#get_result_dataclip_query/1"},{"type":"function","title":"Lightning.Invocation.get_run!/1","doc":"Gets a single run. Raises Ecto.NoResultsError if the Run does not exist. Examples iex> get_run! ( 123 ) % Run { } iex> get_run! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Invocation.html#get_run!/1"},{"type":"function","title":"Lightning.Invocation.get_run_with_job!/1","doc":"Fetches a run and preloads the job via the run's event.","ref":"Lightning.Invocation.html#get_run_with_job!/1"},{"type":"function","title":"Lightning.Invocation.get_workorders_by_ids/1","doc":"","ref":"Lightning.Invocation.html#get_workorders_by_ids/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips/0","doc":"Returns the list of dataclips. Examples iex> list_dataclips ( ) [ % Dataclip { } , ... ]","ref":"Lightning.Invocation.html#list_dataclips/0"},{"type":"function","title":"Lightning.Invocation.list_dataclips/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_for_job/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_for_job/1"},{"type":"function","title":"Lightning.Invocation.list_dataclips_query/1","doc":"","ref":"Lightning.Invocation.html#list_dataclips_query/1"},{"type":"function","title":"Lightning.Invocation.list_runs/0","doc":"Returns the list of runs. Examples iex> list_runs ( ) [ % Run { } , ... ]","ref":"Lightning.Invocation.html#list_runs/0"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project/2","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project/2"},{"type":"function","title":"Lightning.Invocation.list_runs_for_project_query/1","doc":"","ref":"Lightning.Invocation.html#list_runs_for_project_query/1"},{"type":"function","title":"Lightning.Invocation.list_work_orders_for_project_query/2","doc":"","ref":"Lightning.Invocation.html#list_work_orders_for_project_query/2"},{"type":"function","title":"Lightning.Invocation.search_workorders/1","doc":"","ref":"Lightning.Invocation.html#search_workorders/1"},{"type":"function","title":"Lightning.Invocation.search_workorders/3","doc":"","ref":"Lightning.Invocation.html#search_workorders/3"},{"type":"function","title":"Lightning.Invocation.update_dataclip/2","doc":"Updates a dataclip. Examples iex> update_dataclip ( dataclip , %{ field : new_value } ) { :ok , % Dataclip { } } iex> update_dataclip ( dataclip , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_dataclip/2"},{"type":"function","title":"Lightning.Invocation.update_run/2","doc":"Updates a run. Examples iex> update_run ( run , %{ field : new_value } ) { :ok , % Run { } } iex> update_run ( run , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Invocation.html#update_run/2"},{"type":"function","title":"Lightning.Invocation.with_attempts/1","doc":"","ref":"Lightning.Invocation.html#with_attempts/1"},{"type":"module","title":"Lightning.Invocation.Dataclip","doc":"Ecto model for Dataclips. Dataclips represent some data that arrived in the system, and records both the data and the source of the data. Types :http_request The data arrived via a webhook. :global Was created manually, and is intended to be used multiple times. When repetitive static data is needed to be maintained, instead of hard-coding into a Job - a more convenient solution is to create a :global Dataclip and access it inside the Job. :run_result The final state of a successful run. :saved_input An arbitrary input, created by a user. (Only configuration will be overwritten.)","ref":"Lightning.Invocation.Dataclip.html"},{"type":"function","title":"Lightning.Invocation.Dataclip.get_types/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#get_types/0"},{"type":"function","title":"Lightning.Invocation.Dataclip.new/1","doc":"","ref":"Lightning.Invocation.Dataclip.html#new/1"},{"type":"function","title":"Lightning.Invocation.Dataclip.validate_by_type/1","doc":"Append validations based on the type of the Dataclip. :run_result must have an associated Run model.","ref":"Lightning.Invocation.Dataclip.html#validate_by_type/1"},{"type":"type","title":"Lightning.Invocation.Dataclip.source_type/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:source_type/0"},{"type":"type","title":"Lightning.Invocation.Dataclip.t/0","doc":"","ref":"Lightning.Invocation.Dataclip.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.LogLine","doc":"Ecto model for run logs.","ref":"Lightning.Invocation.LogLine.html"},{"type":"function","title":"Lightning.Invocation.LogLine.validate/1","doc":"","ref":"Lightning.Invocation.LogLine.html#validate/1"},{"type":"type","title":"Lightning.Invocation.LogLine.t/0","doc":"","ref":"Lightning.Invocation.LogLine.html#t:t/0"},{"type":"module","title":"Lightning.Invocation.Query","doc":"Query functions for working with Runs and Dataclips","ref":"Lightning.Invocation.Query.html"},{"type":"function","title":"Lightning.Invocation.Query.last_run_for_job/1","doc":"The last run for a job","ref":"Lightning.Invocation.Query.html#last_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.last_successful_run_for_job/1","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#last_successful_run_for_job/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_for/1","doc":"Runs for a specific user","ref":"Lightning.Invocation.Query.html#runs_for/1"},{"type":"function","title":"Lightning.Invocation.Query.runs_with_code/2","doc":"The last run for a job for a particular exit code, used in scheduler","ref":"Lightning.Invocation.Query.html#runs_with_code/2"},{"type":"module","title":"Lightning.Invocation.Run","doc":"Ecto model for Runs. A run represents the work initiated for a Job with an input dataclip. Once completed (successfully) it will have an output_dataclip associated with it as well.","ref":"Lightning.Invocation.Run.html"},{"type":"function","title":"Lightning.Invocation.Run.new/1","doc":"","ref":"Lightning.Invocation.Run.html#new/1"},{"type":"function","title":"Lightning.Invocation.Run.new_from/1","doc":"Creates a new Run changeset, but copies over certain fields. This is used to create new runs for retrys.","ref":"Lightning.Invocation.Run.html#new_from/1"},{"type":"type","title":"Lightning.Invocation.Run.t/0","doc":"","ref":"Lightning.Invocation.Run.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReason","doc":"Ecto model for InvocationReasons.","ref":"Lightning.InvocationReason.html"},{"type":"function","title":"Lightning.InvocationReason.new/1","doc":"","ref":"Lightning.InvocationReason.html#new/1"},{"type":"function","title":"Lightning.InvocationReason.validate_by_trigger_type/1","doc":"","ref":"Lightning.InvocationReason.html#validate_by_trigger_type/1"},{"type":"type","title":"Lightning.InvocationReason.source_type/0","doc":"","ref":"Lightning.InvocationReason.html#t:source_type/0"},{"type":"type","title":"Lightning.InvocationReason.t/0","doc":"","ref":"Lightning.InvocationReason.html#t:t/0"},{"type":"module","title":"Lightning.InvocationReasons","doc":"The InvocationReasons context.","ref":"Lightning.InvocationReasons.html"},{"type":"function","title":"Lightning.InvocationReasons.build/2","doc":"","ref":"Lightning.InvocationReasons.html#build/2"},{"type":"function","title":"Lightning.InvocationReasons.create_reason/1","doc":"Creates a reason. Examples iex> create_reason ( %{ field : value } ) { :ok , % InvocationReason { } } iex> create_reason ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.InvocationReasons.html#create_reason/1"},{"type":"type","title":"Lightning.InvocationReasons.reason_type/0","doc":"","ref":"Lightning.InvocationReasons.html#t:reason_type/0"},{"type":"module","title":"Lightning.Pipeline","doc":"Service class to coordinate the running of jobs, and their downstream jobs.","ref":"Lightning.Pipeline.html"},{"type":"function","title":"Lightning.Pipeline.assemble_logs_for_run/1","doc":"Return all logs for a run as a string of text, separated by new line breaks","ref":"Lightning.Pipeline.html#assemble_logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.logs_for_run/1","doc":"Return all logs for a run as a list","ref":"Lightning.Pipeline.html#logs_for_run/1"},{"type":"function","title":"Lightning.Pipeline.process/1","doc":"","ref":"Lightning.Pipeline.html#process/1"},{"type":"module","title":"Lightning.Pipeline.Runner","doc":"Job running entrypoint","ref":"Lightning.Pipeline.Runner.html"},{"type":"function","title":"Lightning.Pipeline.Runner.create_dataclip_from_result/2","doc":"Creates a dataclip linked to the run that just finished. If either the file doesn't exist or there is a JSON decoding error, it logs and returns an error tuple.","ref":"Lightning.Pipeline.Runner.html#create_dataclip_from_result/2"},{"type":"function","title":"Lightning.Pipeline.Runner.find_or_install_adaptor/1","doc":"Make sure an adaptor matching the name is available. If it is available, return it's Engine.Adaptor struct - if not then install it.","ref":"Lightning.Pipeline.Runner.html#find_or_install_adaptor/1"},{"type":"function","title":"Lightning.Pipeline.Runner.scrub_result/1","doc":"Scrubs values from all keys in configuration, will be replaced by extensions to scrubber.ex, which is currently only used for logs.","ref":"Lightning.Pipeline.Runner.html#scrub_result/1"},{"type":"function","title":"Lightning.Pipeline.Runner.start/2","doc":"Execute a Run. Given a valid run: Persist the Dataclip and the Job's body to disk Create a blank output file on disk Build up a %Lightning.Runtime.Runspec{} with the paths, and adaptor module name And start it via Handler.start/2 . The callbacks implemented on Handler ( c:Handler.on_start/1 and c:Handler.on_finish/2 ) update the run when a Run is started and when it's finished, attaching the exit_code and log when they are available.","ref":"Lightning.Pipeline.Runner.html#start/2"},{"type":"module","title":"Lightning.Pipeline.Runner.Handler","doc":"Custom handler callbacks for Lightnings use of Engine to execute runs.","ref":"Lightning.Pipeline.Runner.Handler.html"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.on_start/1","doc":"The on_start handler updates the run, setting the started_at time and stamping the run with the ID of the credential that was used, if any, to facilitate easier auditing.","ref":"Lightning.Pipeline.Runner.Handler.html#on_start/1"},{"type":"function","title":"Lightning.Pipeline.Runner.Handler.stop/1","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#stop/1"},{"type":"type","title":"Lightning.Pipeline.Runner.Handler.handler_opts/0","doc":"","ref":"Lightning.Pipeline.Runner.Handler.html#t:handler_opts/0"},{"type":"module","title":"Lightning.Pipeline.StateAssembler","doc":"Facilities for building the state for a Run How state is assembled For the most common jobs, an inbound webhook will store an :http_request type dataclip. The reason that is created is associated with the dataclip. At runtime, the initial state for a Run will be in the shape of: { "data" : < the dataclip > , "configuration" : < the job ' s credential > } Saved inputs Saved custom inputs will only have state.configuration changed, everything else will remain as displayed. Flow Jobs When a Job is triggered by a previous Jobs success or failure these are the rules for constructing that Jobs state: For jobs that trigger on it's upstream jobs failure, the event will have the previous runs input dataclip as its input dataclip. The state will also have the log of the previous run attached on the error key. For Jobs triggered by a previous success, the run will have the previous runs output dataclip as its input dataclip. :run_result dataclips are expected to already have a data key, and are merged into the root.","ref":"Lightning.Pipeline.StateAssembler.html"},{"type":"function","title":"Lightning.Pipeline.StateAssembler.assemble/1","doc":"Assemble state for use in a Run.","ref":"Lightning.Pipeline.StateAssembler.html#assemble/1"},{"type":"module","title":"Lightning.Jobs","doc":"The Jobs context.","ref":"Lightning.Jobs.html"},{"type":"function","title":"Lightning.Jobs.change_job/2","doc":"Returns an %Ecto.Changeset{} for tracking job changes. Examples iex> change_job ( job ) % Ecto.Changeset { data : % Job { } }","ref":"Lightning.Jobs.html#change_job/2"},{"type":"function","title":"Lightning.Jobs.create_job/1","doc":"Creates a job. Examples iex> create_job ( %{ field : value } ) { :ok , % Job { } } iex> create_job ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#create_job/1"},{"type":"function","title":"Lightning.Jobs.delete_job/1","doc":"Deletes a job. Examples iex> delete_job ( job ) { :ok , % Job { } } iex> delete_job ( job ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#delete_job/1"},{"type":"function","title":"Lightning.Jobs.get_downstream_jobs_for/2","doc":"Returns the list of downstream jobs for a given job, optionally matching a specific trigger type. When downstream_jobs_for is called without a trigger that means its between jobs when it called with a trigger that means we are starting from outside the pipeline","ref":"Lightning.Jobs.html#get_downstream_jobs_for/2"},{"type":"function","title":"Lightning.Jobs.get_job/1","doc":"","ref":"Lightning.Jobs.html#get_job/1"},{"type":"function","title":"Lightning.Jobs.get_job!/1","doc":"Gets a single job. Raises Ecto.NoResultsError if the Job does not exist. Examples iex> get_job! ( 123 ) % Job { } iex> get_job! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Jobs.html#get_job!/1"},{"type":"function","title":"Lightning.Jobs.get_upstream_jobs_for/1","doc":"Returns the list of jobs excluding the one given.","ref":"Lightning.Jobs.html#get_upstream_jobs_for/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project/1"},{"type":"function","title":"Lightning.Jobs.jobs_for_project_query/1","doc":"","ref":"Lightning.Jobs.html#jobs_for_project_query/1"},{"type":"function","title":"Lightning.Jobs.list_active_cron_jobs/0","doc":"","ref":"Lightning.Jobs.html#list_active_cron_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs/0","doc":"Returns the list of jobs.","ref":"Lightning.Jobs.html#list_jobs/0"},{"type":"function","title":"Lightning.Jobs.list_jobs_for_workflow/1","doc":"","ref":"Lightning.Jobs.html#list_jobs_for_workflow/1"},{"type":"function","title":"Lightning.Jobs.update_job/2","doc":"Updates a job. Examples iex> update_job ( job , %{ field : new_value } ) { :ok , % Job { } } iex> update_job ( job , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Jobs.html#update_job/2"},{"type":"module","title":"Lightning.Jobs.Job","doc":"Ecto model for Jobs. A Job contains the fields for defining a job. body The expression/javascript code name A plain text identifier adaptor An NPM style string that contains both the module name and it's version. E.g. @openfn/language-http@v1.2.3 or @openfn/language-foo@latest . While the version suffix isn't enforced here as it's not strictly necessary in this context, the front end will ensure a version is stated ( @latest being the default).","ref":"Lightning.Jobs.Job.html"},{"type":"function","title":"Lightning.Jobs.Job.new/1","doc":"","ref":"Lightning.Jobs.Job.html#new/1"},{"type":"function","title":"Lightning.Jobs.Job.put_project_credential/2","doc":"","ref":"Lightning.Jobs.Job.html#put_project_credential/2"},{"type":"function","title":"Lightning.Jobs.Job.put_workflow/2","doc":"Attaches a workflow to a job, this is useful when you have an unpersisted Workflow changeset - and want it to be created at the same time as a Job. Example: workflow = Ecto.Changeset . cast ( % Lightning.Workflows.Workflow { } , %{ "project_id" => attrs [ :project_id ] , "id" => Ecto.UUID . generate ( ) } , [ :project_id , :id ] ) job = % Job { } |> Ecto.Changeset . change ( ) |> Job . put_workflow ( workflow ) |> Job . changeset ( attrs )","ref":"Lightning.Jobs.Job.html#put_workflow/2"},{"type":"function","title":"Lightning.Jobs.Job.validate/1","doc":"","ref":"Lightning.Jobs.Job.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Job.t/0","doc":"","ref":"Lightning.Jobs.Job.html#t:t/0"},{"type":"module","title":"Lightning.Jobs.Query","doc":"Query module for finding Jobs.","ref":"Lightning.Jobs.Query.html"},{"type":"function","title":"Lightning.Jobs.Query.enabled_cron_jobs_by_edge/0","doc":"Returns active jobs with their cron triggers for use in the cron scheduling service.","ref":"Lightning.Jobs.Query.html#enabled_cron_jobs_by_edge/0"},{"type":"function","title":"Lightning.Jobs.Query.jobs_for/1","doc":"Returns all jobs accessible to a user, via their projects or all jobs in a given project.","ref":"Lightning.Jobs.Query.html#jobs_for/1"},{"type":"module","title":"Lightning.Jobs.Scheduler","doc":"The Scheduler is responsible for finding jobs that are ready to run based on their cron schedule, and then running them.","ref":"Lightning.Jobs.Scheduler.html"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/0","doc":"Find and start any cronjobs that are scheduled to run for a given time (defaults to the current time).","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/0"},{"type":"function","title":"Lightning.Jobs.Scheduler.enqueue_cronjobs/1","doc":"","ref":"Lightning.Jobs.Scheduler.html#enqueue_cronjobs/1"},{"type":"module","title":"Lightning.Jobs.Trigger","doc":"Ecto model for Triggers. Triggers represent the criteria in which a Job might be invoked. Types Webhook (default) A webhook trigger allows a Job to invoked (via Lightning.Invocation ) when it's endpoint is called.","ref":"Lightning.Jobs.Trigger.html"},{"type":"function","title":"Lightning.Jobs.Trigger.new/1","doc":"","ref":"Lightning.Jobs.Trigger.html#new/1"},{"type":"function","title":"Lightning.Jobs.Trigger.validate/1","doc":"","ref":"Lightning.Jobs.Trigger.html#validate/1"},{"type":"type","title":"Lightning.Jobs.Trigger.t/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:t/0"},{"type":"type","title":"Lightning.Jobs.Trigger.trigger_type/0","doc":"","ref":"Lightning.Jobs.Trigger.html#t:trigger_type/0"},{"type":"module","title":"Lightning.Projects","doc":"The Projects context.","ref":"Lightning.Projects.html"},{"type":"function","title":"Lightning.Projects.cancel_scheduled_deletion/1","doc":"","ref":"Lightning.Projects.html#cancel_scheduled_deletion/1"},{"type":"function","title":"Lightning.Projects.change_project/2","doc":"Returns an %Ecto.Changeset{} for tracking project changes. Examples iex> change_project ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#change_project/2"},{"type":"function","title":"Lightning.Projects.create_project/1","doc":"Creates a project. Examples iex> create_project ( %{ field : value } ) { :ok , % Project { } } iex> create_project ( %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#create_project/1"},{"type":"function","title":"Lightning.Projects.delete_project/1","doc":"Deletes a project and its related data, including workflows, work orders, runs, jobs, attempts, triggers, project users, project credentials, and dataclips Examples iex> delete_project ( project ) { :ok , % Project { } } iex> delete_project ( project ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#delete_project/1"},{"type":"function","title":"Lightning.Projects.export_project/2","doc":"Exports a project as yaml. Examples iex> export_project ( :yaml , project_id ) { :ok , string }","ref":"Lightning.Projects.html#export_project/2"},{"type":"function","title":"Lightning.Projects.get_project/1","doc":"","ref":"Lightning.Projects.html#get_project/1"},{"type":"function","title":"Lightning.Projects.get_project!/1","doc":"Gets a single project. Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project!/1"},{"type":"function","title":"Lightning.Projects.get_project_credential/2","doc":"","ref":"Lightning.Projects.html#get_project_credential/2"},{"type":"function","title":"Lightning.Projects.get_project_user/1","doc":"","ref":"Lightning.Projects.html#get_project_user/1"},{"type":"function","title":"Lightning.Projects.get_project_user/2","doc":"","ref":"Lightning.Projects.html#get_project_user/2"},{"type":"function","title":"Lightning.Projects.get_project_user!/1","doc":"Gets a single project_user. Raises Ecto.NoResultsError if the ProjectUser does not exist. Examples iex> get_project_user! ( 123 ) % ProjectUser { } iex> get_project_user! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_user!/1"},{"type":"function","title":"Lightning.Projects.get_project_user_role/2","doc":"Returns the role of a user in a project. Possible roles are :admin, :viewer, :editor, and :owner Examples iex> get_project_user_role ( user , project ) :admin iex> get_project_user_role ( user , project ) :viewer iex> get_project_user_role ( user , project ) :editor iex> get_project_user_role ( user , project ) :owner","ref":"Lightning.Projects.html#get_project_user_role/2"},{"type":"function","title":"Lightning.Projects.get_project_with_users!/1","doc":"Gets a single project with it's members via project_users . Raises Ecto.NoResultsError if the Project does not exist. Examples iex> get_project! ( 123 ) % Project { } iex> get_project! ( 456 ) ** (Ecto.NoResultsError)","ref":"Lightning.Projects.html#get_project_with_users!/1"},{"type":"function","title":"Lightning.Projects.get_projects_for_user/1","doc":"","ref":"Lightning.Projects.html#get_projects_for_user/1"},{"type":"function","title":"Lightning.Projects.is_member_of?/2","doc":"","ref":"Lightning.Projects.html#is_member_of?/2"},{"type":"function","title":"Lightning.Projects.list_project_credentials/1","doc":"","ref":"Lightning.Projects.html#list_project_credentials/1"},{"type":"function","title":"Lightning.Projects.list_projects/0","doc":"Returns the list of projects. Examples iex> list_projects ( ) [ % Project { } , ... ]","ref":"Lightning.Projects.html#list_projects/0"},{"type":"function","title":"Lightning.Projects.perform/1","doc":"Perform, when called with %{"type" => "purge_deleted"} will find projects that are ready for permanent deletion and purge them.","ref":"Lightning.Projects.html#perform/1"},{"type":"function","title":"Lightning.Projects.project_attempt_run_query/1","doc":"","ref":"Lightning.Projects.html#project_attempt_run_query/1"},{"type":"function","title":"Lightning.Projects.project_attempts_query/1","doc":"","ref":"Lightning.Projects.html#project_attempts_query/1"},{"type":"function","title":"Lightning.Projects.project_credentials_query/1","doc":"","ref":"Lightning.Projects.html#project_credentials_query/1"},{"type":"function","title":"Lightning.Projects.project_dataclip_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_dataclip_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_dataclips_query/1","doc":"","ref":"Lightning.Projects.html#project_dataclips_query/1"},{"type":"function","title":"Lightning.Projects.project_jobs_query/1","doc":"","ref":"Lightning.Projects.html#project_jobs_query/1"},{"type":"function","title":"Lightning.Projects.project_run_invocation_reasons/1","doc":"","ref":"Lightning.Projects.html#project_run_invocation_reasons/1"},{"type":"function","title":"Lightning.Projects.project_runs_query/1","doc":"","ref":"Lightning.Projects.html#project_runs_query/1"},{"type":"function","title":"Lightning.Projects.project_trigger_invocation_reason/1","doc":"","ref":"Lightning.Projects.html#project_trigger_invocation_reason/1"},{"type":"function","title":"Lightning.Projects.project_triggers_query/1","doc":"","ref":"Lightning.Projects.html#project_triggers_query/1"},{"type":"function","title":"Lightning.Projects.project_user_role_query/2","doc":"","ref":"Lightning.Projects.html#project_user_role_query/2"},{"type":"function","title":"Lightning.Projects.project_users_query/1","doc":"","ref":"Lightning.Projects.html#project_users_query/1"},{"type":"function","title":"Lightning.Projects.project_workflows_query/1","doc":"","ref":"Lightning.Projects.html#project_workflows_query/1"},{"type":"function","title":"Lightning.Projects.project_workorders_query/1","doc":"","ref":"Lightning.Projects.html#project_workorders_query/1"},{"type":"function","title":"Lightning.Projects.projects_for_user_query/1","doc":"","ref":"Lightning.Projects.html#projects_for_user_query/1"},{"type":"function","title":"Lightning.Projects.schedule_project_deletion/1","doc":"Given a project, this function sets a scheduled deletion date based on the PURGE_DELETED_AFTER_DAYS environment variable. If no ENV is set, this date defaults to NOW but the automatic project purge cronjob will never run. (Note that subsequent logins will be blocked for projects pending deletion.)","ref":"Lightning.Projects.html#schedule_project_deletion/1"},{"type":"function","title":"Lightning.Projects.select_first_project_for_user/1","doc":"","ref":"Lightning.Projects.html#select_first_project_for_user/1"},{"type":"function","title":"Lightning.Projects.update_project/2","doc":"Updates a project. Examples iex> update_project ( project , %{ field : new_value } ) { :ok , % Project { } } iex> update_project ( project , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project/2"},{"type":"function","title":"Lightning.Projects.update_project_user/2","doc":"Updates a project user. Examples iex> update_project_user ( project_user , %{ field : new_value } ) { :ok , % ProjectUser { } } iex> update_project_user ( projectUser , %{ field : bad_value } ) { :error , % Ecto.Changeset { } }","ref":"Lightning.Projects.html#update_project_user/2"},{"type":"function","title":"Lightning.Projects.url_safe_project_name/1","doc":"","ref":"Lightning.Projects.html#url_safe_project_name/1"},{"type":"function","title":"Lightning.Projects.validate_for_deletion/2","doc":"Returns an %Ecto.Changeset{} for changing the project scheduled_deletion. Examples iex> validate_for_deletion ( project ) % Ecto.Changeset { data : % Project { } }","ref":"Lightning.Projects.html#validate_for_deletion/2"},{"type":"module","title":"Lightning.Projects.Project","doc":"Project model","ref":"Lightning.Projects.Project.html"},{"type":"function","title":"Lightning.Projects.Project.deletion_changeset/2","doc":"Changeset to validate a project deletion request, the user must enter the projects name to confirm.","ref":"Lightning.Projects.Project.html#deletion_changeset/2"},{"type":"function","title":"Lightning.Projects.Project.validate/1","doc":"","ref":"Lightning.Projects.Project.html#validate/1"},{"type":"type","title":"Lightning.Projects.Project.t/0","doc":"","ref":"Lightning.Projects.Project.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectCredential","doc":"Join table to assign credentials to a project","ref":"Lightning.Projects.ProjectCredential.html"},{"type":"type","title":"Lightning.Projects.ProjectCredential.t/0","doc":"","ref":"Lightning.Projects.ProjectCredential.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser","doc":"Join table to assign users to a project","ref":"Lightning.Projects.ProjectUser.html"},{"type":"type","title":"Lightning.Projects.ProjectUser.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.DigestEnum","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.DigestEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.DigestEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.DigestEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.ProjectUser.RolesEnum","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.cast/1","doc":"Callback implementation for Ecto.Type.cast/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#cast/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.create_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#create_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.drop_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#drop_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.dump/1","doc":"Callback implementation for Ecto.Type.dump/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#dump/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.embed_as/1","doc":"Callback implementation for Ecto.Type.embed_as/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#embed_as/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.equal?/2","doc":"Callback implementation for Ecto.Type.equal?/2 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#equal?/2"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.load/1","doc":"Callback implementation for Ecto.Type.load/1 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#load/1"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schema/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schema/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.schemaless_type/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#schemaless_type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.type/0","doc":"Callback implementation for Ecto.Type.type/0 .","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#type/0"},{"type":"function","title":"Lightning.Projects.ProjectUser.RolesEnum.valid_value?/1","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#valid_value?/1"},{"type":"type","title":"Lightning.Projects.ProjectUser.RolesEnum.t/0","doc":"","ref":"Lightning.Projects.ProjectUser.RolesEnum.html#t:t/0"},{"type":"module","title":"Lightning.Projects.Provisioner","doc":"Provides functions for importing projects. This module is used by the provisioning HTTP API. When providing a project to import, all records must have an id field. It's up to the caller to ensure that the id is unique and generated ahead of time in the case of new records.","ref":"Lightning.Projects.Provisioner.html"},{"type":"function","title":"Lightning.Projects.Provisioner.import_document/3","doc":"Import a project.","ref":"Lightning.Projects.Provisioner.html#import_document/3"},{"type":"function","title":"Lightning.Projects.Provisioner.load_project/1","doc":"Load a project by ID, including all workflows and their associated jobs, triggers and edges. Returns nil if the project does not exist.","ref":"Lightning.Projects.Provisioner.html#load_project/1"},{"type":"function","title":"Lightning.Projects.Provisioner.parse_document/2","doc":"","ref":"Lightning.Projects.Provisioner.html#parse_document/2"},{"type":"function","title":"Lightning.Projects.Provisioner.validate_extraneous_params/1","doc":"Validate that there are no extraneous parameters in the changeset. For all params in the changeset, ensure that the param is in the list of known fields in the schema.","ref":"Lightning.Projects.Provisioner.html#validate_extraneous_params/1"},{"type":"module","title":"Lightning.Runtime.ChildProcess","doc":"Provides an interface between a RunSpec and the shell. Internally it calls node , and more specifically the OpenFn core CLI.","ref":"Lightning.Runtime.ChildProcess.html"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_command/1","doc":"Builds up a string for shell execution based on the RunSpec","ref":"Lightning.Runtime.ChildProcess.html#build_command/1"},{"type":"function","title":"Lightning.Runtime.ChildProcess.build_env/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#build_env/2"},{"type":"function","title":"Lightning.Runtime.ChildProcess.run/2","doc":"","ref":"Lightning.Runtime.ChildProcess.html#run/2"},{"type":"behaviour","title":"Lightning.Runtime.Handler","doc":"A strategy for executing things via ChildProcess. This module handles the dirty bits, setting up processes and coordinating results (and logs) as they arrive. Since it is a macro, see Lightning.Pipeline.Runner.Handler for a usage example.","ref":"Lightning.Runtime.Handler.html"},{"type":"function","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.env/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:env/2"},{"type":"callback","title":"Lightning.Runtime.Handler.log_callback/3","doc":"","ref":"Lightning.Runtime.Handler.html#c:log_callback/3"},{"type":"function","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_finish/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_finish/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_finish/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#on_log_emit/2"},{"type":"callback","title":"Lightning.Runtime.Handler.on_log_emit/2","doc":"","ref":"Lightning.Runtime.Handler.html#c:on_log_emit/2"},{"type":"function","title":"Lightning.Runtime.Handler.on_start/1","doc":"","ref":"Lightning.Runtime.Handler.html#on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.on_start/1","doc":"Called with context, if any - when the Run has been started.","ref":"Lightning.Runtime.Handler.html#c:on_start/1"},{"type":"callback","title":"Lightning.Runtime.Handler.start/2","doc":"The entrypoint for executing a run.","ref":"Lightning.Runtime.Handler.html#c:start/2"},{"type":"type","title":"Lightning.Runtime.Handler.t/0","doc":"","ref":"Lightning.Runtime.Handler.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.LogAgent","doc":"Agent facility to consume STDOUT/STDERR byte by byte. Since it works on a byte by byte basis, you will need to perform line-splitting yourself. Usage: { :ok , log } = LogAgent . start_link ( ) "foo" = LogAgent . process_chunk ( log , { :stdout , "foo" } ) "foobar" = LogAgent . process_chunk ( log , { :stdout , "bar" } )","ref":"Lightning.Runtime.LogAgent.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.child_spec/1","doc":"Returns a specification to start this module under a supervisor. See Supervisor .","ref":"Lightning.Runtime.LogAgent.html#child_spec/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.start_link/1","doc":"","ref":"Lightning.Runtime.LogAgent.html#start_link/1"},{"type":"type","title":"Lightning.Runtime.LogAgent.logline/0","doc":"","ref":"Lightning.Runtime.LogAgent.html#t:logline/0"},{"type":"module","title":"Lightning.Runtime.LogAgent.StringBuffer","doc":"Internal datastructure to hold and process new bytes for a list of characters. By checking the if the buffer is a complete grapheme, emitting the buffer once valid and returning nil otherwise. In the case of emojis and other language character sets, a character (in UTF-8) can be between 1-4 bytes - when streaming logs for example it's quite easy to receive less than the whole character which can result in crashes or corrupt text.","ref":"Lightning.Runtime.LogAgent.StringBuffer.html"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.buffer/1","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#buffer/1"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.new/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#new/0"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.process_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#process_chunk/2"},{"type":"function","title":"Lightning.Runtime.LogAgent.StringBuffer.reduce_chunk/2","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#reduce_chunk/2"},{"type":"type","title":"Lightning.Runtime.LogAgent.StringBuffer.t/0","doc":"","ref":"Lightning.Runtime.LogAgent.StringBuffer.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.Result","doc":"Data structure used to represent the result of a Run executed by Lightning.Runtime.ChildProcess .","ref":"Lightning.Runtime.Result.html"},{"type":"function","title":"Lightning.Runtime.Result.new/1","doc":"","ref":"Lightning.Runtime.Result.html#new/1"},{"type":"type","title":"Lightning.Runtime.Result.t/0","doc":"","ref":"Lightning.Runtime.Result.html#t:t/0"},{"type":"module","title":"Lightning.Runtime.RunSpec","doc":"A struct containing all the parameters required to execute a Job.","ref":"Lightning.Runtime.RunSpec.html"},{"type":"function","title":"Lightning.Runtime.RunSpec.new/1","doc":"","ref":"Lightning.Runtime.RunSpec.html#new/1"},{"type":"type","title":"Lightning.Runtime.RunSpec.t/0","doc":"","ref":"Lightning.Runtime.RunSpec.html#t:t/0"},{"type":"exception","title":"Lightning.MetadataService.Error","doc":"","ref":"Lightning.MetadataService.Error.html"},{"type":"function","title":"Lightning.MetadataService.Error.message/1","doc":"Callback implementation for Exception.message/1 .","ref":"Lightning.MetadataService.Error.html#message/1"},{"type":"function","title":"Lightning.MetadataService.Error.new/1","doc":"","ref":"Lightning.MetadataService.Error.html#new/1"},{"type":"type","title":"Lightning.MetadataService.Error.t/0","doc":"","ref":"Lightning.MetadataService.Error.html#t:t/0"},{"type":"extras","title":"Lightning","doc":"OpenFn/Lightning is a fully open source, workflow automation platform that's used to automate critical business processes and integrate information systems. From last-mile services to national-level reporting, it boosts efficiency & effectiveness while enabling secure, stable, scalable interoperability at all levels. Read more about Lightning on OpenFn/Docs . Lightning is the latest "v2" of OpenFn, the Digital Public Good for workflow automation . Use Lightning to visually build, execute and manage workflows. The latest version of the OpenFn technology - first launched in 2014, now tried and tested by NGOs in over 40 countries Fully open source (no premium features or community edition, you get the same product whether using SaaS or self-hosted) Recognised as a Digital Public Good by the DPGA and a Global Good for Health by Digital Square","ref":"readme.html"},{"type":"extras","title":"Lightning - Contents","doc":"Demo Sign up for the BETA Features Getting started Run Lightning via Docker Deploy Lightning on Docker or Kubernetes Run Lightning on your local machine (contributors) Troubleshooting Contribute Project Configuration Quickstart Generate the documentation Security and standards Server specs for self-hosting Questions or feedback?","ref":"readme.html#contents"},{"type":"extras","title":"Lightning - Demo","doc":"Watch a quick demo here: https://www.youtube.com/watch?v=BNaxlHAWb5I Explore our demo app* with username: demo@openfn.org , password: welcome123 , or read through the features section to view screenshots of the app. *Note that the demo app refreshes daily, so do not configure workflows you want to save.","ref":"readme.html#demo"},{"type":"extras","title":"Lightning - Register for a Beta account","doc":"Register for a Beta account at app.openfn.org and go through the quick-start guide to get familiar with the app.","ref":"readme.html#register-for-a-beta-account"},{"type":"extras","title":"Lightning - Features","doc":"Build Plan and build workflows using Lightning's visual interface to quickly define when, where and what you want your automation to do. Use our CLI to quickly build, edit and deploy projects from the comfort of your own code editor. Monitor Monitor all workflow activity in one place. Filter and search runs to identify issues that need addressing and follow how a specific request has been processed Configure alerts to be notified on run failures Receive a project digest for a daily/weekly/monthly summary of your project activity Manage Manage users and access by project Roles and permissions Authorization is a central part of Lightning. As such, users are given different roles which determine what level of access they have for resources in the application. For more details about roles and permissions in Lightning, please refer to our documentation . Roadmap View our public roadmap here .","ref":"readme.html#features"},{"type":"extras","title":"Lightning - Getting Started","doc":"If you only want to RUN Lightning on your own server, we recommend using Docker. If you want to DEPLOY Lightning, we recommend Docker builds and Kubernetes. If you want to CONTRIBUTE to the project, we recommend running Lightning on your local machine .","ref":"readme.html#getting-started"},{"type":"extras","title":"Lightning - Run via Docker","doc":"Install the latest version of Docker Clone this repo using git Copy the .env.example file to .env Run docker compose run --rm web mix ecto.migrate By default the application will be running at localhost:4000 . You can then rebuild and run with docker compose build and docker compose up . See "Problems with Docker" for additional troubleshooting help. Note that you can also create your own docker-compose.yml file, configuring a postgres database and using a pre-built image from Dockerhub.","ref":"readme.html#run-via-docker"},{"type":"extras","title":"Lightning - Deploy on external infrastructure","doc":"See Deployment for more detailed information.","ref":"readme.html#deploy-on-external-infrastructure"},{"type":"extras","title":"Lightning - Run Lightning locally","doc":"Clone the repo and optionally set ENVs git clone git@github.com:OpenFn/Lightning.git # or from YOUR fork! cd Lightning cp .env.example .env # and adjust as necessary! Take note of database names and ports in particular—they've got to match across your Postgres setup and your ENVs. You can run lightning without any ENVs assuming a vanilla postgres setup (see below), but you may want to make adjustments. Database Setup If you're already using Postgres locally, create a new database called lightning_dev , for example. If you'd rather use Docker to set up a Postgres DB, create a new volume and image: docker volume create lightning-postgres-data docker create \\ --name lightning-postgres \\ --mount source=lightning-postgres-data,target=/var/lib/postgresql/data \\ --publish 5432:5432 \\ -e POSTGRES_PASSWORD=postgres \\ postgres:15.3-alpine docker start lightning-postgres Elixir & Ecto Setup We use asdf to configure our local environments. Included in the repo is a .tool-versions file that is read by asdf in order to dynamically make the specified versions of Elixir and Erlang available. You'll need asdf plugins for Erlang , NodeJs Elixir and k6 . asdf install # Install language versions mix local.hex mix deps.get mix local.rebar --force mix ecto.create # Create a development database in Postgres mix ecto.migrate [[ $(uname -m) == 'arm64' ]] && mix compile.rambo # Force compile rambo if on M1 mix lightning.install_runtime mix lightning.install_schemas npm install --prefix assets Run the app Lightning is a web app. To run it in interactive Elixir mode, start the development server by running with your environment variables by running: iex -S mix phx.server or if you have set up custom environment variables, run: env $(cat .env | grep -v "#" | xargs ) iex -S mix phx.server Once the server has started, head to localhost:4000 in your browser. Run the tests Before the first time running the tests, you need a test database setup. MIX_ENV=test mix ecto.create And then after that run the tests using: MIX_ENV=test mix test We also have test.watch installed which can be used to rerun the tests on file changes.","ref":"readme.html#run-lightning-locally"},{"type":"extras","title":"Lightning - Benchmarking","doc":"We are using k6 to benchmark Lightning. Under benchmarking folder you can find a script for benchmarking Webhook Workflows. See Benchmarking for more detailed information.","ref":"readme.html#benchmarking"},{"type":"extras","title":"Lightning - Troubleshooting","doc":"Trouble with environment variables For troubleshooting custom environment variable configuration it's important to know how an Elixir app loads and modifies configuration. The order is as follows: Stuff in config.exs is loaded. That is then modified (think: overwritten ) by stuff your ENV-specific config: dev.exs , prod.exs or test.exs . That is then modified by runtime.exs which is where you are allowed to use System.env() Finally init/2 (if present in a child application) gets called (which takes the config which has been set in steps 1-3) when that child application is started during the parent app startup defined in application.ex . Problems with Postgres If you're having connecting issues with Postgres, check the database section of your .env to ensure the DB url is correctly set for your environment — note that composing a DB url out of other, earlier declared variables, does not work while using xargs . Problems with Debian If you're getting this error on debian == > earmark_parser Compiling 1 file ( . yrl ) / usr / lib / erlang / lib / parsetools - 2.3 . 1 / include / yeccpre . hrl : no such file or directory could not compile dependency :earmark_parser , "mix compile" failed . You can recompile this dependency with "mix deps.compile earmark_parser" , update it with "mix deps.update earmark_parser" or clean it with "mix deps.clean earmark_parser" You need to install erlang development environment sudo apt install erlang-dev refer to this issue Problems with Docker Versions The build may not work on old versions of Docker and Docker compose. It has been tested against: Docker version 20.10 . 17 , build 100 c701 Docker Compose version v2 . 6.0 Problems with rambo When running mix compile.rambo on Apple Silicon (an Apple M1/M2, macarm , aarch64-apple-darwin ) and encountering the following error: ** (RuntimeError) Rambo does not ship with binaries for your environment. aarch64 - apple - darwin22 . 3.0 detected Install the Rust compiler so a binary can be prepared for you . lib / mix / tasks / compile . rambo . ex : 89 : Mix.Tasks.Compile.Rambo . compile! / 0 lib / mix / tasks / compile . rambo . ex : 51 : Mix.Tasks.Compile.Rambo . run / 1 ( mix 1.14 . 2 ) lib / mix / task . ex : 421 : anonymous fn / 3 in Mix.Task . run_task / 4 ( mix 1.14 . 2 ) lib / mix / cli . ex : 84 : Mix.CLI . run_task / 2 You can resolve this error by installing the Rust compiler using Homebrew. Run the following command in your terminal: brew install rust If you have already compiled Rambo explicitly via mix compile.rambo , and you are still seeing the following error: sh : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : No such file or directory sh : line 0 : exec : / path_to_directory / Lightning / _build / dev / lib / rambo / priv / rambo : cannot execute : No such file or directory You can try renaming deps/rambo/priv/rambo-mac to deps/rambo/priv/rambo . If neither of the approaches above work, please raise an issue. Starting from scratch If you're actively working with docker, you start experiencing issues, and you would like to start from scratch you can clean up everything and start over like this: # To remove any ignored files and reset your .env to it's example git clean -fdx && cp .env.example .env # You can skip the line below if you want to keep your database docker compose down --rmi all --volumes docker compose build --no-cache web && \\ docker compose create --force-recreate docker compose run --rm web mix ecto.migrate docker compose up","ref":"readme.html#troubleshooting"},{"type":"extras","title":"Lightning - Project Configuration Quickstart","doc":"For help getting started with your OpenFn/Lightning Project and Workflows configuration, check out the Lightning Quickstart Guidance on OpenFn/Docs.","ref":"readme.html#project-configuration-quickstart"},{"type":"extras","title":"Lightning - Contribute to this project","doc":"First, thanks for being here! You're contributing to a digital public good that will always be free and open source and aimed at serving innovative NGOs, governments, and social impact organizations the world over! You rock. ❤️ FYI, Lightning is built in Elixir , harnessing the Phoenix Framework . Currently, the only unbundled dependency is a PostgreSQL database. If you'd like to contribute to this projects, follow the steps below: Assign yourself to an issue Read through the existing issues , assign yourself to the issue you have chosen. Leave a comment on the issue to let us know you'll be working on it, and if you have any questions of clarifications that would help you get started ask them there - we will get back to you as soon as possible. If there isn't already an issue for the feature you would like to contribute, please start a discussion in our community forum . Open a pull request Clone the Lightning repository, then fork it . Run through setting up your environment and make your changes. Make sure you have written your tests and updated /CHANGELOG.md (in the 'Unreleased' section, add a short description of the changes you are making, along with a link to your issue). Open a draft pull request by clicking "Contribute > Open Pull Request" from your forked repository. Fill out the pull request template (this will be added automatically for you), then make sure to self-review your code and go through the 'Review checklist'. Don't worry about the QA checkbox, our product manager Amber will tick that once she has reviewed your PR. You can leave any notes for the reviewer in a comment. Once you're ready to submit a pull request, you can mark your draft PR as 'Ready for review' and assign @stuartc or @taylordowns2000.","ref":"readme.html#contribute-to-this-project"},{"type":"extras","title":"Lightning - Generating Documentation","doc":"You can generate the HTML and EPUB documentation locally using: mix docs and opening doc/index.html in your browser.","ref":"readme.html#generating-documentation"},{"type":"extras","title":"Lightning - Security and Standards","doc":"We use a host of common Elixir static analysis tools to help us avoid common pitfalls and make sure we keep everything clean and consistent. In addition to our test suite, you can run the following commands: mix format --check-formatted Code formatting checker, run again without the --check-formatted flag to have your code automatically changed. mix dialyzer Static analysis for type mismatches and other common warnings. See dialyxir . mix credo Static analysis for consistency, and coding standards. See Credo . mix sobelow Check for commonly known security exploits. See Sobelow . MIX_ENV=test mix coveralls Test coverage reporter. This command also runs the test suite, and can be used in place of mix test when checking everything before pushing your code. See excoveralls . For convenience there is a verify mix task that runs all of the above and defaults the MIX_ENV to test . For more guidance on security best practices for workflow automation implementations, check out OpenFn Docs: docs.openfn.org/documentation/getting-started/security","ref":"readme.html#security-and-standards"},{"type":"extras","title":"Lightning - Server Specs for Self-Hosting","doc":"For recommend server specifications for self-hosting of Lightning, check out this Community topic: community.openfn.org/t/specs-for-self-hosting-lightning/292","ref":"readme.html#server-specs-for-self-hosting"},{"type":"extras","title":"Lightning - Support","doc":"If you have any questions, feedback, or issues, please: Post on the OpenFn Community at community.openfn.org Open an issue directly on this Github Repo: github.com/OpenFn/Lightning/issues","ref":"readme.html#support"},{"type":"extras","title":"Deployment","doc":"","ref":"deployment.html"},{"type":"extras","title":"Deployment - Encryption","doc":"Lightning enforces encryption at rest for Credentials, for which an encryption key must be provided when running in production. The key is expected to be a randomized set of bytes, 32 long; and Base64 encoded when setting the environment variable. There is a mix task that can generate keys in the correct shape for use as an environment variable: mix lightning.gen_encryption_key 0bJ9w+hn4ebQrsCaWXuA9JY49fP9kbHmywGd5K7k+/s= Copy your key (NOT THIS ONE) and set it as PRIMARY_ENCRYPTION_KEY in your environment.","ref":"deployment.html#encryption"},{"type":"extras","title":"Deployment - Environment Variables","doc":"Note that for secure deployments, it's recommended to use a combination of secrets and configMaps to generate secure environment variables. ADAPTORS_PATH - where you store your locally installed adaptors DISABLE_DB_SSL - in production the use of an SSL conntection to Postgres is required by default, setting this to "true" allows unencrypted connections to the database. This is strongly discouraged in real production environment. K8S_HEADLESS_SERVICE - this environment variable is automatically set if you're running on GKE and it is used to establish an Erlang node cluster. Note that if you're not using Kubernetes, the "gossip" strategy is used for establish clusters. LISTEN_ADDRESS " - the address the web server should bind to, defaults to 127.0.0.1 to block access from other machines. LOG_LEVEL - how noisy you want the logs to be (e.g. debug , info ) MAX_RUN_DURATION - the maximum time (in milliseconds) that jobs are allowed to run (keep this below your termination_grace_period if using kubernetes) MIX_ENV - your mix env, likely prod for deployment NODE_ENV - node env, likely production for deployment ORIGINS - the allowed origins for web traffic to the backend PORT - the port your Phoenix app runs on PRIMARY_ENCRYPTION_KEY - a base64 encoded 32 character long string. See Encryption . SCHEMAS_PATH - path to the credential schemas that provide forms for different adaptors SECRET_KEY_BASE - a secret key used as a base to generate secrets for encrypting and signing data. SENTRY_DSN - if using Sentry for error monitoring, your DSN URL_HOST - the host, used for writing urls (e.g., demo.openfn.org ) URL_PORT - the port, usually 443 for production URL_SCHEME - the scheme for writing urls, (e.g., https ) Google Using your Google Cloud account, provision a new OAuth 2.0 Client with the 'Web application' type. Set the callback url to: https://<ENDPOINT DOMAIN>/authenticate/callback . Replacing ENDPOINT DOMAIN with the host name of your instance. Once the client has been created, get/download the OAuth client JSON and set the following environment variables: GOOGLE_CLIENT_ID - Which is client_id from the client details. GOOGLE_CLIENT_SECRET - client_secret from the client details.","ref":"deployment.html#environment-variables"},{"type":"extras","title":"Benchmarking","doc":"Execute the following steps to run a benchmark on Lightning: Make sure you have k6 installed locally. If you're using asdf you can run asdf install in the project root. Spin up your Lightning local instance Run the demo setup script: mix run --no-start priv/repo/demo.exs The webhookURL is already set to default to the webhook created in the demo data In another terminal (do not stop the Lightning server) run the benchmarking/script.js file using the following command k6 run benchmarking/script.js If the script exits succesfully, this means the app met the defined performance thresholds. To collect the benchmarking data in a CSV file, run the previous command with the --out filename option. k6 run --out csv=test_results.csv benchmarking/script.js See results output for other available output formats.","ref":"benchmarking.html"},{"type":"extras","title":"Provisioning","doc":"Lightning offers the ability to configure projects via the HTTP API. By providing a JSON document with the desired configuration, the project can be configured to your liking.","ref":"provisioning.html"},{"type":"extras","title":"Provisioning - Using the API","doc":"The API is available at /api/provision , and expects an application/json Content-Type. Authentication The API requires a valid auth token to be provided in the Authorization header. Example Request curl -X POST \\ -d @project.json \\ -H "Authorization: Bearer $TOKEN" \\ -H "Content-Type: application/json" \\ $ENDPOINT/api/provision","ref":"provisioning.html#using-the-api"},{"type":"extras","title":"Provisioning - Document Structure","doc":"The provisioning document is a JSON document with the project at the root. All entities must have an id field, which is a UUIDv4 string. In the case of new entities, this must be generated by the client. The API is idempotent, and the distinction between creating and updating is determined by the presence of the id field. { "id": "<<project-id>>", "name": "<<project-name>>", "workflows": [ { "id": "<<workflow-id>>", "name": "<<workflow-name>>", "jobs": [ { "id": "<<job-id>>", "name": "<<job-name>>", "body": "<<job-body>>", "adaptor": "<<adaptor-name>>", "enabled": true } // ... more jobs ], "triggers": [ { "id": "<<trigger-id>>", "name": "<<trigger-name>>", "type": "webhook" } // ... more triggers ], "edges": [ { "id": "<<edge-id>>", "source_trigger_id": "<<trigger-id>>", "target_job_id": "<<job-id>>" } // ... more edges ] } // ... more workflows ] }","ref":"provisioning.html#document-structure"},{"type":"extras","title":"Provisioning - API Behaviour","doc":"The API expects all existing entities to be provided in the provisioning document. If the document provided is out of date (e.g. a new job was added on the server), a new reference document should be fetched and the changes applied to it. Deleting Entities Entities can be deleted by setting the disabled key to true . Example: { "id": "<<project-id>>", "workflows": [ { "id": "<<workflow-id>>", "jobs": [ { "id": "<<job-id>>", "delete": true // <== delete this job } ] } ] }","ref":"provisioning.html#api-behaviour"},{"type":"extras","title":"Provisioning - Relationship with Projects as Code","doc":"The Projects as Code spec is a superset of the provisioning API. Projects as Code allows for the user to specify a key for each entity, which makes it easier to manage the project in the future. For example: name: my-project workflows: workflow-one: jobs: job-one: body: | console.log("Hello World"); adaptor: '@openfn/language-common' enabled: true triggers: trigger-one: type: webhook edges: - source_trigger: trigger-one target_job: job-one The above YAML document illustrates the use of keys being used to identify entities. Allowing the user to provision the same project to multiple environments. The API is unaware of 'keys', and expects IDs to be provided by the client. In order to convert the above YAML document to a provisioning document, the CLI uses a local state file (if available) to map the keys to UUIDs. Using the example above a state file might look like this: { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": {}, "triggers": {} } } } The state file shows that the project and workflow already exist, but the job, trigger and edge do not. In order to create these new entities, IDs will be applied them. On a successful application of the provisioning document, the state file will be updated to reflect the new IDs and entities. { "id": "f6ba9a8c-b687-473a-908e-e250686f1eed", "workflows": { "workflow-one": { "id": "f206aa85-4fce-492e-94eb-ffd32c75d178", "jobs": { "job-one": { "id": "18ed71de-caf8-4822-aefc-5b19351f4016" } }, "triggers": { "trigger-one": { "id": "e0b9f357-9cf9-4206-9924-4d5674aad830" } }, "edges": [ { "id": "c239d994-6662-4637-90f8-0293c924b461", "source_trigger_id": "e0b9f357-9cf9-4206-9924-4d5674aad830", "target_job_id": "18ed71de-caf8-4822-aefc-5b19351f4016" } ] } } }","ref":"provisioning.html#relationship-with-projects-as-code"},{"type":"extras","title":"Changelog","doc":"All notable changes to this project will be documented in this file. The format is based on Keep a Changelog , and this project adheres to Semantic Versioning .","ref":"changelog.html"},{"type":"extras","title":"Changelog - [Unreleased]","doc":"Added Changed Moved to Elixir 1.15 and Erlang 26.0.2 to sort our an annoying ElixirLS issue that was slowing down our engineers. Update Debian base to use bookworm (Debian 12) for our Docker images Change new credential modal to take up less space on the screen #931 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","ref":"changelog.html#unreleased"},{"type":"extras","title":"Changelog - [v0.7.0-pre5] - 2023-07-28","doc":"Added Changed Unless otherwise specified, only show workorders with activity in last 14 days #968 Fixed","ref":"changelog.html#v0-7-0-pre5-2023-07-28"},{"type":"extras","title":"Changelog - [v0.7.0-pre4] - 2023-07-27","doc":"Added Changed Don't add cast fragments if the search_term is nil #968 Fixed","ref":"changelog.html#v0-7-0-pre4-2023-07-27"},{"type":"extras","title":"Changelog - [v0.7.0-pre3] - 2023-07-26","doc":"Added Changed Fixed Fixed an issue with newly created edges that prevented downstream jobs 977","ref":"changelog.html#v0-7-0-pre3-2023-07-26"},{"type":"extras","title":"Changelog - [v0.7.0-pre2] - 2023-07-26","doc":"Note that this is a pre-release with a couple of known bugs that are tracked in the Nodes and Edges epic . Added Added ability for a user to enable MFA on their account; using 2FA apps like Authy, Google Authenticator etc #890 Write/run sql script to convert triggers #875 Export projects as .yaml via UI #249 Changed In v0.7.0 we change the underlying workflow building and execution infrastructure to align with a standard "nodes and edges" design for directed acyclic graphs (DAGs). Make sure to run the migrations! 793 Fixed Propagate url pushState/changes to Workflow Diagram selection #944 Fix issue when deleting nodes from the workflow editor #830 Fix issue when clicking a trigger on a new/unsaved workflow #954","ref":"changelog.html#v0-7-0-pre2-2023-07-26"},{"type":"extras","title":"Changelog - [0.6.7] - 2023-07-13","doc":"Added Add feature to bulk rerun work orders from a specific step in their workflow; e.g., "rerun these 50 work orders, starting each at step 4." #906 Changed Fixed Oban exception: "value too long" when log lines are longer than 255 chars #929","ref":"changelog.html#0-6-7-2023-07-13"},{"type":"extras","title":"Changelog - [0.6.6] - 2023-06-30","doc":"Added Add public API token to the demo site setup script Check and renew OAuth credentials when running a job #646 Fixed Remove google sheets from adaptors list until supporting oauth flow #792 Remove duplicate google sheets adaptor display on credential type picklist #663 Fix demo setup script for calling from outside the app on Kubernetes deployments #917","ref":"changelog.html#0-6-6-2023-06-30"},{"type":"extras","title":"Changelog - [0.6.5] - 2023-06-22","doc":"Added Ability to rerun workorders from start by selecting one of more of them from the History page and clicking the "Rerun" button. #659 Fixed Example runs for demo incorrect #856","ref":"changelog.html#0-6-5-2023-06-22"},{"type":"extras","title":"Changelog - [0.6.3] - 2023-06-15","doc":"Fixed Prevent saving null log lines to the database, fix issue with run display #866","ref":"changelog.html#0-6-3-2023-06-15"},{"type":"extras","title":"Changelog - [0.6.2] - 2023-06-09","doc":"Fixed Fixed viewer permissions for delete workflow Fixed bug with workflow cards #859","ref":"changelog.html#0-6-2-2023-06-09"},{"type":"extras","title":"Changelog - [0.6.1] - 2023-06-08","doc":"Fixed Fixed bug with run logs #864 Correctly stagger demo runs to maintain order #856 Remove Timex use from SetupUtils in favor of DateTime to fix issue when calling it in escript.","ref":"changelog.html#0-6-1-2023-06-08"},{"type":"extras","title":"Changelog - [0.6.0]- 2023-04-12","doc":"Added Create sample runs when generating sample workflow #821 Added a provisioning api for creating and updating projects and their workflows See: PROVISIONING.md #641 Add ability for a superuser to schedule deletion, cancel deletion, and delete projects #757 Add ability for a project owner to schedule deletion, cancel deletion, and delete projects #746 Changed Ability to store run log lines as rows in a separate table #514 Fixed Incorrect project digest queries #768 ] Fix issue when purging deleted users #747 Generate a random name for Workflows when creating one via the UI. #828 Handle error when deleting a job with runs. #814","ref":"changelog.html#0-6-0-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.2]","doc":"Added Add workflow_edges table in preparation for new workflow editor implementation #794 Stamped credential_id on run directly for easier auditing of the history interface. Admins can now see which credential was used to run a run. #800 Better errors when using magic functions: "no magic yet" and "check credential" #812 Changed The delete-project function now delete all associated activities #759 Fixed","ref":"changelog.html#0-5-2"},{"type":"extras","title":"Changelog - [0.5.1] - 2023-04-12","doc":"Added Added ability to create and revoke personal API tokens #147 Add last-used at to API tokens #722 Improved "save" for job builder; users can now press Ctrl + S or ⌘ + S to save new or updated jobs job panel will not close. (Click elsewhere in the canvas or click the "Close" button to close.) #568 Add filtered search params to the history page URL #660 Changed The secret scrubber now ignores booleans 690 Fixed The secret scrubber now properly handles integer secrets from credentials 690 Updated describe-package dependency, fixing sparkles in adaptor-docs 657 Clicks on the workflow canvas were not lining up with the nodes users clicked on; they are now 733 Job panel behaves better when collapsed 774","ref":"changelog.html#0-5-1-2023-04-12"},{"type":"extras","title":"Changelog - [0.5.0] - 2023-04-03","doc":"Added Magic functions that fetch real metadata from connected systems via credentials and suggest completions in the job builder (e.g., pressing control-space when setting the orgUnit attribute for a DHIS2 create operation will pull the actual list of orgUnits with human readable labels and fill in their orgUnit codes upon enter.) 670 A "metadata explorer" to browse actual system metadata for connected instances. 658 Resizable job builder panel for the main canvas/workflow view. 681 Changed Display timezone for cron schedule—it is always UTC. #716 Instance administrators can now configure the interval between when a project owner or user requests deletion and when these records are purged from the database. It defaults to 7, but by providing a PURGE_DELETED_AFTER_DAYS environment variable the grace period can be altered. Note that setting this variable to 0 will make automatic purging never occur but will still make "deleted" projects and users unavailable. This has been requested by certain organizations that must retain audit logs in a Lightning instance. 758 Fixed Locked CLI version to @openfn/cli@0.0.35 . #761","ref":"changelog.html#0-5-0-2023-04-03"},{"type":"extras","title":"Changelog - [0.4.8] - 2023-03-29","doc":"Added Added a test harness for monitoring critical parts of the app using Telemetry #654 Changed Set log level to info for runs. Most of the debug logging is useful for the CLI, but not for Lightning. In the future the log level will be configurable at instance > project > job level by the superuser and any project admin . Renamed license file so that automagic github icon is less confusing Fixed Broken links in failure alert email #732 Registration Submission on app.openfn.org shows internal server error in browser #686 Run the correct runtime install mix task in Dockerfile-dev #541 Users not disabled when scheduled for deletion #719","ref":"changelog.html#0-4-8-2023-03-29"},{"type":"extras","title":"Changelog - [0.4.6] - 2023-03-23","doc":"Added Implement roles and permissions across entire app #645 Fix webhook URL ( https://<<HOST_URL>>/i/cae544ab-03dc-4ccc-a09c-fb4edb255d7a ) for the OpenHIE demo workflow 448 Phoenix Storybook for improved component development Load test for webhook endpoint performance #645 Notify user via email when they're added to a project #306 Added notify user via email when their account is created #307 Changed Improved errors when decoding encryption keys for use with Cloak. #684 Allow users to run ANY job with a custom input. #629 Fixed Ensure JSON schema form inputs are in the same order as they are written in the schema #685","ref":"changelog.html#0-4-6-2023-03-23"},{"type":"extras","title":"Changelog - [0.4.4] - 2023-03-10","doc":"Added Users can receive a digest email reporting on a specified project. #638 #585 Changed Fixed","ref":"changelog.html#0-4-4-2023-03-10"},{"type":"extras","title":"Changelog - [0.4.3] - 2023-03-06","doc":"Added Tooltips on Job Builder panel #650 Changed Upgraded to Phoenix 1.7 (3945856) Fixed Issue with FailureAlerter configuration missing in prod mode.","ref":"changelog.html#0-4-3-2023-03-06"},{"type":"extras","title":"Changelog - [0.4.2] - 2023-02-24","doc":"Added A user can change their own email #247 Added a SCHEMAS_PATH environment variable to override the default folder location for credential schemas #604 Added the ability to configure Google Sheets credentials #536 Function to import a project #574 Changed Users cannot register if they have not selected the terms and conditions #531 Fixed Jobs panel slow for first open after restart #567","ref":"changelog.html#0-4-2-2023-02-24"},{"type":"extras","title":"Changelog - [0.4.0] - 2023-02-08","doc":"Added Added a Delete job button in Inspector Filter workflow runs by text/value in run logs or input body Drop "configuration" key from Run output dataclips after completion Ability to 'rerun' a run from the Run list Attempts and Runs update themselves in the Runs list Configure a project and workflow for a new registering user Run a job with a custom input Added plausible analytics Allow user to click on Webhook Trigger Node to copy webhook URL on workflow diagram Allow any user to delete a credential that they own Create any credential through a form except for OAuth Refit all diagram nodes on browser and container resize Enable distributed Erlang, allowing any number of redundant Lightning nodes to communicate with each other. Users can set up realtime alerts for a project Changed Better code-assist and intelliense in the Job Editor Updated @openfn/workflow-diagram to 0.4.0 Make plus button part of job nodes in Workflow Diagram Updated @openfn/adaptor-docs to 0.0.5 Updated @openfn/describe-package to 0.0.10 Create an follow a manual Run from the Job Inspector View all workflows in a project on the workflows index page Move @openfn/workflow-diagram into the application, the NPM module is now deprecated. Remove workflow name from first node Move the used parts of @openfn/engine into the application. [BREAKING CHANGE] Ported mix openfn.install.runtime into application, use mix lightning.install_runtime . [BREAKING CHANGE] Introduced @openfn/cli as the new runtime for Jobs Rename a workflow through the page heading Hide the dataclips tab for beta Make adaptor default to common@latest Remove jobs list page Better error handling in the docs panel Disable credential ownership transfer in dev and prod environments Add project settings page Change Workorder filters to apply to the aggregate state of the workorder and not the run directly Enable jobs by default Set log level to info Add Beta checkbox to register page User roles and permissions Fixed Don't consider disabled jobs when calculating subsequent runs Fixed overflow on Job Editor Tooltips Fixed auto-scroll when adding a new snippet in the Job Editor Fixed common operation typings in Job Editor","ref":"changelog.html#0-4-0-2023-02-08"},{"type":"extras","title":"Changelog - [0.3.1] - 2022-11-22","doc":"Fixed Fixed bug that attempted to execute HTML scripts in dataclips Fixed bug that prevented workorders from displaying in the order of their last run, descending. Remove alerts after set timeout or close","ref":"changelog.html#0-3-1-2022-11-22"},{"type":"extras","title":"Changelog - [0.3.0] - 2022-11-21","doc":"Added Add seed data for demo site Create adaptor credentials through a form Configure cron expressions through a form View runs grouped by workorders and attempts Run an existing Job with any dataclip uuid from the Job form Changed Redirect users to projects list page when they click on Admin Settings menu Move job, project, input and output Dataclips to Run table Reverse the relationship between Jobs and Triggers. Triggers now can exist on their own; setting the stage for branching and merging workflows Updated Elixir and frontend dependencies [BREAKING CHANGE] Pipeline now uses WorkOrders, previous data is not compatible. Runs, Dataclips and Attempts now all correctly use usec resolution timestamps. Upgraded LiveView to 0.18.0 Upgraded Elixir to 1.14.1 and OTP 25 Workflow Job editor now behaves like a panel Split JobLive.InspectorFormComponent into different plug-able subcomponents Ensure new jobs with cron triggers receive a default frequency Webhooks are now referenced by the trigger id instead of job id. Filter runs by status Filter runs by workflow Filter runs by date View a job run from the runs history View latest matching inputs to run a job with","ref":"changelog.html#0-3-0-2022-11-21"},{"type":"extras","title":"Changelog - [0.2.0] - 2022-09-12","doc":"Changed [BREAKING CHANGE] Add Workflow model, Jobs now belong to a Workflow This is a breaking change to the schema. Use Node.js 18, soon to be in LTS. Visualize success/fail triggers in workflow diagram. Move WorkflowDiagram related actions from DashboardLive into WorkflowLive Move WorkflowDiagram component into liveview, so that we can subscribe to channels (i.e. updating of the diagram when someone changes something). Integrate @openfn/workflow-diagram@0.0.8 and use the new Store interface for updating it. Remove component_mounted event from WorkflowDiagram hook, using a MutationObserver and a Base64 encoded JSON payload. Fixed an issue where the compiler component would try and load a 'nothing adaptor', added a condition to check an adaptor is actually selected. Removed previous Workflow CTE queries, replaced by the introduction of the Workflow model, see ( https://github.com/OpenFn/Lightning/blob/53da6883483e7d8d078783f348da327d1dd72d20/lib/lightning/workflows.ex#L111-L119 ).","ref":"changelog.html#0-2-0-2022-09-12"},{"type":"extras","title":"Changelog - [0.1.13] - 2022-08-29","doc":"Added Allow administrators to configure OIDC providers for authentication (note that this is just for authenticating, not yet for creating new accounts via OIDC) Add Monaco editor to the step/job panel Allow users to delete their own accounts. Schedule their user and credentials data for deletion when they do. Allow superusers to delete a user account. Schedule the user's credentials and user data for deletion when they do. If a user is scheduled for deletion, disable their account and prevent them from logging in. The 'User profile' and 'Credentials' page now have a sidebar menu Changed Project users now have one of the following roles: viewer, editor, admin, owner Users only have the following roles: user, superuser","ref":"changelog.html#0-1-13-2022-08-29"},{"type":"extras","title":"Changelog - [0.1.12] - 2022-08-15","doc":"Added Transfer credential ownership to another user. Create credentials via a form interface* Show "projects with access" in credentials list view. Show job in runs list and run view. Added roles and permissions to workflows and history page #645 *The form is defined by a JSON schema provided by an adaptor, in most cases: e.g., language-dhis2 provides a single schema which defines the required attributes for state.configuration , while language-common provides multiple credential schemas like "oauth" or "basic auth" which define attributes for state.configuration and which might be used by lots of different jobs.) Fixed User menu (top right) appears on top of all other components. User profile screen integrated with the rest of the liveview app.","ref":"changelog.html#0-1-12-2022-08-15"},{"type":"extras","title":"Changelog - [0.1.11] - 2022-08-05","doc":"Fixed Fixed logging in Runner when :debug log level used; note that this caused crashes in Oban","ref":"changelog.html#0-1-11-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.10] - 2022-08-05","doc":"Added Credential auditing Build/version information display for easier debugging Fixed Fixed a bug that enqueued cron-triggered jobs even when they were disabled","ref":"changelog.html#0-1-10-2022-08-05"},{"type":"extras","title":"Changelog - [0.1.9] - 2022-07-27","doc":"Added Navigate to user profile or credentials page and log out through the user icon dropdown Create and edit dataclips Add a production tag to credentials View a dropdown of operations and their description for the language-common v2.0.0-rc2 adaptor (this pattern to be rolled out across adaptors) Changed Navigate between projects through a project picker on the navbar Fixed Run Lightning with docker Security Sensitive credential values are scrubbed from run logs All credentials are encrypted at REST","ref":"changelog.html#0-1-9-2022-07-27"},{"type":"extras","title":"Changelog - [0.1.7] - 2022-06-24","doc":"Added Run a job with a cron trigger Queue jobs via Oban/Postgres Edit jobs via the workflow canvas","ref":"changelog.html#0-1-7-2022-06-24"},{"type":"extras","title":"Changelog - [0.1.6] - 2022-06-07","doc":"Added Register, log in and log out of an account Allow superusers and admin users to create projects Allow admin users to create or disable a user’s account Allow superusers for local deployments to create users and give them access to project spaces Create and edit a job with a webhook, flow/fail or cron trigger Create and edit credentials for a job Copy a job's webhook URL View all workflows in a project visually Deploy lightning locally with Docker Enable a job to automatically process incoming requests Run a job with a webhook or flow/fail trigger View job runs along with their logs, exit code, start and end time View data clips that have initiated job runs (http requests for webhooks, run results) Changed - Removed -","ref":"changelog.html#0-1-6-2022-06-07"}] \ No newline at end of file diff --git a/readme.html b/readme.html index 092e3f8bdd..4181ecc2ac 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 9bbac7f1d3..87f6497cf3 100644 --- a/search.html +++ b/search.html @@ -107,7 +107,7 @@

    - +