From 67fe3bd716eebbcbc91c1d4edcbe0f39f6371632 Mon Sep 17 00:00:00 2001 From: zoldar Date: Tue, 3 Sep 2024 12:50:44 +0000 Subject: [PATCH] deploy: 530d290678279f6c4e0710c60c59e58ff213ae0b --- .build | 2 +- Plausible.AsyncInsertRepo.html | 16 +- Plausible.Cldr.AcceptLanguage.html | 248 ++++++++++----------- Plausible.Cldr.Currency.html | 160 ++++++------- Plausible.Cldr.Locale.html | 48 ++-- Plausible.Cldr.Number.Cardinal.html | 18 +- Plausible.Cldr.Number.Format.html | 48 ++-- Plausible.Cldr.Number.Ordinal.html | 18 +- Plausible.Cldr.Number.Symbol.html | 14 +- Plausible.Cldr.Number.System.html | 54 ++--- Plausible.Cldr.Number.Transliterate.html | 20 +- Plausible.Cldr.Number.html | 176 +++++++-------- Plausible.Cldr.Rbnf.NumberSystem.html | 12 +- Plausible.Cldr.Rbnf.Ordinal.html | 16 +- Plausible.Cldr.Rbnf.Spellout.html | 12 +- Plausible.Cldr.html | 182 +++++++-------- Plausible.ClickhouseRepo.html | 16 +- Plausible.DataMigration.Repo.html | 16 +- Plausible.Exports.html | 18 +- Plausible.Geo.html | 64 +++--- Plausible.ImportDeletionRepo.html | 16 +- Plausible.Imported.CSVImporter.html | 36 +-- Plausible.Imported.Importer.html | 28 +-- Plausible.IngestRepo.html | 16 +- Plausible.S3.html | 14 +- Plausible.Stats.Filters.html | 12 +- Plausible.Stats.SQL.Fragments.html | 4 +- Plausible.epub | Bin 763428 -> 763437 bytes PlausibleWeb.Api.StatsController.html | 16 +- PlausibleWeb.Live.Components.Modal.html | 42 ++-- PlausibleWeb.Plugs.AuthorizePublicAPI.html | 6 +- dist/search_data-25912140.js | 1 + dist/search_data-E784FFA1.js | 1 - search.html | 2 +- 34 files changed, 676 insertions(+), 676 deletions(-) create mode 100644 dist/search_data-25912140.js delete mode 100644 dist/search_data-E784FFA1.js diff --git a/.build b/.build index 084142171c..f99981ee97 100644 --- a/.build +++ b/.build @@ -437,7 +437,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_data-E784FFA1.js +dist/search_data-25912140.js dist/sidebar_items-5F709355.js index.html readme.html diff --git a/Plausible.AsyncInsertRepo.html b/Plausible.AsyncInsertRepo.html index 8f1e9e721c..9e7a57e939 100644 --- a/Plausible.AsyncInsertRepo.html +++ b/Plausible.AsyncInsertRepo.html @@ -1187,23 +1187,23 @@ pool to disconnect within the given interval.

See -

Similar to insert_all/2 but with the following differences:

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
+

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
 
-defmodule Demo do
+defmodule Demo do
   use Ecto.Schema
 
   @primary_key false
-  schema "ecto_ch_demo" do
+  schema "ecto_ch_demo" do
     field :a, Ch, type: "UInt64"
     field :b, :string
-  end
-end
+  end
+end
 
-rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
-{100_000, nil} = Repo.insert_stream(Demo, rows)
+rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
+{100_000, nil} = Repo.insert_stream(Demo, rows)
 
 # schemaless
-{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
+
{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
diff --git a/Plausible.Cldr.AcceptLanguage.html b/Plausible.Cldr.AcceptLanguage.html index 3e1f1e6b17..848ac0a491 100644 --- a/Plausible.Cldr.AcceptLanguage.html +++ b/Plausible.Cldr.AcceptLanguage.html @@ -148,7 +148,7 @@ Pages the set of natural languages that are preferred as a response to the request. Language tags function are provided in Cldr.LanguageTag.

The format of an Accept-Language header is as follows in ABNF format:

   Accept-Language = "Accept-Language" ":"
                      1#( language-range [ ";" "q" "=" qvalue ] )
-   language-range  = ( ( 1*8ALPHA *( "-" 1*8ALPHA ) ) | "*" )

Each language-range MAY be given an associated quality value which represents an + language-range = ( ( 1*8ALPHA *( "-" 1*8ALPHA ) ) | "*" )

Each language-range MAY be given an associated quality value which represents an estimate of the user's preference for the languages specified by that range. The quality value defaults to "q=1". For example,

   Accept-Language: da, en-gb;q=0.8, en;q=0.7

would mean: "I prefer Danish, but will accept British English and other types of English."

@@ -261,54 +261,54 @@ a configured Examples -
iex> Plausible.Cldr.AcceptLanguage.best_match("da;q=0.1,zh-TW;q=0.3", TestBackend.Cldr)
-{:ok,
- %Cldr.LanguageTag{
+
iex> Plausible.Cldr.AcceptLanguage.best_match("da;q=0.1,zh-TW;q=0.3", TestBackend.Cldr)
+{:ok,
+ %Cldr.LanguageTag{
    backend: TestBackend.Cldr,
    canonical_locale_name: "zh-TW",
    cldr_locale_name: :"zh-Hant",
-   language_subtags: [],
-   extensions: %{},
+   language_subtags: [],
+   extensions: %{},
    gettext_locale_name: nil,
    language: "zh",
-   locale: %{},
-   private_use: [],
+   locale: %{},
+   private_use: [],
    rbnf_locale_name: :"zh-Hant",
    requested_locale_name: "zh-TW",
    script: :Hant,
    territory: :TW,
-   transform: %{},
-   language_variants: []
- }}
+   transform: %{},
+   language_variants: []
+ }}
 
-iex> Plausible.Cldr.AcceptLanguage.best_match("da;q=0.1,zh-TW;q=0.3", TestBackend.Cldr)
-{:ok,
- %Cldr.LanguageTag{
+iex> Plausible.Cldr.AcceptLanguage.best_match("da;q=0.1,zh-TW;q=0.3", TestBackend.Cldr)
+{:ok,
+ %Cldr.LanguageTag{
    backend: TestBackend.Cldr,
    canonical_locale_name: "zh-TW",
    cldr_locale_name: :"zh-Hant",
-   language_subtags: [],
-   extensions: %{},
+   language_subtags: [],
+   extensions: %{},
    gettext_locale_name: nil,
    language: "zh",
-   locale: %{},
-   private_use: [],
+   locale: %{},
+   private_use: [],
    rbnf_locale_name: :"zh-Hant",
    requested_locale_name: "zh-TW",
    script: :Hant,
    territory: :TW,
-   transform: %{},
-   language_variants: []
- }}
+   transform: %{},
+   language_variants: []
+ }}
 
-iex> Plausible.Cldr.AcceptLanguage.best_match("xx,yy;q=0.3")
-{:error,
- {Cldr.NoMatchingLocale,
-  "No configured locale could be matched to \"xx,yy;q=0.3\""}}
+iex> Plausible.Cldr.AcceptLanguage.best_match("xx,yy;q=0.3")
+{:error,
+ {Cldr.NoMatchingLocale,
+  "No configured locale could be matched to \"xx,yy;q=0.3\""}}
 
-iex> Plausible.Cldr.AcceptLanguage.best_match("invalid_tag")
-{:error, {Cldr.LanguageTag.ParseError,
-  "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}}
+
iex> Plausible.Cldr.AcceptLanguage.best_match("invalid_tag") +{:error, {Cldr.LanguageTag.ParseError, + "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}}
@@ -369,95 +369,95 @@ with an error tuple for each invalid tag added at the end of the list.

Example

-
iex> Cldr.AcceptLanguage.parse("da,zh-TW;q=0.3", TestBackend.Cldr)
-{:ok,
- [
-   {1.0,
-    %Cldr.LanguageTag{
+
iex> Cldr.AcceptLanguage.parse("da,zh-TW;q=0.3", TestBackend.Cldr)
+{:ok,
+ [
+   {1.0,
+    %Cldr.LanguageTag{
       backend: TestBackend.Cldr,
       canonical_locale_name: "da",
       cldr_locale_name: :da,
-      language_subtags: [],
-      extensions: %{},
+      language_subtags: [],
+      extensions: %{},
       gettext_locale_name: nil,
       language: "da",
-      locale: %{},
-      private_use: [],
+      locale: %{},
+      private_use: [],
       rbnf_locale_name: :da,
       requested_locale_name: "da",
       script: :Latn,
       territory: :DK,
-      transform: %{},
-      language_variants: []
-    }},
-   {0.3,
-    %Cldr.LanguageTag{
+      transform: %{},
+      language_variants: []
+    }},
+   {0.3,
+    %Cldr.LanguageTag{
       backend: TestBackend.Cldr,
       canonical_locale_name: "zh-TW",
       cldr_locale_name: :"zh-Hant",
-      language_subtags: [],
-      extensions: %{},
+      language_subtags: [],
+      extensions: %{},
       gettext_locale_name: nil,
       language: "zh",
-      locale: %{},
-      private_use: [],
+      locale: %{},
+      private_use: [],
       rbnf_locale_name: :"zh-Hant",
       requested_locale_name: "zh-TW",
       script: :Hant,
       territory: :TW,
-      transform: %{},
-      language_variants: []
-    }}
- ]}
+      transform: %{},
+      language_variants: []
+    }}
+ ]}
 
-iex> Plausible.Cldr.AcceptLanguage.parse("invalid_tag")
-{:error,
- {Cldr.LanguageTag.ParseError,
-  "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}}
+iex> Plausible.Cldr.AcceptLanguage.parse("invalid_tag")
+{:error,
+ {Cldr.LanguageTag.ParseError,
+  "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}}
 
-iex> Plausible.Cldr.AcceptLanguage.parse("da,zh-TW;q=0.3,invalid_tag")
-{:ok,
- [
-   {1.0,
-    %Cldr.LanguageTag{
+iex> Plausible.Cldr.AcceptLanguage.parse("da,zh-TW;q=0.3,invalid_tag")
+{:ok,
+ [
+   {1.0,
+    %Cldr.LanguageTag{
       backend: TestBackend.Cldr,
       canonical_locale_name: "da",
       cldr_locale_name: :da,
-      language_subtags: [],
-      extensions: %{},
+      language_subtags: [],
+      extensions: %{},
       gettext_locale_name: nil,
       language: "da",
-      locale: %{},
-      private_use: [],
+      locale: %{},
+      private_use: [],
       rbnf_locale_name: :da,
       requested_locale_name: "da",
       script: :Latn,
       territory: :DK,
-      transform: %{},
-      language_variants: []
-    }},
-   {0.3,
-    %Cldr.LanguageTag{
+      transform: %{},
+      language_variants: []
+    }},
+   {0.3,
+    %Cldr.LanguageTag{
       backend: TestBackend.Cldr,
       canonical_locale_name: "zh-TW",
       cldr_locale_name: :"zh-Hant",
-      language_subtags: [],
-      extensions: %{},
+      language_subtags: [],
+      extensions: %{},
       gettext_locale_name: nil,
       language: "zh",
-      locale: %{},
-      private_use: [],
+      locale: %{},
+      private_use: [],
       rbnf_locale_name: :"zh-Hant",
       requested_locale_name: "zh-TW",
       script: :Hant,
       territory: :TW,
-      transform: %{},
-      language_variants: []
-    }},
-   {:error,
-    {Cldr.LanguageTag.ParseError,
-     "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}}
- ]}
+
transform: %{}, + language_variants: [] + }}, + {:error, + {Cldr.LanguageTag.ParseError, + "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}} + ]}
@@ -502,92 +502,92 @@ with an error tuple for each invalid tag added at the end of the list.

Example

-
iex> Plausible.Cldr.AcceptLanguage.parse!("da,zh-TW;q=0.3")
-[
-  {1.0,
-   %Cldr.LanguageTag{
+
iex> Plausible.Cldr.AcceptLanguage.parse!("da,zh-TW;q=0.3")
+[
+  {1.0,
+   %Cldr.LanguageTag{
      backend: TestBackend.Cldr,
      canonical_locale_name: "da",
      cldr_locale_name: :da,
-     language_subtags: [],
-     extensions: %{},
+     language_subtags: [],
+     extensions: %{},
      gettext_locale_name: nil,
      language: "da",
-     locale: %{},
-     private_use: [],
+     locale: %{},
+     private_use: [],
      rbnf_locale_name: :da,
      requested_locale_name: "da",
      script: :Latn,
      territory: :DK,
-     transform: %{},
-     language_variants: []
-   }},
-  {0.3,
-   %Cldr.LanguageTag{
+     transform: %{},
+     language_variants: []
+   }},
+  {0.3,
+   %Cldr.LanguageTag{
      backend: TestBackend.Cldr,
      canonical_locale_name: "zh-TW",
      cldr_locale_name: :"zh-Hant",
-     language_subtags: [],
-     extensions: %{},
+     language_subtags: [],
+     extensions: %{},
      gettext_locale_name: nil,
      language: "zh",
-     locale: %{},
-     private_use: [],
+     locale: %{},
+     private_use: [],
      rbnf_locale_name: :"zh-Hant",
      requested_locale_name: "zh-TW",
      script: :Hant,
      territory: :TW,
-     transform: %{},
-     language_variants: []
-   }}
-]
+     transform: %{},
+     language_variants: []
+   }}
+]
 
 Plausible.Cldr.AcceptLanguage.parse! "invalid_tag"
 ** (Cldr.AcceptLanguageError) "Expected a BCP47 language tag. Could not parse the remaining "g" starting at position 11
     (ex_cldr) lib/cldr/accept_language.ex:304: Cldr.AcceptLanguage.parse!/1
 
-iex> Plausible.Cldr.AcceptLanguage.parse!("da,zh-TW;q=0.3,invalid_tag")
-[
-  {1.0,
-   %Cldr.LanguageTag{
+iex> Plausible.Cldr.AcceptLanguage.parse!("da,zh-TW;q=0.3,invalid_tag")
+[
+  {1.0,
+   %Cldr.LanguageTag{
      backend: TestBackend.Cldr,
      canonical_locale_name: "da",
      cldr_locale_name: :da,
-     language_subtags: [],
-     extensions: %{},
+     language_subtags: [],
+     extensions: %{},
      gettext_locale_name: nil,
      language: "da",
-     locale: %{},
-     private_use: [],
+     locale: %{},
+     private_use: [],
      rbnf_locale_name: :da,
      requested_locale_name: "da",
      script: :Latn,
      territory: :DK,
-     transform: %{},
-     language_variants: []
-   }},
-  {0.3,
-   %Cldr.LanguageTag{
+     transform: %{},
+     language_variants: []
+   }},
+  {0.3,
+   %Cldr.LanguageTag{
      backend: TestBackend.Cldr,
      canonical_locale_name: "zh-TW",
      cldr_locale_name: :"zh-Hant",
-     language_subtags: [],
-     extensions: %{},
+     language_subtags: [],
+     extensions: %{},
      gettext_locale_name: nil,
      language: "zh",
-     locale: %{},
-     private_use: [],
+     locale: %{},
+     private_use: [],
      rbnf_locale_name: :"zh-Hant",
      requested_locale_name: "zh-TW",
      script: :Hant,
      territory: :TW,
-     transform: %{},
-     language_variants: []
-   }},
-  {:error,
-   {Cldr.LanguageTag.ParseError,
-    "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}}
-]
+
transform: %{}, + language_variants: [] + }}, + {:error, + {Cldr.LanguageTag.ParseError, + "Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11"}} +]
diff --git a/Plausible.Cldr.Currency.html b/Plausible.Cldr.Currency.html index b50c8ae762..5799f7e28b 100644 --- a/Plausible.Cldr.Currency.html +++ b/Plausible.Cldr.Currency.html @@ -419,11 +419,11 @@ The default is :all. See
FJD: %Cldr.Currency{ + %{

 FJD: %Cldr.Currency{
    cash_digits: 2,
    cash_rounding: 0,
    code: "FJD",
-   count: %{one: "Fijian dollar", other: "Fijian dollars"},
+   count: %{one: "Fijian dollar", other: "Fijian dollars"},
    digits: 2,
    from: nil,
    iso_digits: 2,
@@ -433,12 +433,12 @@ The default is :all. See symbol: "FJD",
    tender: true,
    to: nil
- },
- SUR: %Cldr.Currency{
+ },
+ SUR: %Cldr.Currency{
    cash_digits: 2,
    cash_rounding: 0,
    code: "SUR",
-   count: %{one: "Soviet rouble", other: "Soviet roubles"},
+   count: %{one: "Soviet rouble", other: "Soviet roubles"},
    digits: 2,
    from: nil,
    iso_digits: nil,
@@ -448,7 +448,7 @@ The default is :all. See symbol: "SUR",
    tender: true,
    to: nil
- },
+ },
  ...
 }}
@@ -499,11 +499,11 @@ The default is :all. See Example

MyApp.Cldr.Currency.currencies_for_locale!("en") - => %{

FJD: %Cldr.Currency{
+  => %{

FJD: %Cldr.Currency{
   cash_digits: 2,
   cash_rounding: 0,
   code: "FJD",
-  count: %{one: "Fijian dollar", other: "Fijian dollars"},
+  count: %{one: "Fijian dollar", other: "Fijian dollars"},
   digits: 2,
   from: nil,
   iso_digits: 2,
@@ -513,12 +513,12 @@ The default is :all. See symbol: "FJD",
   tender: true,
   to: nil
-},
-SUR: %Cldr.Currency{
+},
+SUR: %Cldr.Currency{
   cash_digits: 2,
   cash_rounding: 0,
   code: "SUR",
-  count: %{one: "Soviet rouble", other: "Soviet roubles"},
+  count: %{one: "Soviet rouble", other: "Soviet roubles"},
   digits: 2,
   from: nil,
   iso_digits: nil,
@@ -528,7 +528,7 @@ The default is :all. See symbol: "SUR",
   tender: true,
   to: nil
-},
+},
 ...

}

@@ -586,13 +586,13 @@ or a Examples -
iex> Plausible.Cldr.Currency.currency_for_code("AUD")
-{:ok,
-  %Cldr.Currency{
+
iex> Plausible.Cldr.Currency.currency_for_code("AUD")
+{:ok,
+  %Cldr.Currency{
     cash_digits: 2,
     cash_rounding: 0,
     code: "AUD",
-    count: %{one: "Australian dollar", other: "Australian dollars"},
+    count: %{one: "Australian dollar", other: "Australian dollars"},
     digits: 2,
     iso_digits: 2,
     name: "Australian Dollar",
@@ -600,15 +600,15 @@ or a rounding: 0,
     symbol: "A$",
     tender: true
-}}
+}}
 
-iex> Plausible.Cldr.Currency.currency_for_code("THB")
-{:ok,
-  %Cldr.Currency{
+iex> Plausible.Cldr.Currency.currency_for_code("THB")
+{:ok,
+  %Cldr.Currency{
     cash_digits: 2,
     cash_rounding: 0,
     code: "THB",
-    count: %{one: "Thai baht", other: "Thai baht"},
+    count: %{one: "Thai baht", other: "Thai baht"},
     digits: 2,
     iso_digits: 2,
     name: "Thai Baht",
@@ -616,7 +616,7 @@ or a rounding: 0,
     symbol: "THB",
     tender: true
-}}
+}}
@@ -675,12 +675,12 @@ or a Examples -
iex> Plausible.Cldr.Currency.currency_for_code!("AUD")
-%Cldr.Currency{
+
iex> Plausible.Cldr.Currency.currency_for_code!("AUD")
+%Cldr.Currency{
   cash_digits: 2,
   cash_rounding: 0,
   code: "AUD",
-  count: %{one: "Australian dollar", other: "Australian dollars"},
+  count: %{one: "Australian dollar", other: "Australian dollars"},
   digits: 2,
   iso_digits: 2,
   name: "Australian Dollar",
@@ -688,14 +688,14 @@ or a rounding: 0,
   symbol: "A$",
   tender: true
-}
+}
 
-iex> Plausible.Cldr.Currency.currency_for_code!("THB")
-%Cldr.Currency{
+iex> Plausible.Cldr.Currency.currency_for_code!("THB")
+%Cldr.Currency{
   cash_digits: 2,
   cash_rounding: 0,
   code: "THB",
-  count: %{one: "Thai baht", other: "Thai baht"},
+  count: %{one: "Thai baht", other: "Thai baht"},
   digits: 2,
   iso_digits: 2,
   name: "Thai Baht",
@@ -703,7 +703,7 @@ or a rounding: 0,
   symbol: "THB",
   tender: true
-}
+}
@@ -744,15 +744,15 @@ or a Examples -
iex> {:ok, locale} = Plausible.Cldr.validate_locale("en")
+
iex> {:ok, locale} = Plausible.Cldr.validate_locale("en")
 iex> Plausible.Cldr.Currency.currency_from_locale locale
 :USD
 
-iex> {:ok, locale} = Plausible.Cldr.validate_locale("en-AU")
+iex> {:ok, locale} = Plausible.Cldr.validate_locale("en-AU")
 iex> Plausible.Cldr.Currency.currency_from_locale locale
 :AUD
 
-iex> Plausible.Cldr.Currency.currency_from_locale("en-GB")
+iex> Plausible.Cldr.Currency.currency_from_locale("en-GB")
 :GBP
@@ -796,14 +796,14 @@ or a Example -
iex> MyApp.Cldr.Currency.currency_history_for_locale("en")
-{:ok,
-    %{
-    USD: %{from: ~D[1792-01-01], to: nil},
-    USN: %{tender: false},
-    USS: %{from: nil, tender: false, to: ~D[2014-03-01]}
-  }
-}
+
iex> MyApp.Cldr.Currency.currency_history_for_locale("en")
+{:ok,
+    %{
+    USD: %{from: ~D[1792-01-01], to: nil},
+    USN: %{tender: false},
+    USS: %{from: nil, tender: false, to: ~D[2014-03-01]}
+  }
+}
@@ -862,9 +862,9 @@ The default is :all. See Example -
MyApp.Cldr.Currency.currency_strings("en")
-=> {:ok,
- %{
+
MyApp.Cldr.Currency.currency_strings("en")
+=> {:ok,
+ %{
    "mexican silver pesos" => :MXP,
    "sudanese dinar" => :SDD,
    "bad" => :BAD,
@@ -874,7 +874,7 @@ The default is :all. See "guyanaese dollars" => :GYD,
    "equatorial guinean ekwele" => :GQE,
    ...
-  }}
+ }}
@@ -933,8 +933,8 @@ The default is :all. See Example -
MyApp.Cldr.Currency.currency_strings!("en")
-=> %{
+
MyApp.Cldr.Currency.currency_strings!("en")
+=> %{
   "mexican silver pesos" => :MXP,
   "sudanese dinar" => :SDD,
   "bad" => :BAD,
@@ -944,7 +944,7 @@ The default is :all. See "guyanaese dollars" => :GYD,
   "equatorial guinean ekwele" => :GQE,
   ...
- }
+ }
@@ -982,10 +982,10 @@ or a Example -
iex> MyApp.Cldr.Currency.current_currency_from_locale("en")
+
iex> MyApp.Cldr.Currency.current_currency_from_locale("en")
 :USD
 
-iex> MyApp.Cldr.Currency.current_currency_from_locale("en-AU")
+iex> MyApp.Cldr.Currency.current_currency_from_locale("en-AU")
 :AUD
@@ -1033,7 +1033,7 @@ mapping is returned for that territory.

Example

-
iex> Plausible.Cldr.Currency.current_territory_currencies()
+
iex> Plausible.Cldr.Currency.current_territory_currencies()
@@ -1125,11 +1125,11 @@ currency code

Examples

-
iex> Plausible.Cldr.Currency.known_currency_code("AUD")
-{:ok, :AUD}
+
iex> Plausible.Cldr.Currency.known_currency_code("AUD")
+{:ok, :AUD}
 
-iex> Plausible.Cldr.Currency.known_currency_code("GGG")
-{:error, {Cldr.UnknownCurrencyError, "The currency \"GGG\" is invalid"}}
+
iex> Plausible.Cldr.Currency.known_currency_code("GGG") +{:error, {Cldr.UnknownCurrencyError, "The currency \"GGG\" is invalid"}}
@@ -1176,13 +1176,13 @@ currency code

Examples

-
iex> Plausible.Cldr.Currency.known_currency_code?("AUD")
+
iex> Plausible.Cldr.Currency.known_currency_code?("AUD")
 true
 
-iex> Plausible.Cldr.Currency.known_currency_code?("GGG")
+iex> Plausible.Cldr.Currency.known_currency_code?("GGG")
 false
 
-iex> Plausible.Cldr.Currency.known_currency_code?(:XCV)
+iex> Plausible.Cldr.Currency.known_currency_code?(:XCV)
 false
@@ -1217,7 +1217,7 @@ currency code

Example

-
iex> Plausible.Cldr.Currency.known_currency_codes()
+
iex> Plausible.Cldr.Currency.known_currency_codes()
@@ -1276,14 +1276,14 @@ such as 0.05. Optional.

Example

-
iex> Plausible.Cldr.Currency.new(:XAE, name: "Custom Name", digits: 0)
-{:ok,
- %Cldr.Currency{
+
iex> Plausible.Cldr.Currency.new(:XAE, name: "Custom Name", digits: 0)
+{:ok,
+ %Cldr.Currency{
    alt_code: :XAE,
    cash_digits: 0,
    cash_rounding: nil,
    code: :XAE,
-   count: %{other: "Custom Name"},
+   count: %{other: "Custom Name"},
    digits: 0,
    from: nil,
    iso_digits: 0,
@@ -1293,11 +1293,11 @@ such as 0.05. Optional.

symbol: "XAE", tender: false, to: nil - }} -iex> MyApp.Cldr.Currency.new(:XAH, name: "Custom Name") -{:error, "Required options are missing. Required options are [:name, :digits]"} -iex> Plausible.Cldr.Currency.new(:XAE, name: "XAE", digits: 0) -{:error, {Cldr.CurrencyAlreadyDefined, "Currency :XAE is already defined."}}

+
}} +iex> MyApp.Cldr.Currency.new(:XAH, name: "Custom Name") +{:error, "Required options are missing. Required options are [:name, :digits]"} +iex> Plausible.Cldr.Currency.new(:XAE, name: "XAE", digits: 0) +{:error, {Cldr.CurrencyAlreadyDefined, "Currency :XAE is already defined."}}
@@ -1355,20 +1355,20 @@ default is Plaus Examples -
iex> Plausible.Cldr.Currency.pluralize(1, :USD)
-{:ok, "US dollar"}
+
iex> Plausible.Cldr.Currency.pluralize(1, :USD)
+{:ok, "US dollar"}
 
-iex> Plausible.Cldr.Currency.pluralize(3, :USD)
-{:ok, "US dollars"}
+iex> Plausible.Cldr.Currency.pluralize(3, :USD)
+{:ok, "US dollars"}
 
-iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: "zh")
-{:ok, "美元"}
+iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: "zh")
+{:ok, "美元"}
 
-iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: "fr")
-{:ok, "dollars des États-Unis"}
+iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: "fr")
+{:ok, "dollars des États-Unis"}
 
-iex> Plausible.Cldr.Currency.pluralize(1, :USD, locale: "fr")
-{:ok, "dollar des États-Unis"}
+
iex> Plausible.Cldr.Currency.pluralize(1, :USD, locale: "fr") +{:ok, "dollar des États-Unis"}
@@ -1410,8 +1410,8 @@ or a Example -
iex> MyApp.Cldr.Currency.strings_for_currency :AUD,("en")
-["a$", "australian dollars", "aud", "australian dollar"]
+
iex> MyApp.Cldr.Currency.strings_for_currency :AUD,("en")
+["a$", "australian dollars", "aud", "australian dollar"]
diff --git a/Plausible.Cldr.Locale.html b/Plausible.Cldr.Locale.html index b12a630b88..a3d49cbe7b 100644 --- a/Plausible.Cldr.Locale.html +++ b/Plausible.Cldr.Locale.html @@ -326,16 +326,16 @@ this specific locale..

Examples

-
iex> Plausible.Cldr.Locale.fallback_locale_names(:"fr-CA")
-{:ok, [:"fr-CA", :fr, :und]}
+
iex> Plausible.Cldr.Locale.fallback_locale_names(:"fr-CA")
+{:ok, [:"fr-CA", :fr, :und]}
 
 # Fallbacks are typically formed by progressively
 # stripping variant, territory and script from the
 # given locale name. But not always - there are
 # certain fallbacks that take a different path.
 
-iex> Plausible.Cldr.Locale.fallback_locale_names(:nb)
-{:ok, [:nb, :no, :und]}
+
iex> Plausible.Cldr.Locale.fallback_locale_names(:nb) +{:ok, [:nb, :no, :und]}
@@ -389,20 +389,20 @@ this specific locale.

Examples

-
Plausible.Cldr.Locale.fallback_locales(:"fr-CA")
-=> {:ok,
-     [#Cldr.LanguageTag<fr-CA [validated]>, #Cldr.LanguageTag<fr [validated]>,
-      #Cldr.LanguageTag<und [validated]>]}
+
Plausible.Cldr.Locale.fallback_locales(:"fr-CA")
+=> {:ok,
+     [#Cldr.LanguageTag<fr-CA [validated]>, #Cldr.LanguageTag<fr [validated]>,
+      #Cldr.LanguageTag<und [validated]>]}
 
 # Fallbacks are typically formed by progressively
 # stripping variant, territory and script from the
 # given locale name. But not always - there are
 # certain fallbacks that take a different path.
 
-Plausible.Cldr.Locale.fallback_locales(:nb))
-=> {:ok,
-     [#Cldr.LanguageTag<nb [validated]>, #Cldr.LanguageTag<no [validated]>,
-      #Cldr.LanguageTag<und [validated]>]}
+
Plausible.Cldr.Locale.fallback_locales(:nb)) +=> {:ok, + [#Cldr.LanguageTag<nb [validated]>, #Cldr.LanguageTag<no [validated]>, + #Cldr.LanguageTag<und [validated]>]}
@@ -533,15 +533,15 @@ generic top-level domain names.

Examples

iex> Plausible.Cldr.Locale.locale_from_host "a.b.com.au"
-Elixir.Plausible.Cldr.validate_locale(:"en-AU")
+Elixir.Plausible.Cldr.validate_locale(:"en-AU")
 
-iex> Plausible.Cldr.Locale.locale_from_host("a.b.com.tv")
-{:error,
- {Cldr.UnknownLocaleError, "No locale was identified for territory \"tv\""}}
+iex> Plausible.Cldr.Locale.locale_from_host("a.b.com.tv")
+{:error,
+ {Cldr.UnknownLocaleError, "No locale was identified for territory \"tv\""}}
 
-iex> Plausible.Cldr.Locale.locale_from_host("a.b.com")
-{:error,
- {Cldr.UnknownLocaleError, "No locale was identified for territory \"com\""}}
+iex> Plausible.Cldr.Locale.locale_from_host("a.b.com") +{:error, + {Cldr.UnknownLocaleError, "No locale was identified for territory \"com\""}}
@@ -683,12 +683,12 @@ be a territory.

Examples

-
iex> Cldr.Locale.territory_from_host("a.b.com.au")
-{:ok, :AU}
+
iex> Cldr.Locale.territory_from_host("a.b.com.au")
+{:ok, :AU}
 
-iex> Cldr.Locale.territory_from_host("a.b.com")
-{:error,
- {Cldr.UnknownLocaleError, "No locale was identified for territory \"com\""}}
+
iex> Cldr.Locale.territory_from_host("a.b.com") +{:error, + {Cldr.UnknownLocaleError, "No locale was identified for territory \"com\""}}
diff --git a/Plausible.Cldr.Number.Cardinal.html b/Plausible.Cldr.Number.Cardinal.html index 75cbcc8ef1..2d90bf7461 100644 --- a/Plausible.Cldr.Number.Cardinal.html +++ b/Plausible.Cldr.Number.Cardinal.html @@ -461,31 +461,31 @@ The valid substitution keys are :zero, Examples -
iex> Plausible.Cldr.Number.Cardinal.pluralize 1, "en", %{one: "one"}
+
iex> Plausible.Cldr.Number.Cardinal.pluralize 1, "en", %{one: "one"}
 "one"
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize 2, "en", %{one: "one"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize 2, "en", %{one: "one"}
 nil
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize 2, "en", %{one: "one", two: "two", other: "other"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize 2, "en", %{one: "one", two: "two", other: "other"}
 "other"
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize 22, "en", %{one: "one", two: "two", other: "other"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize 22, "en", %{one: "one", two: "two", other: "other"}
 "other"
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(1), "en", %{one: "one"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(1), "en", %{one: "one"}
 "one"
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), "en", %{one: "one"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), "en", %{one: "one"}
 nil
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), "en", %{one: "one", two: "two"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), "en", %{one: "one", two: "two"}
 nil
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, "ar", %{one: "one", few: "few", other: "other"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, "ar", %{one: "one", few: "few", other: "other"}
 "few"
 
-iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, "en", %{one: "one", few: "few", other: "other"}
+iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, "en", %{one: "one", few: "few", other: "other"}
 "other"
diff --git a/Plausible.Cldr.Number.Format.html b/Plausible.Cldr.Number.Format.html index 364925b60a..7964e52911 100644 --- a/Plausible.Cldr.Number.Format.html +++ b/Plausible.Cldr.Number.Format.html @@ -465,7 +465,7 @@ to precompile all the known formats at compile time.

Example

#=> Plausible.Cldr.Number.Format.Format.decimal_format_list
-["#", "#,##,##0%",
+["#", "#,##,##0%",
 "#,##,##0.###", "#,##,##0.00¤", "#,##,##0.00¤;(#,##,##0.00¤)",
 "#,##,##0 %", "#,##0%", "#,##0.###", "#,##0.00 ¤",
 "#,##0.00 ¤;(#,##0.00 ¤)", "#,##0.00¤", "#,##0.00¤;(#,##0.00¤)",
@@ -475,7 +475,7 @@ to precompile all the known formats at compile time.

"000 B ¤", "000 E ¤", "000 K ¤", "000 MRD ¤", "000 Md ¤", "000 Mio'.' ¤", "000 Mio ¤", "000 Mld ¤", "000 Mln ¤", "000 Mn ¤", "000 Mrd'.' ¤", "000 Mrd ¤", "000 Mr ¤", "000 M ¤", "000 NT ¤", "000 N ¤", "000 Tn ¤", -"000 Tr ¤", ...]

+"000 Tr ¤", ...]
@@ -523,8 +523,8 @@ the known formats at compile time. Its use is not otherwise recommended.

Example

-
iex> Plausible.Cldr.Number.Format.decimal_format_list_for(:en)
-{:ok, ["#,##0%", "#,##0.###", "#,##0.00", "#,##0.00;(#,##0.00)","#E0",
+
iex> Plausible.Cldr.Number.Format.decimal_format_list_for(:en)
+{:ok, ["#,##0%", "#,##0.###", "#,##0.00", "#,##0.00;(#,##0.00)","#E0",
  "0 billion", "0 million", "0 thousand",
  "0 trillion", "00 billion", "00 million", "00 thousand", "00 trillion",
  "000 billion", "000 million", "000 thousand", "000 trillion", "000B", "000K",
@@ -532,7 +532,7 @@ the known formats at compile time. Its use is not otherwise recommended.

"¤#,##0.00", "¤#,##0.00;(¤#,##0.00)", "¤000B", "¤000K", "¤000M", "¤000T", "¤00B", "¤00K", "¤00M", "¤00T", "¤0B", "¤0K", "¤0M", "¤0T", "¤ #,##0.00", "¤ #,##0.00;(¤ #,##0.00)", "¤ 000B", "¤ 000K", "¤ 000M", - "¤ 000T", "¤ 00B", "¤ 00K", "¤ 00M", "¤ 00T", "¤ 0B", "¤ 0K", "¤ 0M", "¤ 0T"]}

+
"¤ 000T", "¤ 00B", "¤ 00K", "¤ 00M", "¤ 00T", "¤ 0B", "¤ 0K", "¤ 0M", "¤ 0T"]}
@@ -585,8 +585,8 @@ is Plausible.Cld Examples -
iex> Plausible.Cldr.Number.Format.default_grouping_for(:en)
-{:ok, %{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}}
+
iex> Plausible.Cldr.Number.Format.default_grouping_for(:en)
+{:ok, %{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}}
@@ -642,8 +642,8 @@ is Plausible.Cld Examples -
iex> Plausible.Cldr.Number.Format.default_grouping_for!(:en)
-%{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}
+
iex> Plausible.Cldr.Number.Format.default_grouping_for!(:en)
+%{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}
@@ -700,18 +700,18 @@ by percent: "#,##0 %", scientific: "#E0", standard: "#,##0.###" - currency_short: [{"1000", [one: "0 k ¤", other: "0 k ¤"]}, - {"10000", [one: "00 k ¤", other: "00 k ¤"]}, - {"100000", [one: "000 k ¤", other: "000 k ¤"]}, - {"1000000", [one: "0 M ¤", other: "0 M ¤"]}, - {"10000000", [one: "00 M ¤", other: "00 M ¤"]}, - {"100000000", [one: "000 M ¤", other: "000 M ¤"]}, - {"1000000000", [one: "0 Md ¤", other: "0 Md ¤"]}, - {"10000000000", [one: "00 Md ¤", other: "00 Md ¤"]}, - {"100000000000", [one: "000 Md ¤", other: "000 Md ¤"]}, - {"1000000000000", [one: "0 Bn ¤", other: "0 Bn ¤"]}, - {"10000000000000", [one: "00 Bn ¤", other: "00 Bn ¤"]}, - {"100000000000000", [one: "000 Bn ¤", other: "000 Bn ¤"]}], + currency_short: [{"1000", [one: "0 k ¤", other: "0 k ¤"]}, + {"10000", [one: "00 k ¤", other: "00 k ¤"]}, + {"100000", [one: "000 k ¤", other: "000 k ¤"]}, + {"1000000", [one: "0 M ¤", other: "0 M ¤"]}, + {"10000000", [one: "00 M ¤", other: "00 M ¤"]}, + {"100000000", [one: "000 M ¤", other: "000 M ¤"]}, + {"1000000000", [one: "0 Md ¤", other: "0 Md ¤"]}, + {"10000000000", [one: "00 Md ¤", other: "00 Md ¤"]}, + {"100000000000", [one: "000 Md ¤", other: "000 Md ¤"]}, + {"1000000000000", [one: "0 Bn ¤", other: "0 Bn ¤"]}, + {"10000000000000", [one: "00 Bn ¤", other: "00 Bn ¤"]}, + {"100000000000000", [one: "000 Bn ¤", other: "000 Bn ¤"]}], ... }
@@ -802,8 +802,8 @@ is Plausible.Cld Examples -
iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for("en")
-{:ok, 1}
+
iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for("en")
+{:ok, 1}
@@ -855,7 +855,7 @@ is Plausible.Cld Examples -
iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for!("en")
+
iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for!("en")
 1
diff --git a/Plausible.Cldr.Number.Ordinal.html b/Plausible.Cldr.Number.Ordinal.html index 2fe38a632a..46934e6a52 100644 --- a/Plausible.Cldr.Number.Ordinal.html +++ b/Plausible.Cldr.Number.Ordinal.html @@ -464,31 +464,31 @@ The valid substitution keys are :zero, Examples -
iex> Plausible.Cldr.Number.Ordinal.pluralize 1, :en, %{one: "one"}
+
iex> Plausible.Cldr.Number.Ordinal.pluralize 1, :en, %{one: "one"}
 "one"
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: "one"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: "one"}
 nil
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: "one", two: "two"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: "one", two: "two"}
 "two"
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize 22, :en, %{one: "one", two: "two", other: "other"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize 22, :en, %{one: "one", two: "two", other: "other"}
 "two"
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(1), :en, %{one: "one"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(1), :en, %{one: "one"}
 "one"
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: "one"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: "one"}
 nil
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: "one", two: "two"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: "one", two: "two"}
 "two"
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, "ar", %{one: "one", few: "few", other: "other"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, "ar", %{one: "one", few: "few", other: "other"}
 "other"
 
-iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, "en", %{one: "one", few: "few", other: "other"}
+iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, "en", %{one: "one", few: "few", other: "other"}
 "other"
diff --git a/Plausible.Cldr.Number.Symbol.html b/Plausible.Cldr.Number.Symbol.html index 314245a785..cdf8daeb55 100644 --- a/Plausible.Cldr.Number.Symbol.html +++ b/Plausible.Cldr.Number.Symbol.html @@ -378,9 +378,9 @@ is Plausible.Cld Example: -
iex> Plausible.Cldr.Number.Symbol.number_symbols_for(:th)
-{:ok, %{
-   latn: %Cldr.Number.Symbol{
+
iex> Plausible.Cldr.Number.Symbol.number_symbols_for(:th)
+{:ok, %{
+   latn: %Cldr.Number.Symbol{
      decimal: ".",
      exponential: "E",
      group: ",",
@@ -393,8 +393,8 @@ is Plausible.Cld
      plus_sign: "+",
      superscripting_exponent: "×",
      time_separator: ":"
-   },
-   thai: %Cldr.Number.Symbol{
+   },
+   thai: %Cldr.Number.Symbol{
      decimal: ".",
      exponential: "E",
      group: ",",
@@ -407,8 +407,8 @@ is Plausible.Cld
      plus_sign: "+",
      superscripting_exponent: "×",
      time_separator: ":"
-   }
- }}
+
} + }}
diff --git a/Plausible.Cldr.Number.System.html b/Plausible.Cldr.Number.System.html index a389cff11a..717079048f 100644 --- a/Plausible.Cldr.Number.System.html +++ b/Plausible.Cldr.Number.System.html @@ -317,23 +317,23 @@ it is returned as is.

Examples

iex> Plausible.Cldr.Number.System.number_system_for "th", :latn
-{:ok, %{digits: "0123456789", type: :numeric}}
+{:ok, %{digits: "0123456789", type: :numeric}}
 
 iex> Plausible.Cldr.Number.System.number_system_for "en", :default
-{:ok, %{digits: "0123456789", type: :numeric}}
+{:ok, %{digits: "0123456789", type: :numeric}}
 
 iex> Plausible.Cldr.Number.System.number_system_for "he", :traditional
-{:ok, %{rules: "hebrew", type: :algorithmic}}
+{:ok, %{rules: "hebrew", type: :algorithmic}}
 
 iex> Plausible.Cldr.Number.System.number_system_for "en", :native
-{:ok, %{digits: "0123456789", type: :numeric}}
+{:ok, %{digits: "0123456789", type: :numeric}}
 
 iex> Plausible.Cldr.Number.System.number_system_for "en", :finance
-{
+{
   :error,
-  {Cldr.UnknownNumberSystemError,
-    "The number system :finance is unknown for the locale named :en. Valid number systems are %{default: :latn, native: :latn}"}
-}
+ {Cldr.UnknownNumberSystemError, + "The number system :finance is unknown for the locale named :en. Valid number systems are %{default: :latn, native: :latn}"} +}
@@ -424,10 +424,10 @@ or a Examples
iex> Plausible.Cldr.Number.System.number_system_names_for "en"
-{:ok, [:latn]}
+{:ok, [:latn]}
 
 iex> Plausible.Cldr.Number.System.number_system_names_for "zz"
-{:error, {Cldr.InvalidLanguageError, "The language \"zz\" is invalid"}}
+{:error, {Cldr.InvalidLanguageError, "The language \"zz\" is invalid"}}
@@ -509,13 +509,13 @@ or a Examples
iex> Plausible.Cldr.Number.System.number_systems_for "en"
-{:ok, %{default: :latn, native: :latn}}
+{:ok, %{default: :latn, native: :latn}}
 
 iex> Plausible.Cldr.Number.System.number_systems_for "th"
-{:ok, %{default: :latn, native: :thai}}
+{:ok, %{default: :latn, native: :thai}}
 
 iex> Plausible.Cldr.Number.System.number_systems_for "zz"
-{:error, {Cldr.InvalidLanguageError, "The language \"zz\" is invalid"}}
+{:error, {Cldr.InvalidLanguageError, "The language \"zz\" is invalid"}}
@@ -614,20 +614,20 @@ actual system name.

Examples

-
ex> Plausible.Cldr.Number.System.system_name_from(:default, "en")
-{:ok, :latn}
+
ex> Plausible.Cldr.Number.System.system_name_from(:default, "en")
+{:ok, :latn}
 
-iex> Plausible.Cldr.Number.System.system_name_from("latn", "en")
-{:ok, :latn}
+iex> Plausible.Cldr.Number.System.system_name_from("latn", "en")
+{:ok, :latn}
 
-iex> Plausible.Cldr.Number.System.system_name_from(:native, "en")
-{:ok, :latn}
+iex> Plausible.Cldr.Number.System.system_name_from(:native, "en")
+{:ok, :latn}
 
-iex> Plausible.Cldr.Number.System.system_name_from(:nope, "en")
-{
+iex> Plausible.Cldr.Number.System.system_name_from(:nope, "en")
+{
   :error,
-  {Cldr.UnknownNumberSystemError, "The number system :nope is unknown"}
-}

Note that return value is not guaranteed to be a valid + {Cldr.UnknownNumberSystemError, "The number system :nope is unknown"} +}

Note that return value is not guaranteed to be a valid number system for the given locale as demonstrated in the third example.

@@ -674,16 +674,16 @@ is recommended.

Examples

iex> Plausible.Cldr.Number.System.to_system 123456, :hebr
-{:ok, "קכ״ג׳תנ״ו"}
+{:ok, "קכ״ג׳תנ״ו"}
 
 iex> Plausible.Cldr.Number.System.to_system 123, :hans
-{:ok, "一百二十三"}
+{:ok, "一百二十三"}
 
 iex> Plausible.Cldr.Number.System.to_system 123, :hant
-{:ok, "一百二十三"}
+{:ok, "一百二十三"}
 
 iex> Plausible.Cldr.Number.System.to_system 123, :hansfin
-{:ok, "壹佰贰拾叁"}
+{:ok, "壹佰贰拾叁"}
diff --git a/Plausible.Cldr.Number.Transliterate.html b/Plausible.Cldr.Number.Transliterate.html index f12cc7cc25..79a1daf8be 100644 --- a/Plausible.Cldr.Number.Transliterate.html +++ b/Plausible.Cldr.Number.Transliterate.html @@ -162,12 +162,12 @@ digits between number systems. For example from :arabic to :latn. Since genera transliteration map is slow, pairs of transliterations can be configured so that the transliteration map is created at compile time and therefore speeding up transliteration at run time.

To configure these transliteration pairs, add the to the use Cldr configuration -in a backend module:

defmodule MyApp.Cldr do
+in a backend module:

defmodule MyApp.Cldr do
   use Cldr,
-  locale: ["en", "fr", "th"],
+  locale: ["en", "fr", "th"],
   default_locale: "en",
-  precompile_transliterations: [{:latn, :thai}, {:arab, :thai}]
-end

Where each tuple in the list configures one transliteration map. In this example, two maps are + precompile_transliterations: [{:latn, :thai}, {:arab, :thai}] +end

Where each tuple in the list configures one transliteration map. In this example, two maps are configured: from :latn to :thai and from :arab to :thai.

A list of configurable number systems is returned by Cldr.Number.System.numeric_systems/0.

If a transliteration is requested between two number pairs that have not been configured for precompilation, a warning is logged.

@@ -281,22 +281,22 @@ and Examples -
iex> Plausible.Cldr.Number.Transliterate.transliterate("123556")
+
iex> Plausible.Cldr.Number.Transliterate.transliterate("123556")
 "123556"
 
-iex> Plausible.Cldr.Number.Transliterate.transliterate("123,556.000", "fr", :default)
+iex> Plausible.Cldr.Number.Transliterate.transliterate("123,556.000", "fr", :default)
 "123 556,000"
 
-iex> Plausible.Cldr.Number.Transliterate.transliterate("123556", "th", :default)
+iex> Plausible.Cldr.Number.Transliterate.transliterate("123556", "th", :default)
 "123556"
 
-iex> Plausible.Cldr.Number.Transliterate.transliterate("123556", "th", "thai")
+iex> Plausible.Cldr.Number.Transliterate.transliterate("123556", "th", "thai")
 "๑๒๓๕๕๖"
 
-iex> Plausible.Cldr.Number.Transliterate.transliterate("123556", "th", :native)
+iex> Plausible.Cldr.Number.Transliterate.transliterate("123556", "th", :native)
 "๑๒๓๕๕๖"
 
-iex> Plausible.Cldr.Number.Transliterate.transliterate("Some number is: 123556", "th", "thai")
+iex> Plausible.Cldr.Number.Transliterate.transliterate("Some number is: 123556", "th", "thai")
 "Some number is: ๑๒๓๕๕๖"
diff --git a/Plausible.Cldr.Number.html b/Plausible.Cldr.Number.html index 1a11acf7fa..190a8e4232 100644 --- a/Plausible.Cldr.Number.html +++ b/Plausible.Cldr.Number.html @@ -441,19 +441,19 @@ using the Elixir standard library functions.

Examples

-
iex> Plausible.Cldr.Number.parse("+1.000,34", locale: "de")
-{:ok, 1000.34}
+
iex> Plausible.Cldr.Number.parse("+1.000,34", locale: "de")
+{:ok, 1000.34}
 
-iex> Plausible.Cldr.Number.parse("-1_000_000.34")
-{:ok, -1000000.34}
+iex> Plausible.Cldr.Number.parse("-1_000_000.34")
+{:ok, -1000000.34}
 
-iex> Plausible.Cldr.Number.parse("1.000", locale: "de", number: :integer)
-{:ok, 1000}
+iex> Plausible.Cldr.Number.parse("1.000", locale: "de", number: :integer)
+{:ok, 1000}
 
-iex> Plausible.Cldr.Number.parse("+1.000,34", locale: "de", number: :integer)
-{:error,
-  {Cldr.Number.ParseError,
-   "The string \"+1.000,34\" could not be parsed as a number"}}
+
iex> Plausible.Cldr.Number.parse("+1.000,34", locale: "de", number: :integer) +{:error, + {Cldr.Number.ParseError, + "The string \"+1.000,34\" could not be parsed as a number"}}
@@ -525,17 +525,17 @@ financial instruments.

Examples

-
iex> Plausible.Cldr.Number.scan("100 US dollars")
+
iex> Plausible.Cldr.Number.scan("100 US dollars")
 ...> |> Plausible.Cldr.Number.resolve_currencies
-[100, :USD]
+[100, :USD]
 
-iex> Plausible.Cldr.Number.scan("100 eurosports")
-...> |> Plausible.Cldr.Number.resolve_currencies(fuzzy: 0.75)
-[100, :EUR]
+iex> Plausible.Cldr.Number.scan("100 eurosports")
+...> |> Plausible.Cldr.Number.resolve_currencies(fuzzy: 0.75)
+[100, :EUR]
 
-iex> Plausible.Cldr.Number.scan("100 dollars des États-Unis")
-...> |> Plausible.Cldr.Number.resolve_currencies(locale: "fr")
-[100, :USD]
+
iex> Plausible.Cldr.Number.scan("100 dollars des États-Unis") +...> |> Plausible.Cldr.Number.resolve_currencies(locale: "fr") +[100, :USD]
@@ -612,19 +612,19 @@ financial instruments.

Examples

-
iex> Plausible.Cldr.Number.resolve_currency("US dollars")
-[:USD]
+
iex> Plausible.Cldr.Number.resolve_currency("US dollars")
+[:USD]
 
-iex> Plausible.Cldr.Number.resolve_currency("100 eurosports", fuzzy: 0.75)
-[:EUR]
+iex> Plausible.Cldr.Number.resolve_currency("100 eurosports", fuzzy: 0.75)
+[:EUR]
 
-iex> Plausible.Cldr.Number.resolve_currency("dollars des États-Unis", locale: "fr")
-[:USD]
+iex> Plausible.Cldr.Number.resolve_currency("dollars des États-Unis", locale: "fr")
+[:USD]
 
-iex> Plausible.Cldr.Number.resolve_currency("not a known currency", locale: "fr")
-{:error,
- {Cldr.UnknownCurrencyError,
-  "The currency \"not a known currency\" is unknown or not supported"}}
+
iex> Plausible.Cldr.Number.resolve_currency("not a known currency", locale: "fr") +{:error, + {Cldr.UnknownCurrencyError, + "The currency \"not a known currency\" is unknown or not supported"}}
@@ -688,13 +688,13 @@ The default is options[:backend].get_locale()Examples
iex> Plausible.Cldr.Number.resolve_per "11%"
-["11", :percent]
+["11", :percent]
 
 iex> Plausible.Cldr.Number.resolve_per "% of linguists"
-[:percent, " of linguists"]
+[:percent, " of linguists"]
 
 iex> Plausible.Cldr.Number.resolve_per "% of linguists %"
-[:percent, " of linguists ", :percent]
+[:percent, " of linguists ", :percent]
@@ -751,9 +751,9 @@ The default is options[:backend].get_locale() Examples -
iex> Plausible.Cldr.Number.scan("100%")
-...> |> Plausible.Cldr.Number.resolve_pers()
-[100, :percent]
+
iex> Plausible.Cldr.Number.scan("100%")
+...> |> Plausible.Cldr.Number.resolve_pers()
+[100, :percent]
@@ -813,20 +813,20 @@ and any options provided are passed to that function.

Examples

-
iex> Plausible.Cldr.Number.scan("£1_000_000.34")
-["£", 1000000.34]
+
iex> Plausible.Cldr.Number.scan("£1_000_000.34")
+["£", 1000000.34]
 
-iex> Plausible.Cldr.Number.scan("I want £1_000_000 dollars")
-["I want £", 1000000, " dollars"]
+iex> Plausible.Cldr.Number.scan("I want £1_000_000 dollars")
+["I want £", 1000000, " dollars"]
 
-iex> Plausible.Cldr.Number.scan("The prize is 23")
-["The prize is ", 23]
+iex> Plausible.Cldr.Number.scan("The prize is 23")
+["The prize is ", 23]
 
-iex> Plausible.Cldr.Number.scan("The lottery number is 23 for the next draw")
-["The lottery number is ", 23, " for the next draw"]
+iex> Plausible.Cldr.Number.scan("The lottery number is 23 for the next draw")
+["The lottery number is ", 23, " for the next draw"]
 
-iex> Plausible.Cldr.Number.scan("The loss is -1.000 euros", locale: "de", number: :integer)
-["The loss is ", -1000, " euros"]
+
iex> Plausible.Cldr.Number.scan("The loss is -1.000 euros", locale: "de", number: :integer) +["The loss is ", -1000, " euros"]
@@ -873,7 +873,7 @@ options.

Example

iex> Plausible.Cldr.Number.to_approx_string 1234
-{:ok, "~1,234"}
+{:ok, "~1,234"}
@@ -920,7 +920,7 @@ options.

Example

iex> Plausible.Cldr.Number.to_at_least_string 1234
-{:ok, "1,234+"}
+{:ok, "1,234+"}
@@ -967,7 +967,7 @@ options.

Example

iex> Plausible.Cldr.Number.to_at_most_string 1234
-{:ok, "≤1,234"}
+{:ok, "≤1,234"}
@@ -1014,7 +1014,7 @@ options.

Example

iex> Plausible.Cldr.Number.to_range_string 1234..5678
-{:ok, "1,234–5,678"}
+{:ok, "1,234–5,678"}
@@ -1133,21 +1133,21 @@ extensions.

iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn +applied to the symbol than the number. For example:

iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn
 ...>   string, :currency_symbol -> "<span class=\"symbol\">" <> string <> "</span>"
 ...>   string, :number -> "<span class=\"number\">" <> string <> "</span>"
 ...>   string, :currency_space -> "<span>" <> string <> "</span>"
 ...>   string, _other -> string
-...> end)
-{:ok, "<span class=\"symbol\">$</span><span class=\"number\">100.00</span>"}

It is also possible and recommended to use the Phoenix.HTML.Tag.content_tag/3 +...> end) +{:ok, "<span class=\"symbol\">$</span><span class=\"number\">100.00</span>"}

It is also possible and recommended to use the Phoenix.HTML.Tag.content_tag/3 function if wrapping HTML tags since these will ensure HTML entities are -correctly encoded. For example:

iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn
-...>   string, :currency_symbol -> Phoenix.HTML.Tag.content_tag(:span, string, class: "symbol")
-...>   string, :number -> Phoenix.HTML.Tag.content_tag(:span, string, class: "number")
-...>   string, :currency_space -> Phoenix.HTML.Tag.content_tag(:span, string)
+correctly encoded.  For example:

iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn
+...>   string, :currency_symbol -> Phoenix.HTML.Tag.content_tag(:span, string, class: "symbol")
+...>   string, :number -> Phoenix.HTML.Tag.content_tag(:span, string, class: "number")
+...>   string, :currency_space -> Phoenix.HTML.Tag.content_tag(:span, string)
 ...>   string, _other -> string
-...> end)
-{:ok, "<span class=\"symbol\">$</span><span class=\"number\">100.00</span>"}

When formatting a number the format is parsed into format elements that might include +...> end) +{:ok, "<span class=\"symbol\">$</span><span class=\"number\">100.00</span>"}

When formatting a number the format is parsed into format elements that might include a currency symbol, a literal string, inserted text between a currency symbol and the currency amount, a percent sign, the number itself and several other elements. In some cases it is helpful to be apply specific formatting to each element. @@ -1168,80 +1168,80 @@ inserted in the final formatted number.

Examples

iex> Plausible.Cldr.Number.to_string 12345
-{:ok, "12,345"}
+{:ok, "12,345"}
 
 iex> Plausible.Cldr.Number.to_string 12345, locale: "fr"
-{:ok, "12 345"}
+{:ok, "12 345"}
 
 iex> Plausible.Cldr.Number.to_string 1345.32, currency: :EUR, locale: "es", minimum_grouping_digits: 1
-{:ok, "1.345,32 €"}
+{:ok, "1.345,32 €"}
 
 iex> Plausible.Cldr.Number.to_string 1345.32, currency: :EUR, locale: "es"
-{:ok, "1345,32 €"}
+{:ok, "1345,32 €"}
 
 iex> Plausible.Cldr.Number.to_string 12345, locale: "fr", currency: "USD"
-{:ok, "12 345,00 $US"}
+{:ok, "12 345,00 $US"}
 
 iex> Plausible.Cldr.Number.to_string 12345, format: "#E0"
-{:ok, "1.2345E4"}
+{:ok, "1.2345E4"}
 
 iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: "THB"
-{:ok, "THB 12,345.00"}
+{:ok, "THB 12,345.00"}
 
 iex> Plausible.Cldr.Number.to_string -12345, format: :accounting, currency: "THB"
-{:ok, "(THB 12,345.00)"}
+{:ok, "(THB 12,345.00)"}
 
 iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: "THB",
 ...> locale: "th"
-{:ok, "฿12,345.00"}
+{:ok, "฿12,345.00"}
 
 iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: "THB",
 ...> locale: "th", number_system: :native
-{:ok, "฿๑๒,๓๔๕.๐๐"}
+{:ok, "฿๑๒,๓๔๕.๐๐"}
 
 iex> Plausible.Cldr.Number.to_string 1244.30, format: :long
-{:ok, "1 thousand"}
+{:ok, "1 thousand"}
 
 iex> Plausible.Cldr.Number.to_string 1244.30, format: :long, currency: "USD"
-{:ok, "1,244 US dollars"}
+{:ok, "1,244 US dollars"}
 
 iex> Plausible.Cldr.Number.to_string 1244.30, format: :short
-{:ok, "1K"}
+{:ok, "1K"}
 
 iex> Plausible.Cldr.Number.to_string 1244.30, format: :short, currency: "EUR"
-{:ok, "€1K"}
+{:ok, "€1K"}
 
 iex> Plausible.Cldr.Number.to_string 1234, format: :spellout
-{:ok, "one thousand two hundred thirty-four"}
+{:ok, "one thousand two hundred thirty-four"}
 
 iex> Plausible.Cldr.Number.to_string 1234, format: :spellout_verbose
-{:ok, "one thousand two hundred and thirty-four"}
+{:ok, "one thousand two hundred and thirty-four"}
 
 iex> Plausible.Cldr.Number.to_string 1989, format: :spellout_year
-{:ok, "nineteen eighty-nine"}
+{:ok, "nineteen eighty-nine"}
 
 iex> Plausible.Cldr.Number.to_string 123, format: :ordinal
-{:ok, "123rd"}
+{:ok, "123rd"}
 
 iex> Plausible.Cldr.Number.to_string 123, format: :roman
-{:ok, "CXXIII"}
+{:ok, "CXXIII"}
 
 iex> Plausible.Cldr.Number.to_string 123, locale: "th-u-nu-thai"
-{:ok, "๑๒๓"}

+{:ok, "๑๒๓"}

Errors

An error tuple {:error, reason} will be returned if an error is detected. -The two most likely causes of an error return are:

  • A format cannot be compiled. In this case the error tuple will look like:
    iex> Plausible.Cldr.Number.to_string(12345, format: "0#")
-    {:error, {Cldr.FormatCompileError,
-      "Decimal format compiler: syntax error before: \"#\""}}
  • The format style requested is not defined for the locale and +The two most likely causes of an error return are:

    • A format cannot be compiled. In this case the error tuple will look like:
        iex> Plausible.Cldr.Number.to_string(12345, format: "0#")
    +    {:error, {Cldr.FormatCompileError,
    +      "Decimal format compiler: syntax error before: \"#\""}}
    • The format style requested is not defined for the locale and number_system. This happens typically when the number system is :algorithmic rather than the more common :numeric. In this case the error -return looks like:
        iex> Plausible.Cldr.Number.to_string(1234, locale: "he", number_system: "hebr", format: :percent)
    -    {:error, {Cldr.UnknownFormatError,
    -      "The locale :he with number system :hebr does not define a format :percent"}}
    +return looks like:
    iex> Plausible.Cldr.Number.to_string(1234, locale: "he", number_system: "hebr", format: :percent)
+    {:error, {Cldr.UnknownFormatError,
+      "The locale :he with number system :hebr does not define a format :percent"}}
@@ -1346,17 +1346,17 @@ returned by Examples
iex> Plausible.Cldr.Number.validate_number_system "en", :latn
-{:ok, :latn}
+{:ok, :latn}
 
 iex> Plausible.Cldr.Number.validate_number_system "en", :default
-{:ok, :latn}
+{:ok, :latn}
 
 iex> Plausible.Cldr.Number.validate_number_system "en", :unknown
-{:error,
- {Cldr.UnknownNumberSystemError, "The number system :unknown is unknown"}}
+{:error,
+ {Cldr.UnknownNumberSystemError, "The number system :unknown is unknown"}}
 
 iex> Plausible.Cldr.Number.validate_number_system "zz", :default
-{:error, {Cldr.InvalidLanguageError, "The language \"zz\" is invalid"}}
+{:error, {Cldr.InvalidLanguageError, "The language \"zz\" is invalid"}}
diff --git a/Plausible.Cldr.Rbnf.NumberSystem.html b/Plausible.Cldr.Rbnf.NumberSystem.html index a9d953eb05..ec641c38fb 100644 --- a/Plausible.Cldr.Rbnf.NumberSystem.html +++ b/Plausible.Cldr.Rbnf.NumberSystem.html @@ -145,9 +145,9 @@ Pages

Functions to implement the number system rule-based-number-format rules of CLDR.

These rules are defined only on the "und" locale and represent specialised number formatting.

The standard public API for RBNF is via the Cldr.Number.to_string/2 function.

The functions on this module are defined at compile time based upon the RBNF rules -defined in the Unicode CLDR data repository. Available rules are identified by:

iex> Plausible.Cldr.Rbnf.NumberSystem.rule_sets(:und)
-...> |> Enum.sort()
-[
+defined in the Unicode CLDR data repository.  Available rules are identified by:

iex> Plausible.Cldr.Rbnf.NumberSystem.rule_sets(:und)
+...> |> Enum.sort()
+[
   :armenian_lower,
   :armenian_upper,
   :cyrillic_lower,
@@ -161,9 +161,9 @@ defined in the Unicode CLDR data repository.  Available rules are identified by:
   :roman_upper,
   :tamil,
   :zz_default
-]

A rule can then be invoked on an available rule_set. For example

iex> Plausible.Cldr.Rbnf.NumberSystem.roman_upper(123, :und)
-"CXXIII"

This particular call is equivalent to the call through the public API of:

iex> Plausible.Cldr.Number.to_string(123, format: :roman)
-{:ok, "CXXIII"}
+
]

A rule can then be invoked on an available rule_set. For example

iex> Plausible.Cldr.Rbnf.NumberSystem.roman_upper(123, :und)
+"CXXIII"

This particular call is equivalent to the call through the public API of:

iex> Plausible.Cldr.Number.to_string(123, format: :roman)
+{:ok, "CXXIII"}
diff --git a/Plausible.Cldr.Rbnf.Ordinal.html b/Plausible.Cldr.Rbnf.Ordinal.html index 4c6d404f65..c230c7ab2c 100644 --- a/Plausible.Cldr.Rbnf.Ordinal.html +++ b/Plausible.Cldr.Rbnf.Ordinal.html @@ -145,20 +145,20 @@ Pages

Functions to implement the ordinal rule-based-number-format rules of CLDR.

As CLDR notes, the data is incomplete or non-existent for many languages. It is considered complete for English however.

The standard public API for RBNF is via the Cldr.Number.to_string/2 function.

The functions on this module are defined at compile time based upon the RBNF rules -defined in the Unicode CLDR data repository. Available rules are identified by:

iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets(:en)
-[:digits_ordinal]
+defined in the Unicode CLDR data repository.  Available rules are identified by:

iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets(:en)
+[:digits_ordinal]
 
-iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets("fr")
-...> |> Enum.sort()
-[
+iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets("fr")
+...> |> Enum.sort()
+[
   :digits_ordinal,
   :digits_ordinal_feminine,
   :digits_ordinal_feminine_plural,
   :digits_ordinal_masculine,
   :digits_ordinal_masculine_plural
-]

A rule can then be invoked on an available rule_set. For example

iex> Plausible.Cldr.Rbnf.Ordinal.digits_ordinal(123, :en)
-"123rd"

This call is equivalent to the call through the public API of:

iex> Plausible.Cldr.Number.to_string(123, format: :ordinal)
-{:ok, "123rd"}
+
]

A rule can then be invoked on an available rule_set. For example

iex> Plausible.Cldr.Rbnf.Ordinal.digits_ordinal(123, :en)
+"123rd"

This call is equivalent to the call through the public API of:

iex> Plausible.Cldr.Number.to_string(123, format: :ordinal)
+{:ok, "123rd"}
diff --git a/Plausible.Cldr.Rbnf.Spellout.html b/Plausible.Cldr.Rbnf.Spellout.html index e8d607a0ba..f1e24215dc 100644 --- a/Plausible.Cldr.Rbnf.Spellout.html +++ b/Plausible.Cldr.Rbnf.Spellout.html @@ -145,9 +145,9 @@ Pages

Functions to implement the spellout rule-based-number-format rules of CLDR.

As CLDR notes, the data is incomplete or non-existent for many languages. It is considered complete for English however.

The standard public API for RBNF is via the Cldr.Number.to_string/2 function.

The functions on this module are defined at compile time based upon the RBNF rules -defined in the Unicode CLDR data repository. Available rules are identified by:

iex> Plausible.Cldr.Rbnf.Spellout.rule_sets("en")
-...> |> Enum.sort()
-[
+defined in the Unicode CLDR data repository.  Available rules are identified by:

iex> Plausible.Cldr.Rbnf.Spellout.rule_sets("en")
+...> |> Enum.sort()
+[
   :spellout_cardinal,
   :spellout_cardinal_verbose,
   :spellout_numbering,
@@ -155,9 +155,9 @@ defined in the Unicode CLDR data repository.  Available rules are identified by:
   :spellout_numbering_year,
   :spellout_ordinal,
   :spellout_ordinal_verbose
-]

A rule can then be invoked on an available rule_set. For example:

iex> Plausible.Cldr.Rbnf.Spellout.spellout_ordinal(123, "en")
-"one hundred twenty-third"

This call is equivalent to the call through the public API of:

iex> Plausible.Cldr.Number.to_string(123, format: :spellout)
-{:ok, "one hundred twenty-three"}
+
]

A rule can then be invoked on an available rule_set. For example:

iex> Plausible.Cldr.Rbnf.Spellout.spellout_ordinal(123, "en")
+"one hundred twenty-third"

This call is equivalent to the call through the public API of:

iex> Plausible.Cldr.Number.to_string(123, format: :spellout)
+{:ok, "one hundred twenty-three"}
diff --git a/Plausible.Cldr.html b/Plausible.Cldr.html index b21d1bc0dd..fba423c009 100644 --- a/Plausible.Cldr.html +++ b/Plausible.Cldr.html @@ -558,24 +558,24 @@ current locale is restored after the function.

Example -
iex> Plausible.Cldr.default_locale()
-%Cldr.LanguageTag{
+
iex> Plausible.Cldr.default_locale()
+%Cldr.LanguageTag{
   backend: Plausible.Cldr,
   canonical_locale_name: "en-001",
   cldr_locale_name: :"en-001",
-  language_subtags: [],
-  extensions: %{},
+  language_subtags: [],
+  extensions: %{},
   gettext_locale_name: "en",
   language: "en",
-  locale: %{},
-  private_use: [],
+  locale: %{},
+  private_use: [],
   rbnf_locale_name: :en,
   requested_locale_name: "en-001",
   script: :Latn,
   territory: :"001",
-  transform: %{},
-  language_variants: []
-}
+
transform: %{}, + language_variants: [] +}
@@ -610,7 +610,7 @@ does not specify one and none can be inferred.

Example

-
iex> Plausible.Cldr.default_territory()
+
iex> Plausible.Cldr.default_territory()
 :"001"
@@ -670,16 +670,16 @@ is inserted between words or sentences. The valid options are Examples -
iex> Plausible.Cldr.ellipsis("And furthermore")
+
iex> Plausible.Cldr.ellipsis("And furthermore")
 "And furthermore…"
 
-iex> Plausible.Cldr.ellipsis(["And furthermore", "there is much to be done"], locale: :ja)
+iex> Plausible.Cldr.ellipsis(["And furthermore", "there is much to be done"], locale: :ja)
 "And furthermore…there is much to be done"
 
-iex> Plausible.Cldr.ellipsis("And furthermore", format: :word)
+iex> Plausible.Cldr.ellipsis("And furthermore", format: :word)
 "And furthermore …"
 
-iex> Plausible.Cldr.ellipsis(["And furthermore", "there is much to be done"], locale: :ja, format: :word)
+iex> Plausible.Cldr.ellipsis(["And furthermore", "there is much to be done"], locale: :ja, format: :word)
 "And furthermore … there is much to be done"
@@ -715,23 +715,23 @@ take an optional locale parameter for which a locale is not supplied.

Example

-
iex> Plausible.Cldr.put_locale("pl")
-iex> Plausible.Cldr.get_locale()
-%Cldr.LanguageTag{
+
iex> Plausible.Cldr.put_locale("pl")
+iex> Plausible.Cldr.get_locale()
+%Cldr.LanguageTag{
    backend: Elixir.Plausible.Cldr,
    canonical_locale_name: "pl",
    cldr_locale_name: :pl,
-   extensions: %{},
+   extensions: %{},
    language: "pl",
-   locale: %{},
-   private_use: [],
+   locale: %{},
+   private_use: [],
    rbnf_locale_name: :pl,
    territory: :PL,
    requested_locale_name: "pl",
    script: :Latn,
-   transform: %{},
-   language_variants: []
- }
+
transform: %{}, + language_variants: [] + }
@@ -818,10 +818,10 @@ take an optional locale parameter for which a locale is not supplied.

Examples

-
iex> Plausible.Cldr.known_gettext_locale_name("en")
+
iex> Plausible.Cldr.known_gettext_locale_name("en")
 "en"
 
-iex> Plausible.Cldr.known_gettext_locale_name("en-SA")
+iex> Plausible.Cldr.known_gettext_locale_name("en-SA")
 false
@@ -864,10 +864,10 @@ name is configured and available in Gettext.

Examples

-
iex> Plausible.Cldr.known_gettext_locale_name?("en")
+
iex> Plausible.Cldr.known_gettext_locale_name?("en")
 true
 
-iex> Plausible.Cldr.known_gettext_locale_name?("!!")
+iex> Plausible.Cldr.known_gettext_locale_name?("!!")
 false
@@ -941,10 +941,10 @@ to return the first known locale name from a list.

Examples

-
iex> Plausible.Cldr.known_locale_name(:"en-AU")
+
iex> Plausible.Cldr.known_locale_name(:"en-AU")
 :"en-AU"
 
-iex> Plausible.Cldr.known_locale_name(:"en-SA")
+iex> Plausible.Cldr.known_locale_name(:"en-SA")
 false
@@ -986,10 +986,10 @@ name is configured and available in Cldr.

Examples

-
iex> Plausible.Cldr.known_locale_name?(:en)
+
iex> Plausible.Cldr.known_locale_name?(:en)
 true
 
-iex> Plausible.Cldr.known_locale_name?(:"!!")
+iex> Plausible.Cldr.known_locale_name?(:"!!")
 false
@@ -1043,8 +1043,8 @@ in this module or in Example -
iex> Plausible.Cldr.known_number_system_types()
-[:default, :finance, :native, :traditional]
+
iex> Plausible.Cldr.known_number_system_types()
+[:default, :finance, :native, :traditional]
@@ -1109,10 +1109,10 @@ and has RBNF rules defined.

Examples

-
iex> Plausible.Cldr.known_rbnf_locale_name(:en)
+
iex> Plausible.Cldr.known_rbnf_locale_name(:en)
 :en
 
-iex> Plausible.Cldr.known_rbnf_locale_name(:"en-SA")
+iex> Plausible.Cldr.known_rbnf_locale_name(:"en-SA")
 false
@@ -1155,10 +1155,10 @@ rules based number formats (RBNF).

Examples

-
iex> Plausible.Cldr.known_rbnf_locale_name?(:en)
+
iex> Plausible.Cldr.known_rbnf_locale_name?(:en)
 true
 
-iex> Plausible.Cldr.known_rbnf_locale_name?(:"!!")
+iex> Plausible.Cldr.known_rbnf_locale_name?(:"!!")
 false
@@ -1294,18 +1294,18 @@ CLDR backend defined by the t:Cldr.LanguageTag is se Examples
iex> import Cldr.LanguageTag.Sigil
-iex> Plausible.Cldr.put_gettext_locale(~l"en")
-{:ok, "en"}
+iex> Plausible.Cldr.put_gettext_locale(~l"en")
+{:ok, "en"}
 
 iex> import Cldr.LanguageTag.Sigil
-iex> Plausible.Cldr.put_gettext_locale(~l"de")
-{
+iex> Plausible.Cldr.put_gettext_locale(~l"de")
+{
   :error,
-  {
+  {
     Cldr.UnknownLocaleError,
     "Locale TestBackend.Cldr.Locale.new!(\"de-DE\") does not map to a known gettext locale name"
-  }
-}
+
} +}
@@ -1349,29 +1349,29 @@ of a language tag.

Examples

-
iex> Plausible.Cldr.put_locale("en")
-{:ok,
- %Cldr.LanguageTag{
+
iex> Plausible.Cldr.put_locale("en")
+{:ok,
+ %Cldr.LanguageTag{
    backend: Plausible.Cldr,
    canonical_locale_name: "en",
    cldr_locale_name: :en,
-   language_subtags: [],
-   extensions: %{},
+   language_subtags: [],
+   extensions: %{},
    gettext_locale_name: "en",
    language: "en",
-   locale: %{},
-   private_use: [],
+   locale: %{},
+   private_use: [],
    rbnf_locale_name: :en,
    requested_locale_name: "en",
    script: :Latn,
    territory: :US,
-   transform: %{},
-   language_variants: []
- }}
+   transform: %{},
+   language_variants: []
+ }}
 
-iex> Plausible.Cldr.put_locale("invalid-locale!")
-{:error, {Cldr.LanguageTag.ParseError,
-  "Expected a BCP47 language tag. Could not parse the remaining \"!\" starting at position 15"}}
+
iex> Plausible.Cldr.put_locale("invalid-locale!") +{:error, {Cldr.LanguageTag.ParseError, + "Expected a BCP47 language tag. Could not parse the remaining \"!\" starting at position 15"}}
@@ -1420,10 +1420,10 @@ The default is Examples -
iex> Plausible.Cldr.quote("Quoted String")
+
iex> Plausible.Cldr.quote("Quoted String")
 "“Quoted String”"
 
-iex> Plausible.Cldr.quote("Quoted String", locale: :ja)
+iex> Plausible.Cldr.quote("Quoted String", locale: :ja)
 "「Quoted String」"
@@ -1554,47 +1554,47 @@ of a language tag.

Examples

-
iex> Plausible.Cldr.validate_locale(:en)
-{:ok,
-%Cldr.LanguageTag{
+
iex> Plausible.Cldr.validate_locale(:en)
+{:ok,
+%Cldr.LanguageTag{
   backend: Plausible.Cldr,
   canonical_locale_name: "en",
   cldr_locale_name: :en,
-  extensions: %{},
+  extensions: %{},
   gettext_locale_name: "en",
   language: "en",
-  locale: %{},
-  private_use: [],
+  locale: %{},
+  private_use: [],
   rbnf_locale_name: :en,
   requested_locale_name: "en",
   script: :Latn,
   territory: :US,
-  transform: %{},
-  language_variants: []
-}}
+  transform: %{},
+  language_variants: []
+}}
 
 
-iex> Plausible.Cldr.validate_locale Plausible.Cldr.default_locale()
-{:ok,
-%Cldr.LanguageTag{
+iex> Plausible.Cldr.validate_locale Plausible.Cldr.default_locale()
+{:ok,
+%Cldr.LanguageTag{
   backend: Plausible.Cldr,
   canonical_locale_name: "en-001",
   cldr_locale_name: :"en-001",
-  extensions: %{},
+  extensions: %{},
   gettext_locale_name: "en",
   language: "en",
-  locale: %{},
-  private_use: [],
+  locale: %{},
+  private_use: [],
   rbnf_locale_name: :en,
   requested_locale_name: "en-001",
   script: :Latn,
   territory: :"001",
-  transform: %{},
-  language_variants: []
-}}
+  transform: %{},
+  language_variants: []
+}}
 
-iex> Plausible.Cldr.validate_locale("zzz")
-{:error, {Cldr.InvalidLanguageError, "The language \"zzz\" is invalid"}}
+
iex> Plausible.Cldr.validate_locale("zzz") +{:error, {Cldr.InvalidLanguageError, "The language \"zzz\" is invalid"}}
@@ -1664,23 +1664,23 @@ of a language tag.

Examples

-
iex> Plausible.Cldr.validate_number_system_type(:default)
-{:ok, :default}
+
iex> Plausible.Cldr.validate_number_system_type(:default)
+{:ok, :default}
 
-iex> Plausible.Cldr.validate_number_system_type(:traditional)
-{:ok, :traditional}
+iex> Plausible.Cldr.validate_number_system_type(:traditional)
+{:ok, :traditional}
 
-iex> Plausible.Cldr.validate_number_system_type(:latn)
-{
+iex> Plausible.Cldr.validate_number_system_type(:latn)
+{
   :error,
-  {Cldr.UnknownNumberSystemTypeError, "The number system type :latn is unknown"}
-}
+  {Cldr.UnknownNumberSystemTypeError, "The number system type :latn is unknown"}
+}
 
-iex> Plausible.Cldr.validate_number_system_type("bork")
-{
+iex> Plausible.Cldr.validate_number_system_type("bork")
+{
   :error,
-  {Cldr.UnknownNumberSystemTypeError, "The number system type \"bork\" is invalid"}
-}
+
{Cldr.UnknownNumberSystemTypeError, "The number system type \"bork\" is invalid"} +}
diff --git a/Plausible.ClickhouseRepo.html b/Plausible.ClickhouseRepo.html index 182a3f760a..e59879e2fe 100644 --- a/Plausible.ClickhouseRepo.html +++ b/Plausible.ClickhouseRepo.html @@ -879,23 +879,23 @@ pool to disconnect within the given interval.

See -

Similar to insert_all/2 but with the following differences:

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
+

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
 
-defmodule Demo do
+defmodule Demo do
   use Ecto.Schema
 
   @primary_key false
-  schema "ecto_ch_demo" do
+  schema "ecto_ch_demo" do
     field :a, Ch, type: "UInt64"
     field :b, :string
-  end
-end
+  end
+end
 
-rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
-{100_000, nil} = Repo.insert_stream(Demo, rows)
+rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
+{100_000, nil} = Repo.insert_stream(Demo, rows)
 
 # schemaless
-{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
+
{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
diff --git a/Plausible.DataMigration.Repo.html b/Plausible.DataMigration.Repo.html index 31dc6d0d2f..70e161cca5 100644 --- a/Plausible.DataMigration.Repo.html +++ b/Plausible.DataMigration.Repo.html @@ -1196,23 +1196,23 @@ pool to disconnect within the given interval.

See -

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
+

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
 
-defmodule Demo do
+defmodule Demo do
   use Ecto.Schema
 
   @primary_key false
-  schema "ecto_ch_demo" do
+  schema "ecto_ch_demo" do
     field :a, Ch, type: "UInt64"
     field :b, :string
-  end
-end
+  end
+end
 
-rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
-{100_000, nil} = Repo.insert_stream(Demo, rows)
+rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
+{100_000, nil} = Repo.insert_stream(Demo, rows)
 
 # schemaless
-{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
+
{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
diff --git a/Plausible.Exports.html b/Plausible.Exports.html index a17cec9da3..ee103fb59a 100644 --- a/Plausible.Exports.html +++ b/Plausible.Exports.html @@ -381,7 +381,7 @@ tables into the format of imported_* tables for a we
-

Renders export archive filename.

Examples:

iex> archive_filename("plausible.io", _created_on = ~D[2024-12-31])
+

Renders export archive filename.

Examples:

iex> archive_filename("plausible.io", _created_on = ~D[2024-12-31])
 "plausible_io_20241231.zip"
@@ -404,10 +404,10 @@ tables into the format of imported_* tables for a we
-

Safely renders content disposition for an arbitrary export filename.

Examples:

iex> content_disposition("plausible_io_20241231.zip")
+

Safely renders content disposition for an arbitrary export filename.

Examples:

iex> content_disposition("plausible_io_20241231.zip")
 "attachment; filename=\"plausible_io_20241231.zip\""
 
-iex> content_disposition("📊.zip")
+iex> content_disposition("📊.zip")
 "attachment; filename=\"plausible-export.zip\"; filename*=utf-8''%F0%9F%93%8A.zip"
@@ -726,14 +726,14 @@ tables into the format of imported_* tables for a we -

Creates a streamable Zip archive from the provided (named) Ecto queries.

Example usage:

{:ok, pool} = Ch.start_link(pool_size: 1)
+

Creates a streamable Zip archive from the provided (named) Ecto queries.

Example usage:

{:ok, pool} = Ch.start_link(pool_size: 1)
 
-DBConnection.run(pool, fn conn ->
+DBConnection.run(pool, fn conn ->
   conn
-  |> stream_archive(export_queries(_site_id = 1), format: "CSVWithNames")
-  |> Stream.into(File.stream!("export.zip"))
-  |> Stream.run()
-end)
+
|> stream_archive(export_queries(_site_id = 1), format: "CSVWithNames") + |> Stream.into(File.stream!("export.zip")) + |> Stream.run() +end)
diff --git a/Plausible.Geo.html b/Plausible.Geo.html index d7665902ca..0eae745acf 100644 --- a/Plausible.Geo.html +++ b/Plausible.Geo.html @@ -260,8 +260,8 @@ and MaxMind license key.

Examples -

In the case of a DB-IP database:

iex> database_type()
-"DBIP-City-Lite"

In the case of a MaxMind database:

iex> database_type()
+

In the case of a DB-IP database:

iex> database_type()
+"DBIP-City-Lite"

In the case of a MaxMind database:

iex> database_type()
 "GeoLite2-City"
@@ -302,8 +302,8 @@ asynchronously.

Examples

-

Loading from a local file:

iex> load_db(path: "/etc/plausible/dbip-city.mmdb")
-:ok

Downloading a MaxMind DB (this license key is no longer active):

iex> load_db(license_key: "LNpsJCCKPis6XvBP", edition: "GeoLite2-City", async: true)
+

Loading from a local file:

iex> load_db(path: "/etc/plausible/dbip-city.mmdb")
+:ok

Downloading a MaxMind DB (this license key is no longer active):

iex> load_db(license_key: "LNpsJCCKPis6XvBP", edition: "GeoLite2-City", async: true)
 :ok
@@ -332,21 +332,21 @@ asynchronously.

Examples

-
iex> lookup("8.7.6.5")
-%{
-  "city" => %{
+
iex> lookup("8.7.6.5")
+%{
+  "city" => %{
     "geoname_id" => 5349755,
-    "names" => %{
+    "names" => %{
       "de" => "Fontana",
       "en" => "Fontana",
       "ja" => "フォンタナ",
       "ru" => "Фонтана"
-    }
-  },
-  "continent" => %{
+    }
+  },
+  "continent" => %{
     "code" => "NA",
     "geoname_id" => 6255149,
-    "names" => %{
+    "names" => %{
       "de" => "Nordamerika",
       "en" => "North America",
       "es" => "Norteamérica",
@@ -355,12 +355,12 @@ asynchronously.

"pt-BR" => "América do Norte", "ru" => "Северная Америка", "zh-CN" => "北美洲" - } - }, - "country" => %{ + } + }, + "country" => %{ "geoname_id" => 6252001, "iso_code" => "US", - "names" => %{ + "names" => %{ "de" => "Vereinigte Staaten", "en" => "United States", "es" => "Estados Unidos", @@ -369,20 +369,20 @@ asynchronously.

"pt-BR" => "EUA", "ru" => "США", "zh-CN" => "美国" - } - }, - "location" => %{ + } + }, + "location" => %{ "accuracy_radius" => 50, "latitude" => 34.1211, "longitude" => -117.4362, "metro_code" => 803, "time_zone" => "America/Los_Angeles" - }, - "postal" => %{"code" => "92336"}, - "registered_country" => %{ + }, + "postal" => %{"code" => "92336"}, + "registered_country" => %{ "geoname_id" => 6252001, "iso_code" => "US", - "names" => %{ + "names" => %{ "de" => "Vereinigte Staaten", "en" => "United States", "es" => "Estados Unidos", @@ -391,13 +391,13 @@ asynchronously.

"pt-BR" => "EUA", "ru" => "США", "zh-CN" => "美国" - } - }, - "subdivisions" => [ - %{ + } + }, + "subdivisions" => [ + %{ "geoname_id" => 5332921, "iso_code" => "CA", - "names" => %{ + "names" => %{ "de" => "Kalifornien", "en" => "California", "es" => "California", @@ -406,10 +406,10 @@ asynchronously.

"pt-BR" => "Califórnia", "ru" => "Калифорния", "zh-CN" => "加州" - } - } - ] -}

+
} + } + ] +}
diff --git a/Plausible.ImportDeletionRepo.html b/Plausible.ImportDeletionRepo.html index 36c9a595a2..9b39fee2f3 100644 --- a/Plausible.ImportDeletionRepo.html +++ b/Plausible.ImportDeletionRepo.html @@ -1187,23 +1187,23 @@ pool to disconnect within the given interval.

See -

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
+

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
 
-defmodule Demo do
+defmodule Demo do
   use Ecto.Schema
 
   @primary_key false
-  schema "ecto_ch_demo" do
+  schema "ecto_ch_demo" do
     field :a, Ch, type: "UInt64"
     field :b, :string
-  end
-end
+  end
+end
 
-rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
-{100_000, nil} = Repo.insert_stream(Demo, rows)
+rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
+{100_000, nil} = Repo.insert_stream(Demo, rows)
 
 # schemaless
-{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
+
{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
diff --git a/Plausible.Imported.CSVImporter.html b/Plausible.Imported.CSVImporter.html index 9ee86d999b..58dddf2cca 100644 --- a/Plausible.Imported.CSVImporter.html +++ b/Plausible.Imported.CSVImporter.html @@ -257,13 +257,13 @@ or from local storage for which it uses iex> date_range([ -...> %{"filename" => "imported_devices_20190101_20210101.csv"}, +

Extracts min/max date range from a list of uploads.

Examples:

iex> date_range([
+...>   %{"filename" => "imported_devices_20190101_20210101.csv"},
 ...>   "pages_20200101_20220101.csv"
-...> ])
-Date.range(~D[2019-01-01], ~D[2022-01-01])
+...> ])
+Date.range(~D[2019-01-01], ~D[2022-01-01])
 
-iex> date_range([])
+iex> date_range([])
 nil
@@ -292,13 +292,13 @@ or from local storage for which it uses iex> extract_table("my_data.csv") +

Extracts the table name from the provided filename.

Raises if the filename doesn't conform to the expected format.

Examples:

iex> extract_table("my_data.csv")
 ** (ArgumentError) invalid filename
 
-iex> extract_table("imported_devices_00010101_20250101.csv")
+iex> extract_table("imported_devices_00010101_20250101.csv")
 "imported_devices"
 
-iex> extract_table("devices_00010101_20250101.csv")
+iex> extract_table("devices_00010101_20250101.csv")
 "imported_devices"
@@ -321,8 +321,8 @@ or from local storage for which it uses
-

Returns local directory for CSV imports storage.

Builds upon $DATA_DIR, $PERSISTENT_CACHE_DIR or $DEFAULT_DATA_DIR (if set) and falls back to /tmp.

$DEFAULT_DATA_DIR is set to /var/lib/plausible in container images.

Examples:

iex> local_dir = local_dir(_site_id = 37)
-iex> String.ends_with?(local_dir, "/plausible-imports/37")
+

Returns local directory for CSV imports storage.

Builds upon $DATA_DIR, $PERSISTENT_CACHE_DIR or $DEFAULT_DATA_DIR (if set) and falls back to /tmp.

$DEFAULT_DATA_DIR is set to /var/lib/plausible in container images.

Examples:

iex> local_dir = local_dir(_site_id = 37)
+iex> String.ends_with?(local_dir, "/plausible-imports/37")
 true
@@ -382,14 +382,14 @@ or from local storage for which it uses
iex> parse_filename!("my_data.csv") +

Extracts table name and min/max dates from the filename.

Examples:

iex> parse_filename!("my_data.csv")
 ** (ArgumentError) invalid filename
 
-iex> parse_filename!("imported_devices_00010101_20250101.csv")
-{"imported_devices", ~D[0001-01-01], ~D[2025-01-01]}
+iex> parse_filename!("imported_devices_00010101_20250101.csv")
+{"imported_devices", ~D[0001-01-01], ~D[2025-01-01]}
 
-iex> parse_filename!("devices_00010101_20250101.csv")
-{"imported_devices", ~D[0001-01-01], ~D[2025-01-01]}
+
iex> parse_filename!("devices_00010101_20250101.csv") +{"imported_devices", ~D[0001-01-01], ~D[2025-01-01]}
@@ -417,13 +417,13 @@ or from local storage for which it uses iex> valid_filename?("my_data.csv") +

Checks if the provided filename conforms to the expected format.

Examples:

iex> valid_filename?("my_data.csv")
 false
 
-iex> valid_filename?("imported_devices_00010101_20250101.csv")
+iex> valid_filename?("imported_devices_00010101_20250101.csv")
 true
 
-iex> valid_filename?("devices_00010101_20250101.csv")
+iex> valid_filename?("devices_00010101_20250101.csv")
 true
diff --git a/Plausible.Imported.Importer.html b/Plausible.Imported.Importer.html index 286cdd7ad1..a17dd5a645 100644 --- a/Plausible.Imported.Importer.html +++ b/Plausible.Imported.Importer.html @@ -183,30 +183,30 @@ scope of importer logic and is expected to be implemented separately.

In case it's necessary to run the whole import job fully synchronously, the Plausible.Workers.ImportAnalytics worker sends an Oban.Notifier message -on completion, failure or transient failure of the import.

A basic usage scenario looks like this:

{:ok, job} = Plausible.Imported.NoopImporter.new_import(
+on completion, failure or transient failure of the import.

A basic usage scenario looks like this:

{:ok, job} = Plausible.Imported.NoopImporter.new_import(
   site,
   user,
   start_date: ~D[2005-01-01],
-  end_date: Date.utc_today(),
+  end_date: Date.utc_today(),
   # this option is necessary to setup the calling process as listener
   listen?: true
-)
+)
 
-import_id = job.args[:import_id]
+import_id = job.args[:import_id]
 
-receive do
-  {:notification, :analytics_imports_jobs, %{"event" => "complete", "import_id" => ^import_id}} ->
-    IO.puts("Job completed")
+receive do
+  {:notification, :analytics_imports_jobs, %{"event" => "complete", "import_id" => ^import_id}} ->
+    IO.puts("Job completed")
 
-  {:notification, :analytics_imports_jobs, %{"event" => "transient_fail", "import_id" => ^import_id}} ->
-    IO.puts("Job failed transiently")
+  {:notification, :analytics_imports_jobs, %{"event" => "transient_fail", "import_id" => ^import_id}} ->
+    IO.puts("Job failed transiently")
 
-  {:notification, :analytics_imports_jobs, %{"event" => "fail", "import_id" => ^import_id}} ->
-    IO.puts("Job failed permanently")
-after
+  {:notification, :analytics_imports_jobs, %{"event" => "fail", "import_id" => ^import_id}} ->
+    IO.puts("Job failed permanently")
+after
   15_000 ->
-    IO.puts("Job didn't finish in 15 seconds")
-end

In a more realistic scenario, job scheduling will be done inside a GenServer process + IO.puts("Job didn't finish in 15 seconds") +end

In a more realistic scenario, job scheduling will be done inside a GenServer process like LiveView, where notifications can be listened for via handle_info/2.

diff --git a/Plausible.IngestRepo.html b/Plausible.IngestRepo.html index f3fe572e67..391986f716 100644 --- a/Plausible.IngestRepo.html +++ b/Plausible.IngestRepo.html @@ -1217,23 +1217,23 @@ pool to disconnect within the given interval.

See -

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
+

Similar to insert_all/2 but with the following differences:

  • accepts rows as streams or lists
  • sends rows as a chunked request
  • doesn't autogenerate ids or does any other preprocessing

Example:

Repo.query!("create table ecto_ch_demo(a UInt64, b String) engine Null")
 
-defmodule Demo do
+defmodule Demo do
   use Ecto.Schema
 
   @primary_key false
-  schema "ecto_ch_demo" do
+  schema "ecto_ch_demo" do
     field :a, Ch, type: "UInt64"
     field :b, :string
-  end
-end
+  end
+end
 
-rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
-{100_000, nil} = Repo.insert_stream(Demo, rows)
+rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)
+{100_000, nil} = Repo.insert_stream(Demo, rows)
 
 # schemaless
-{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
+
{100_000, nil} = Repo.insert_stream("ecto_ch_demo", rows, types: [a: Ch.Types.u64(), b: :string])
diff --git a/Plausible.S3.html b/Plausible.S3.html index 09ea98af54..24aa3b2685 100644 --- a/Plausible.S3.html +++ b/Plausible.S3.html @@ -316,7 +316,7 @@ The URL expires in 300 seconds, which should be enough for a redirect.

In

Returns the pre-configured S3 bucket for CSV exports.

config :plausible, Plausible.S3,
-  exports_bucket: System.fetch_env!("S3_EXPORTS_BUCKET")

Example:

iex> exports_bucket()
+  exports_bucket: System.fetch_env!("S3_EXPORTS_BUCKET")

Example:

iex> exports_bucket()
 "test-exports"
@@ -348,8 +348,8 @@ The URL expires in 300 seconds, which should be enough for a redirect.

In -

Returns access_key_id and secret_access_key to be used by ClickHouse during imports from S3.

Example:

iex> import_clickhouse_credentials()
-%{access_key_id: "minioadmin", secret_access_key: "minioadmin"}
+

Returns access_key_id and secret_access_key to be used by ClickHouse during imports from S3.

Example:

iex> import_clickhouse_credentials()
+%{access_key_id: "minioadmin", secret_access_key: "minioadmin"}
@@ -371,9 +371,9 @@ The URL expires in 300 seconds, which should be enough for a redirect.

In

-

Presigns an upload for an imported file.

In the current implementation the bucket always goes into the path component.

Example:

iex> upload = import_presign_upload(_site_id = 123, _filename = "imported_browsers.csv")
-iex> true = String.ends_with?(upload.s3_url, "/test-imports/123/imported_browsers.csv")
-iex> true = String.contains?(upload.presigned_url, "/test-imports/123/imported_browsers.csv?X-Amz-Algorithm=AWS4-HMAC-SHA256&")
+

Presigns an upload for an imported file.

In the current implementation the bucket always goes into the path component.

Example:

iex> upload = import_presign_upload(_site_id = 123, _filename = "imported_browsers.csv")
+iex> true = String.ends_with?(upload.s3_url, "/test-imports/123/imported_browsers.csv")
+iex> true = String.contains?(upload.presigned_url, "/test-imports/123/imported_browsers.csv?X-Amz-Algorithm=AWS4-HMAC-SHA256&")
@@ -402,7 +402,7 @@ The URL expires in 300 seconds, which should be enough for a redirect.

In

Returns the pre-configured S3 bucket for CSV imports.

config :plausible, Plausible.S3,
-  imports_bucket: System.fetch_env!("S3_IMPORTS_BUCKET")

Example:

iex> imports_bucket()
+  imports_bucket: System.fetch_env!("S3_IMPORTS_BUCKET")

Example:

iex> imports_bucket()
 "test-imports"
diff --git a/Plausible.Stats.Filters.html b/Plausible.Stats.Filters.html index eb67f30c9c..0c318c28ae 100644 --- a/Plausible.Stats.Filters.html +++ b/Plausible.Stats.Filters.html @@ -283,14 +283,14 @@ Pages Examples:

-
iex> Filters.parse("{\"page\":\"/blog/**\"}")
-[[:matches, "event:page", ["/blog/**"]]]
+
iex> Filters.parse("{\"page\":\"/blog/**\"}")
+[[:matches, "event:page", ["/blog/**"]]]
 
-iex> Filters.parse("visit:browser!=Chrome")
-[[:is_not, "visit:browser", ["Chrome"]]]
+iex> Filters.parse("visit:browser!=Chrome")
+[[:is_not, "visit:browser", ["Chrome"]]]
 
-iex> Filters.parse(nil)
-[]
+
iex> Filters.parse(nil) +[]
diff --git a/Plausible.Stats.SQL.Fragments.html b/Plausible.Stats.SQL.Fragments.html index c724c58f90..7bd4e8ff3e 100644 --- a/Plausible.Stats.SQL.Fragments.html +++ b/Plausible.Stats.SQL.Fragments.html @@ -668,12 +668,12 @@ boundary, not_before is returned.

not_before boundary is set to the past Saturday, which is before the weekstart, therefore the cap does not apply.

  > this_wednesday = ~D[2022-11-09]
   > past_saturday = ~D[2022-11-05]
-  > weekstart_not_before(this_wednesday, past_saturday)
+  > weekstart_not_before(this_wednesday, past_saturday)
   ~D[2022-11-07]

In this other example, the fragment returns Tuesday and not the weekstart. The not_before boundary is set to Tuesday, which is past the weekstart, therefore the cap applies.

  > this_wednesday = ~D[2022-11-09]
   > this_tuesday = ~D[2022-11-08]
-  > weekstart_not_before(this_wednesday, this_tuesday)
+  > weekstart_not_before(this_wednesday, this_tuesday)
   ~D[2022-11-08]

diff --git a/Plausible.epub b/Plausible.epub index c13be939d618b66be9967ef10187299209b3df25..ca21b841a5b68e2f661a20a52a949ec9d6f059dc 100644 GIT binary patch delta 145338 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�@TpmrPkey`rg8|Jw8-p+A?N|UG}y61Ci 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 QRn*ofg{UK(5D|_358T{zmH+?% delta 144955 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 NP$udkUD3G3{{bDJyvP6m diff --git a/PlausibleWeb.Api.StatsController.html b/PlausibleWeb.Api.StatsController.html index c3b1bed6aa..078a43643d 100644 --- a/PlausibleWeb.Api.StatsController.html +++ b/PlausibleWeb.Api.StatsController.html @@ -785,13 +785,13 @@ and month. The default depends on the Plausible.Query.from/2 for each default.

  • with_imported - boolean indicating whether to include Google Analytics -imported data or not. Defaults to false.

  • Full example:

    %{
    +imported data or not. Defaults to false.

    Full example:

    %{
       "from" => "2021-09-06",
       "interval" => "month",
       "metric" => "visitors",
       "period" => "custom",
       "to" => "2021-12-13"
    -}

    +}

    @@ -806,20 +806,20 @@ cut off by the requested date range or not. For example, if looking at a month week-by-week, some weeks may be cut off by the month boundaries. It's useful to adjust the graph display slightly in case the interval is not 'full' so that the user understands why the numbers might be lower for -those partial periods.

    Full example:

    %{
    -  "full_intervals" => %{
    +those partial periods.

    Full example:

    %{
    +  "full_intervals" => %{
         "2021-09-01" => false,
         "2021-10-01" => true,
         "2021-11-01" => true,
         "2021-12-01" => false
    -  },
    +  },
       "imports_exist" => false,
       "interval" => "month",
    -  "labels" => ["2021-09-01", "2021-10-01", "2021-11-01", "2021-12-01"],
    -  "plot" => [0, 0, 0, 0],
    +  "labels" => ["2021-09-01", "2021-10-01", "2021-11-01", "2021-12-01"],
    +  "plot" => [0, 0, 0, 0],
       "present_index" => nil,
       "includes_imported" => false
    -}
    +
    }
    diff --git a/PlausibleWeb.Live.Components.Modal.html b/PlausibleWeb.Live.Components.Modal.html index 7a619e2c74..a3ab69acea 100644 --- a/PlausibleWeb.Live.Components.Modal.html +++ b/PlausibleWeb.Live.Components.Modal.html @@ -153,18 +153,18 @@ on connections with high latency.

    Usage

    An example use case for a modal is embedding a form inside -existing live view which allows adding new entries of some kind:

    <.live_component module={Modal} id="some-form-modal" :let={modal_unique_id}>
    +existing live view which allows adding new entries of some kind:

    <.live_component module={Modal} id="some-form-modal" :let={modal_unique_id}>
       <.live_component
    -    module={SomeForm}
    -    id={"some-form-#{modal_unique_id}"}
    -    on_save_form={
    -      fn entry, socket ->
    -        send(self(), {:entry_added, entry})
    -        Modal.close(socket, "some-form-modal")
    -      end
    -    }
    +    module={SomeForm}
    +    id={"some-form-#{modal_unique_id}"}
    +    on_save_form={
    +      fn entry, socket ->
    +        send(self(), {:entry_added, entry})
    +        Modal.close(socket, "some-form-modal")
    +      end
    +    }
       />
    -</.live_component>

    Then somewhere in the same live view the modal is rendered in:

    <.button x-data x-on:click={Modal.JS.open("goals-form-modal")}>
    +</.live_component>

    Then somewhere in the same live view the modal is rendered in:

    <.button x-data x-on:click={Modal.JS.open("goals-form-modal")}>
       + Add Entry
     </.button>

    @@ -199,26 +199,26 @@ and providing the state. Should be used together with Modal.JS.preopen/1 for optimal user experience.

  • Modal.close/2 - to close the modal from the backend; usually done inside wrapped component's handle_event/2. The example quoted above shows one way to implement this, under that assumption -that the component exposes a callback, like this:

    defmodule SomeForm do
    +that the component exposes a callback, like this:

    defmodule SomeForm do
       use Phoenix.LiveComponent
     
    -  def update(assigns, socket) do
    +  def update(assigns, socket) do
         # ...
     
    -    {:ok, assign(socket, :on_save_form, assigns.on_save_form)}
    -  end
    +    {:ok, assign(socket, :on_save_form, assigns.on_save_form)}
    +  end
     
       #...
     
    -  def handle_event("save-form", %{"form" => form}, socket) do
    -    case save_entry(form) do
    -      {:ok, entry} ->
    -        {:noreply, socket.assigns.on_save_form(entry, socket)}
    +  def handle_event("save-form", %{"form" => form}, socket) do
    +    case save_entry(form) do
    +      {:ok, entry} ->
    +        {:noreply, socket.assigns.on_save_form(entry, socket)}
     
           # error case handling ...
    -    end
    -  end
    -end

    Using callback approach has an added benefit of making the + end + end +end

    Using callback approach has an added benefit of making the component more flexible.

  • diff --git a/PlausibleWeb.Plugs.AuthorizePublicAPI.html b/PlausibleWeb.Plugs.AuthorizePublicAPI.html index 1a3e5c8339..8f580aa1e8 100644 --- a/PlausibleWeb.Plugs.AuthorizePublicAPI.html +++ b/PlausibleWeb.Plugs.AuthorizePublicAPI.html @@ -145,12 +145,12 @@ Pages

    Plug for authorizing access to Stats and Sites APIs.

    The plug expects :api_scope to be provided in the assigns. The scope will then be used to check for API key validity. The assign can be -provided in the router configuration in a following way:

    scope "/api/v1/stats", PlausibleWeb.Api, assigns: %{api_scope: "some:scope:*"} do
    -  pipe_through [:public_api, PlausibleWeb.Plugs.AuthorizePublicAPI]
    +provided in the router configuration in a following way:

    scope "/api/v1/stats", PlausibleWeb.Api, assigns: %{api_scope: "some:scope:*"} do
    +  pipe_through [:public_api, PlausibleWeb.Plugs.AuthorizePublicAPI]
     
       # route definitions follow
       # ...
    -end

    The scope from :api_scope is checked for match against all scopes from API key's +end

    The scope from :api_scope is checked for match against all scopes from API key's scopes field. If the scope is among @implicit_scopes, it's considered to be present for any valid API key. Scopes are checked for match by prefix, so if we have some:scope:* in matching route :api_scope and the API key has some:* in its diff --git a/dist/search_data-25912140.js b/dist/search_data-25912140.js new file mode 100644 index 0000000000..82427237c4 --- /dev/null +++ b/dist/search_data-25912140.js @@ -0,0 +1 @@ +searchData={"items":[{"type":"task","doc":"This task is meant to replicate the behavior of cancelling\na subscription. On production, this action is initiated by\na Paddle webhook. Currently, only the subscription status\nis changed with that action.","title":"mix cancel_subscription","ref":"Mix.Tasks.CancelSubscription.html"},{"type":"function","doc":"","title":"Mix.Tasks.CancelSubscription.run/1","ref":"Mix.Tasks.CancelSubscription.html#run/1"},{"type":"task","doc":"","title":"mix clean_clickhouse","ref":"Mix.Tasks.CleanClickhouse.html"},{"type":"function","doc":"","title":"Mix.Tasks.CleanClickhouse.run/1","ref":"Mix.Tasks.CleanClickhouse.html#run/1"},{"type":"task","doc":"","title":"mix create_free_subscription","ref":"Mix.Tasks.CreateFreeSubscription.html"},{"type":"function","doc":"","title":"Mix.Tasks.CreateFreeSubscription.execute/1","ref":"Mix.Tasks.CreateFreeSubscription.html#execute/1"},{"type":"function","doc":"","title":"Mix.Tasks.CreateFreeSubscription.run/1","ref":"Mix.Tasks.CreateFreeSubscription.html#run/1"},{"type":"task","doc":"This task downloads the Country Lite database from DB-IP for self-hosted or development purposes.\nPlausible Cloud runs a paid version of DB-IP with more detailed geolocation data.","title":"mix download_country_database","ref":"Mix.Tasks.DownloadCountryDatabase.html"},{"type":"function","doc":"","title":"Mix.Tasks.DownloadCountryDatabase.run/1","ref":"Mix.Tasks.DownloadCountryDatabase.html#run/1"},{"type":"task","doc":"","title":"mix generate_referrer_favicons","ref":"Mix.Tasks.GenerateReferrerFavicons.html"},{"type":"function","doc":"","title":"Mix.Tasks.GenerateReferrerFavicons.run/1","ref":"Mix.Tasks.GenerateReferrerFavicons.html#run/1"},{"type":"task","doc":"","title":"mix pull_sandbox_subscription","ref":"Mix.Tasks.PullSandboxSubscription.html"},{"type":"function","doc":"","title":"Mix.Tasks.PullSandboxSubscription.run/1","ref":"Mix.Tasks.PullSandboxSubscription.html#run/1"},{"type":"task","doc":"It's often necessary to generate fake events for development and testing purposes. This Mix Task provides a quick and easy\nway to generate a pageview or custom event, either in your development environment or a remote Plausible instance.\n\nSee Mix.Tasks.SendPageview.usage/1 for more detailed documentation.","title":"mix send_pageview","ref":"Mix.Tasks.SendPageview.html"},{"type":"function","doc":"","title":"Mix.Tasks.SendPageview.run/1","ref":"Mix.Tasks.SendPageview.html#run/1"},{"type":"module","doc":"","title":"ObanErrorReporter","ref":"ObanErrorReporter.html"},{"type":"function","doc":"","title":"ObanErrorReporter.handle_event/4","ref":"ObanErrorReporter.html#handle_event/4"},{"type":"module","doc":"Build-related macros","title":"Plausible","ref":"Plausible.html"},{"type":"function","doc":"","title":"Plausible.ce?/0","ref":"Plausible.html#ce?/0"},{"type":"function","doc":"","title":"Plausible.ee?/0","ref":"Plausible.html#ee?/0"},{"type":"macro","doc":"","title":"Plausible.on_ce/1","ref":"Plausible.html#on_ce/1"},{"type":"macro","doc":"","title":"Plausible.on_ee/1","ref":"Plausible.html#on_ee/1"},{"type":"function","doc":"","title":"Plausible.product_name/0","ref":"Plausible.html#product_name/0"},{"type":"module","doc":"Clickhouse access with async inserts enabled","title":"Plausible.AsyncInsertRepo","ref":"Plausible.AsyncInsertRepo.html"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.aggregate/3","ref":"Plausible.AsyncInsertRepo.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.aggregate/4","ref":"Plausible.AsyncInsertRepo.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.all/2","ref":"Plausible.AsyncInsertRepo.html#all/2"},{"type":"function","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","title":"Plausible.AsyncInsertRepo.alter_update_all/3","ref":"Plausible.AsyncInsertRepo.html#alter_update_all/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.checked_out?/0","ref":"Plausible.AsyncInsertRepo.html#checked_out?/0"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.checkout/2","ref":"Plausible.AsyncInsertRepo.html#checkout/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.child_spec/1","ref":"Plausible.AsyncInsertRepo.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.config/0","ref":"Plausible.AsyncInsertRepo.html#config/0"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.default_options/1","ref":"Plausible.AsyncInsertRepo.html#default_options/1"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.delete/2","ref":"Plausible.AsyncInsertRepo.html#delete/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.delete!/2","ref":"Plausible.AsyncInsertRepo.html#delete!/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.delete_all/2","ref":"Plausible.AsyncInsertRepo.html#delete_all/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","title":"Plausible.AsyncInsertRepo.disconnect_all/2","ref":"Plausible.AsyncInsertRepo.html#disconnect_all/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.exists?/2","ref":"Plausible.AsyncInsertRepo.html#exists?/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.get/3","ref":"Plausible.AsyncInsertRepo.html#get/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.get!/3","ref":"Plausible.AsyncInsertRepo.html#get!/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.get_by/3","ref":"Plausible.AsyncInsertRepo.html#get_by/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.get_by!/3","ref":"Plausible.AsyncInsertRepo.html#get_by!/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.get_dynamic_repo/0","ref":"Plausible.AsyncInsertRepo.html#get_dynamic_repo/0"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.insert/2","ref":"Plausible.AsyncInsertRepo.html#insert/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.insert!/2","ref":"Plausible.AsyncInsertRepo.html#insert!/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.insert_all/3","ref":"Plausible.AsyncInsertRepo.html#insert_all/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.insert_or_update/2","ref":"Plausible.AsyncInsertRepo.html#insert_or_update/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.insert_or_update!/2","ref":"Plausible.AsyncInsertRepo.html#insert_or_update!/2"},{"type":"function","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","title":"Plausible.AsyncInsertRepo.insert_stream/3","ref":"Plausible.AsyncInsertRepo.html#insert_stream/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.load/2","ref":"Plausible.AsyncInsertRepo.html#load/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.one/2","ref":"Plausible.AsyncInsertRepo.html#one/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.one!/2","ref":"Plausible.AsyncInsertRepo.html#one!/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.preload/3","ref":"Plausible.AsyncInsertRepo.html#preload/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.prepare_query/3","ref":"Plausible.AsyncInsertRepo.html#prepare_query/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.put_dynamic_repo/1","ref":"Plausible.AsyncInsertRepo.html#put_dynamic_repo/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","title":"Plausible.AsyncInsertRepo.query/3","ref":"Plausible.AsyncInsertRepo.html#query/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","title":"Plausible.AsyncInsertRepo.query!/3","ref":"Plausible.AsyncInsertRepo.html#query!/3"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.reload/2","ref":"Plausible.AsyncInsertRepo.html#reload/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.reload!/2","ref":"Plausible.AsyncInsertRepo.html#reload!/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.start_link/1","ref":"Plausible.AsyncInsertRepo.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.stop/1","ref":"Plausible.AsyncInsertRepo.html#stop/1"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.stream/2","ref":"Plausible.AsyncInsertRepo.html#stream/2"},{"type":"function","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","title":"Plausible.AsyncInsertRepo.to_inline_sql/2","ref":"Plausible.AsyncInsertRepo.html#to_inline_sql/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","title":"Plausible.AsyncInsertRepo.to_sql/2","ref":"Plausible.AsyncInsertRepo.html#to_sql/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.update/2","ref":"Plausible.AsyncInsertRepo.html#update/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.update!/2","ref":"Plausible.AsyncInsertRepo.html#update!/2"},{"type":"function","doc":"","title":"Plausible.AsyncInsertRepo.update_all/3","ref":"Plausible.AsyncInsertRepo.html#update_all/3"},{"type":"module","doc":"Functions for user authentication context.","title":"Plausible.Auth","ref":"Plausible.Auth.html"},{"type":"function","doc":"","title":"Plausible.Auth.check_password/2","ref":"Plausible.Auth.html#check_password/2"},{"type":"function","doc":"","title":"Plausible.Auth.create_api_key/3","ref":"Plausible.Auth.html#create_api_key/3"},{"type":"function","doc":"","title":"Plausible.Auth.create_user/3","ref":"Plausible.Auth.html#create_user/3"},{"type":"function","doc":"","title":"Plausible.Auth.delete_api_key/2","ref":"Plausible.Auth.html#delete_api_key/2"},{"type":"function","doc":"","title":"Plausible.Auth.delete_user/1","ref":"Plausible.Auth.html#delete_user/1"},{"type":"function","doc":"","title":"Plausible.Auth.enterprise_configured?/1","ref":"Plausible.Auth.html#enterprise_configured?/1"},{"type":"function","doc":"","title":"Plausible.Auth.find_api_key/1","ref":"Plausible.Auth.html#find_api_key/1"},{"type":"function","doc":"","title":"Plausible.Auth.find_user_by/1","ref":"Plausible.Auth.html#find_user_by/1"},{"type":"function","doc":"","title":"Plausible.Auth.get_user_by/1","ref":"Plausible.Auth.html#get_user_by/1"},{"type":"function","doc":"","title":"Plausible.Auth.has_active_sites?/2","ref":"Plausible.Auth.html#has_active_sites?/2"},{"type":"function","doc":"","title":"Plausible.Auth.is_super_admin?/1","ref":"Plausible.Auth.html#is_super_admin?/1"},{"type":"function","doc":"","title":"Plausible.Auth.rate_limit/2","ref":"Plausible.Auth.html#rate_limit/2"},{"type":"function","doc":"","title":"Plausible.Auth.rate_limits/0","ref":"Plausible.Auth.html#rate_limits/0"},{"type":"function","doc":"","title":"Plausible.Auth.user_owns_sites?/1","ref":"Plausible.Auth.html#user_owns_sites?/1"},{"type":"type","doc":"","title":"Plausible.Auth.rate_limit_type/0","ref":"Plausible.Auth.html#t:rate_limit_type/0"},{"type":"module","doc":"","title":"Plausible.Auth.ApiKey","ref":"Plausible.Auth.ApiKey.html"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKey.changeset/2","ref":"Plausible.Auth.ApiKey.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKey.do_hash/1","ref":"Plausible.Auth.ApiKey.html#do_hash/1"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKey.process_key/1","ref":"Plausible.Auth.ApiKey.html#process_key/1"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKey.update/2","ref":"Plausible.Auth.ApiKey.html#update/2"},{"type":"type","doc":"","title":"Plausible.Auth.ApiKey.t/0","ref":"Plausible.Auth.ApiKey.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Auth.ApiKeyAdmin","ref":"Plausible.Auth.ApiKeyAdmin.html"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKeyAdmin.create_changeset/2","ref":"Plausible.Auth.ApiKeyAdmin.html#create_changeset/2"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKeyAdmin.custom_index_query/3","ref":"Plausible.Auth.ApiKeyAdmin.html#custom_index_query/3"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKeyAdmin.form_fields/1","ref":"Plausible.Auth.ApiKeyAdmin.html#form_fields/1"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKeyAdmin.index/1","ref":"Plausible.Auth.ApiKeyAdmin.html#index/1"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKeyAdmin.search_fields/1","ref":"Plausible.Auth.ApiKeyAdmin.html#search_fields/1"},{"type":"function","doc":"","title":"Plausible.Auth.ApiKeyAdmin.update_changeset/2","ref":"Plausible.Auth.ApiKeyAdmin.html#update_changeset/2"},{"type":"module","doc":"Schema for email activation codes.","title":"Plausible.Auth.EmailActivationCode","ref":"Plausible.Auth.EmailActivationCode.html"},{"type":"function","doc":"","title":"Plausible.Auth.EmailActivationCode.generate_code/0","ref":"Plausible.Auth.EmailActivationCode.html#generate_code/0"},{"type":"function","doc":"","title":"Plausible.Auth.EmailActivationCode.new/2","ref":"Plausible.Auth.EmailActivationCode.html#new/2"},{"type":"type","doc":"","title":"Plausible.Auth.EmailActivationCode.t/0","ref":"Plausible.Auth.EmailActivationCode.html#t:t/0"},{"type":"module","doc":"API for verifying emails.","title":"Plausible.Auth.EmailVerification","ref":"Plausible.Auth.EmailVerification.html"},{"type":"function","doc":"","title":"Plausible.Auth.EmailVerification.any?/1","ref":"Plausible.Auth.EmailVerification.html#any?/1"},{"type":"function","doc":"","title":"Plausible.Auth.EmailVerification.expired?/1","ref":"Plausible.Auth.EmailVerification.html#expired?/1"},{"type":"function","doc":"","title":"Plausible.Auth.EmailVerification.issue_code/2","ref":"Plausible.Auth.EmailVerification.html#issue_code/2"},{"type":"function","doc":"","title":"Plausible.Auth.EmailVerification.verify_code/2","ref":"Plausible.Auth.EmailVerification.html#verify_code/2"},{"type":"module","doc":"This embedded schema stores information about the account locking grace\nperiod.\n\nUsers are given this 7-day grace period to upgrade their account after\noutgrowing their subscriptions. The actual account locking happens in\nbackground with `Plausible.Workers.LockSites`.\n\nThe grace period can also be manual, without an end date, being controlled\nmanually from the CRM, and not by the background site locker job. This is\nuseful for enterprise subscriptions.","title":"Plausible.Auth.GracePeriod","ref":"Plausible.Auth.GracePeriod.html"},{"type":"function","doc":"Returns whether the grace period is still active for a User. Defaults to\nfalse if the user is nil or there is no grace period.","title":"Plausible.Auth.GracePeriod.active?/1","ref":"Plausible.Auth.GracePeriod.html#active?/1"},{"type":"function","doc":"Ends an existing grace period by `setting users.grace_period.is_over` to true.\nThis means the grace period has expired.","title":"Plausible.Auth.GracePeriod.end_changeset/1","ref":"Plausible.Auth.GracePeriod.html#end_changeset/1"},{"type":"function","doc":"Returns whether the grace period has already expired for a User. Defaults to\nfalse if the user is nil or there is no grace period.","title":"Plausible.Auth.GracePeriod.expired?/1","ref":"Plausible.Auth.GracePeriod.html#expired?/1"},{"type":"function","doc":"Removes the grace period from the User completely.","title":"Plausible.Auth.GracePeriod.remove_changeset/1","ref":"Plausible.Auth.GracePeriod.html#remove_changeset/1"},{"type":"function","doc":"Starts a account locking grace period of 7 days by changing the User struct.","title":"Plausible.Auth.GracePeriod.start_changeset/1","ref":"Plausible.Auth.GracePeriod.html#start_changeset/1"},{"type":"function","doc":"Starts a manual account locking grace period by changing the User struct.\nManual locking means the grace period can only be removed manually from the\nCRM.","title":"Plausible.Auth.GracePeriod.start_manual_lock_changeset/1","ref":"Plausible.Auth.GracePeriod.html#start_manual_lock_changeset/1"},{"type":"type","doc":"","title":"Plausible.Auth.GracePeriod.t/0","ref":"Plausible.Auth.GracePeriod.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Auth.Invitation","ref":"Plausible.Auth.Invitation.html"},{"type":"function","doc":"","title":"Plausible.Auth.Invitation.new/1","ref":"Plausible.Auth.Invitation.html#new/1"},{"type":"type","doc":"","title":"Plausible.Auth.Invitation.t/0","ref":"Plausible.Auth.Invitation.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Auth.Password","ref":"Plausible.Auth.Password.html"},{"type":"function","doc":"","title":"Plausible.Auth.Password.dummy_calculation/0","ref":"Plausible.Auth.Password.html#dummy_calculation/0"},{"type":"function","doc":"","title":"Plausible.Auth.Password.hash/1","ref":"Plausible.Auth.Password.html#hash/1"},{"type":"function","doc":"","title":"Plausible.Auth.Password.match?/2","ref":"Plausible.Auth.Password.html#match?/2"},{"type":"module","doc":"TOTP auth context\n\nHandles all the aspects of TOTP setup, management and validation for users.","title":"Plausible.Auth.TOTP","ref":"Plausible.Auth.TOTP.html"},{"type":"module","doc":"TOTP setup is started with `initiate/1`. At this stage, a random secret\nbinary is generated for user and stored under `User.totp_secret`. The secret\nis additionally encrypted while stored in the database using `Cloak`. The\nvault for safe storage is configured in `Plausible.Auth.TOTP.Vault` via\na dedicated `Ecto` type defined in `Plausible.Auth.TOTP.EncryptedBinary`.\nThe function returns updated user along with TOTP URI and a readable form\nof secret. Both - the URI and readable secret - are meant for exposure\nin the user's setup screen. The URI should be encoded as a QR code.\n\nAfter initiation, user is expected to confirm valid setup with `enable/2`,\nproviding TOTP code from their authenticator app. After code validation\npasses successfully, the `User.totp_enabled` flag is set to `true`.\nFinally, the user must be immediately presented with a list of recovery codes\nreturned by the same call of `enable/2`. The codes should be presented\nin copy/paste friendly form, ideally also with a print-friendly view option.\n\nThe `initiate/1` and `enable/1` functions can be safely called multiple\ntimes, allowing user to abort and restart setup up to these stages.","title":"Setup - Plausible.Auth.TOTP","ref":"Plausible.Auth.TOTP.html#module-setup"},{"type":"module","doc":"The state of TOTP for a particular user can be chcecked by calling\n`enabled?/1` or `initiated?/1`.\n\nTOTP can be disabled with `disable/2`. User is expected to provide their\ncurrent password for safety. Once disabled, all TOTP user settings are\ncleared and any remaining generated recovery codes are removed. The function\ncan be safely run more than once. There's also alternative call for forced\ndisabling of TOTP for a given user without sending any notification,\n`force_disable/1`. It's meant for use in situation where user lost both,\n2FA device and recovery codes and their identity is verified independently.\n\nIf the user needs to regenerate the recovery codes outside of setup procedure,\nthey must do it via `generate_recovery_codes/2`, providing their current\npassword for safety. They must be warned that any existing recovery codes\nwill be invalidated.","title":"Management - Plausible.Auth.TOTP","ref":"Plausible.Auth.TOTP.html#module-management"},{"type":"module","doc":"After logging in, user's TOTP state must be checked with `enabled?/1`.\n\nIf enabled, user must be presented with TOTP code input form accepting\n6 digit characters. The code must be checked using `validate_code/2`.\n\nUser must have an option to alternatively input one of their recovery\ncodes. Those codes must be checked with `use_recovery_code/2`.","title":"Validation - Plausible.Auth.TOTP","ref":"Plausible.Auth.TOTP.html#module-validation"},{"type":"module","doc":"In case of TOTP codes, a grace period of 30 seconds is applied, which\nallows user to use their current and previous TOTP code, assuming 30\nsecond validity window of each. This allows user to use code that was\nabout to expire before the submission. Regardless of that, each TOTP\ncode can be used only once. Validation procedure rejects repeat use\nof the same code for safety. It's done by tracking last time a TOTP\ncode was used successfully, stored under `User.totp_last_used_at`.\n\nIn case of recovery codes, each code is deleted immediately after use.\nThey are strictly one-time use only.","title":"Code validity - Plausible.Auth.TOTP","ref":"Plausible.Auth.TOTP.html#module-code-validity"},{"type":"module","doc":"TOTP token is an alternate method of authenticating user session.\nIt's main use case is \"trust this device\" functionality, where user\ncan decide to skip 2FA verification for a particular browser session\nfor next N days. The token should then be stored in an encrypted,\nsigned cookie with a proper expiration timestamp.\n\nThe token should be reset each time it either fails to match\nor when other credentials (like password) are reset. This should\neffectively invalidate all trusted devices for a given user.","title":"TOTP Token - Plausible.Auth.TOTP","ref":"Plausible.Auth.TOTP.html#module-totp-token"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.disable/2","ref":"Plausible.Auth.TOTP.html#disable/2"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.enable/3","ref":"Plausible.Auth.TOTP.html#enable/3"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.enabled?/1","ref":"Plausible.Auth.TOTP.html#enabled?/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.force_disable/1","ref":"Plausible.Auth.TOTP.html#force_disable/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.generate_recovery_codes/1","ref":"Plausible.Auth.TOTP.html#generate_recovery_codes/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.generate_recovery_codes/2","ref":"Plausible.Auth.TOTP.html#generate_recovery_codes/2"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.initiate/1","ref":"Plausible.Auth.TOTP.html#initiate/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.initiated?/1","ref":"Plausible.Auth.TOTP.html#initiated?/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.reset_token/1","ref":"Plausible.Auth.TOTP.html#reset_token/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.use_recovery_code/2","ref":"Plausible.Auth.TOTP.html#use_recovery_code/2"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.validate_code/3","ref":"Plausible.Auth.TOTP.html#validate_code/3"},{"type":"module","doc":"Defines an Ecto type so Cloak.Ecto can encrypt/decrypt a binary field.","title":"Plausible.Auth.TOTP.EncryptedBinary","ref":"Plausible.Auth.TOTP.EncryptedBinary.html"},{"type":"module","doc":"Schema for TOTP recovery codes.","title":"Plausible.Auth.TOTP.RecoveryCode","ref":"Plausible.Auth.TOTP.RecoveryCode.html"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.RecoveryCode.changeset/2","ref":"Plausible.Auth.TOTP.RecoveryCode.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.RecoveryCode.changeset_to_map/2","ref":"Plausible.Auth.TOTP.RecoveryCode.html#changeset_to_map/2"},{"type":"function","doc":"Generates `count` unique recovery codes, each alphanumeric\nand 10 characters long.","title":"Plausible.Auth.TOTP.RecoveryCode.generate_codes/1","ref":"Plausible.Auth.TOTP.RecoveryCode.html#generate_codes/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.RecoveryCode.match?/2","ref":"Plausible.Auth.TOTP.RecoveryCode.html#match?/2"},{"type":"type","doc":"","title":"Plausible.Auth.TOTP.RecoveryCode.t/0","ref":"Plausible.Auth.TOTP.RecoveryCode.html#t:t/0"},{"type":"module","doc":"Provides a vault that will be used to encrypt/decrypt the TOTP secrets of users who enable it.","title":"Plausible.Auth.TOTP.Vault","ref":"Plausible.Auth.TOTP.Vault.html"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.Auth.TOTP.Vault.child_spec/1","ref":"Plausible.Auth.TOTP.Vault.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Auth.TOTP.Vault.start_link/1","ref":"Plausible.Auth.TOTP.Vault.html#start_link/1"},{"type":"module","doc":"","title":"Plausible.Auth.Token","ref":"Plausible.Auth.Token.html"},{"type":"function","doc":"","title":"Plausible.Auth.Token.sign_password_reset/1","ref":"Plausible.Auth.Token.html#sign_password_reset/1"},{"type":"function","doc":"","title":"Plausible.Auth.Token.sign_shared_link/1","ref":"Plausible.Auth.Token.html#sign_shared_link/1"},{"type":"function","doc":"","title":"Plausible.Auth.Token.verify_password_reset/1","ref":"Plausible.Auth.Token.html#verify_password_reset/1"},{"type":"function","doc":"","title":"Plausible.Auth.Token.verify_shared_link/1","ref":"Plausible.Auth.Token.html#verify_shared_link/1"},{"type":"module","doc":"","title":"Plausible.Auth.User","ref":"Plausible.Auth.User.html"},{"type":"function","doc":"","title":"Plausible.Auth.User.cancel_email_changeset/1","ref":"Plausible.Auth.User.html#cancel_email_changeset/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.changeset/2","ref":"Plausible.Auth.User.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Auth.User.email_changeset/2","ref":"Plausible.Auth.User.html#email_changeset/2"},{"type":"function","doc":"","title":"Plausible.Auth.User.end_trial/1","ref":"Plausible.Auth.User.html#end_trial/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.hash_password/1","ref":"Plausible.Auth.User.html#hash_password/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.new/1","ref":"Plausible.Auth.User.html#new/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.password_strength/1","ref":"Plausible.Auth.User.html#password_strength/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.profile_img_url/1","ref":"Plausible.Auth.User.html#profile_img_url/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.remove_trial_expiry/1","ref":"Plausible.Auth.User.html#remove_trial_expiry/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.set_password/2","ref":"Plausible.Auth.User.html#set_password/2"},{"type":"function","doc":"","title":"Plausible.Auth.User.settings_changeset/2","ref":"Plausible.Auth.User.html#settings_changeset/2"},{"type":"function","doc":"","title":"Plausible.Auth.User.start_trial/1","ref":"Plausible.Auth.User.html#start_trial/1"},{"type":"function","doc":"","title":"Plausible.Auth.User.subscription_accept_traffic_until_offset_days/0","ref":"Plausible.Auth.User.html#subscription_accept_traffic_until_offset_days/0"},{"type":"function","doc":"","title":"Plausible.Auth.User.trial_accept_traffic_until_offset_days/0","ref":"Plausible.Auth.User.html#trial_accept_traffic_until_offset_days/0"},{"type":"type","doc":"","title":"Plausible.Auth.User.t/0","ref":"Plausible.Auth.User.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Auth.UserAdmin","ref":"Plausible.Auth.UserAdmin.html"},{"type":"function","doc":"","title":"Plausible.Auth.UserAdmin.custom_index_query/3","ref":"Plausible.Auth.UserAdmin.html#custom_index_query/3"},{"type":"function","doc":"","title":"Plausible.Auth.UserAdmin.delete/2","ref":"Plausible.Auth.UserAdmin.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Auth.UserAdmin.disable_2fa/1","ref":"Plausible.Auth.UserAdmin.html#disable_2fa/1"},{"type":"function","doc":"","title":"Plausible.Auth.UserAdmin.form_fields/1","ref":"Plausible.Auth.UserAdmin.html#form_fields/1"},{"type":"function","doc":"","title":"Plausible.Auth.UserAdmin.index/1","ref":"Plausible.Auth.UserAdmin.html#index/1"},{"type":"function","doc":"","title":"Plausible.Auth.UserAdmin.resource_actions/1","ref":"Plausible.Auth.UserAdmin.html#resource_actions/1"},{"type":"module","doc":"Schema for storing user session data.","title":"Plausible.Auth.UserSession","ref":"Plausible.Auth.UserSession.html"},{"type":"function","doc":"","title":"Plausible.Auth.UserSession.new_session/3","ref":"Plausible.Auth.UserSession.html#new_session/3"},{"type":"function","doc":"","title":"Plausible.Auth.UserSession.timeout_duration/0","ref":"Plausible.Auth.UserSession.html#timeout_duration/0"},{"type":"function","doc":"","title":"Plausible.Auth.UserSession.touch_session/2","ref":"Plausible.Auth.UserSession.html#touch_session/2"},{"type":"type","doc":"","title":"Plausible.Auth.UserSession.t/0","ref":"Plausible.Auth.UserSession.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Billing","ref":"Plausible.Billing.html"},{"type":"function","doc":"","title":"Plausible.Billing.active_subscription_for/1","ref":"Plausible.Billing.html#active_subscription_for/1"},{"type":"function","doc":"","title":"Plausible.Billing.cancelled_subscription_notice_dismiss_id/1","ref":"Plausible.Billing.html#cancelled_subscription_notice_dismiss_id/1"},{"type":"function","doc":"","title":"Plausible.Billing.change_plan/2","ref":"Plausible.Billing.html#change_plan/2"},{"type":"function","doc":"","title":"Plausible.Billing.change_plan_preview/2","ref":"Plausible.Billing.html#change_plan_preview/2"},{"type":"function","doc":"","title":"Plausible.Billing.check_needs_to_upgrade/1","ref":"Plausible.Billing.html#check_needs_to_upgrade/1"},{"type":"function","doc":"","title":"Plausible.Billing.format_price/1","ref":"Plausible.Billing.html#format_price/1"},{"type":"function","doc":"","title":"Plausible.Billing.has_active_subscription?/1","ref":"Plausible.Billing.html#has_active_subscription?/1"},{"type":"function","doc":"","title":"Plausible.Billing.paddle_api/0","ref":"Plausible.Billing.html#paddle_api/0"},{"type":"function","doc":"","title":"Plausible.Billing.subscription_cancelled/1","ref":"Plausible.Billing.html#subscription_cancelled/1"},{"type":"function","doc":"","title":"Plausible.Billing.subscription_created/1","ref":"Plausible.Billing.html#subscription_created/1"},{"type":"function","doc":"","title":"Plausible.Billing.subscription_payment_succeeded/1","ref":"Plausible.Billing.html#subscription_payment_succeeded/1"},{"type":"function","doc":"","title":"Plausible.Billing.subscription_updated/1","ref":"Plausible.Billing.html#subscription_updated/1"},{"type":"module","doc":"Ecto type representing a feature. Features are cast and stored in the\ndatabase as strings and loaded as modules, for example: `\"props\"` is loaded\nas `Plausible.Billing.Feature.Props`.","title":"Plausible.Billing.Ecto.Feature","ref":"Plausible.Billing.Ecto.Feature.html"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Feature.cast/1","ref":"Plausible.Billing.Ecto.Feature.html#cast/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Feature.dump/1","ref":"Plausible.Billing.Ecto.Feature.html#dump/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Feature.embed_as/1","ref":"Plausible.Billing.Ecto.Feature.html#embed_as/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Feature.equal?/2","ref":"Plausible.Billing.Ecto.Feature.html#equal?/2"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Feature.load/1","ref":"Plausible.Billing.Ecto.Feature.html#load/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Feature.type/0","ref":"Plausible.Billing.Ecto.Feature.html#type/0"},{"type":"module","doc":"Ecto type representing a list of features. This is a proxy for \n`{:array, Plausible.Billing.Ecto.Feature}` and is required for Kaffy to\nrender the HTML input correctly.","title":"Plausible.Billing.Ecto.FeatureList","ref":"Plausible.Billing.Ecto.FeatureList.html"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.cast/1","ref":"Plausible.Billing.Ecto.FeatureList.html#cast/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.dump/1","ref":"Plausible.Billing.Ecto.FeatureList.html#dump/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.embed_as/1","ref":"Plausible.Billing.Ecto.FeatureList.html#embed_as/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.equal?/2","ref":"Plausible.Billing.Ecto.FeatureList.html#equal?/2"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.load/1","ref":"Plausible.Billing.Ecto.FeatureList.html#load/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.render_form/5","ref":"Plausible.Billing.Ecto.FeatureList.html#render_form/5"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.FeatureList.type/0","ref":"Plausible.Billing.Ecto.FeatureList.html#type/0"},{"type":"module","doc":"Ecto type representing a limit, that can be either a number or unlimited.\nUnlimited is dumped to the database as `-1` and loaded as `:unlimited` to\nkeep compatibility with the rest of the codebase.","title":"Plausible.Billing.Ecto.Limit","ref":"Plausible.Billing.Ecto.Limit.html"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.cast/1","ref":"Plausible.Billing.Ecto.Limit.html#cast/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.dump/1","ref":"Plausible.Billing.Ecto.Limit.html#dump/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.embed_as/1","ref":"Plausible.Billing.Ecto.Limit.html#embed_as/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.equal?/2","ref":"Plausible.Billing.Ecto.Limit.html#equal?/2"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.load/1","ref":"Plausible.Billing.Ecto.Limit.html#load/1"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.render_form/5","ref":"Plausible.Billing.Ecto.Limit.html#render_form/5"},{"type":"function","doc":"","title":"Plausible.Billing.Ecto.Limit.type/0","ref":"Plausible.Billing.Ecto.Limit.html#type/0"},{"type":"module","doc":"","title":"Plausible.Billing.EnterprisePlan","ref":"Plausible.Billing.EnterprisePlan.html"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlan.changeset/2","ref":"Plausible.Billing.EnterprisePlan.html#changeset/2"},{"type":"module","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin","ref":"Plausible.Billing.EnterprisePlanAdmin.html"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin.create_changeset/2","ref":"Plausible.Billing.EnterprisePlanAdmin.html#create_changeset/2"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin.custom_index_query/3","ref":"Plausible.Billing.EnterprisePlanAdmin.html#custom_index_query/3"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin.form_fields/1","ref":"Plausible.Billing.EnterprisePlanAdmin.html#form_fields/1"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin.index/1","ref":"Plausible.Billing.EnterprisePlanAdmin.html#index/1"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin.search_fields/1","ref":"Plausible.Billing.EnterprisePlanAdmin.html#search_fields/1"},{"type":"function","doc":"","title":"Plausible.Billing.EnterprisePlanAdmin.update_changeset/2","ref":"Plausible.Billing.EnterprisePlanAdmin.html#update_changeset/2"},{"type":"behaviour","doc":"This module provides an interface for managing features, e.g. Revenue Goals,\nFunnels and Custom Properties.\n\nFeature modules have functions for toggling the feature on/off and checking\nwhether the feature is available for a site/user.\n\nWhen defining new features, the following options are expected by the\n`__using__` macro:\n\n * `:name` - an atom representing the feature name in the plan JSON\n file (see also Plausible.Billing.Plan).\n\n * `:display_name` - human-readable display name of the feature\n\n * `:toggle_field` - the field in the %Plausible.Site{} schema that toggles\n the feature. If `nil` or not set, toggle/2 silently returns `:ok`\n\n * `:free` - if set to `true`, makes the `check_availability/1` function\n always return `:ok` (no matter the user's subscription status)\n\nFunctions defined by `__using__` can be overridden if needed.","title":"Plausible.Billing.Feature","ref":"Plausible.Billing.Feature.html"},{"type":"callback","doc":"Checks whether the site owner or the user plan includes the given feature.","title":"Plausible.Billing.Feature.check_availability/1","ref":"Plausible.Billing.Feature.html#c:check_availability/1"},{"type":"callback","doc":"Returns the human-readable display name of the feature.","title":"Plausible.Billing.Feature.display_name/0","ref":"Plausible.Billing.Feature.html#c:display_name/0"},{"type":"callback","doc":"Checks whether a feature is enabled or not. Returns false when the feature is\ndisabled or the user does not have access to it.","title":"Plausible.Billing.Feature.enabled?/1","ref":"Plausible.Billing.Feature.html#c:enabled?/1"},{"type":"callback","doc":"Returns whether the feature is free to use or not.","title":"Plausible.Billing.Feature.free?/0","ref":"Plausible.Billing.Feature.html#c:free?/0"},{"type":"function","doc":"Lists all available feature modules.","title":"Plausible.Billing.Feature.list/0","ref":"Plausible.Billing.Feature.html#list/0"},{"type":"macro","doc":"Lists all the feature short names, e.g. RevenueGoals","title":"Plausible.Billing.Feature.list_short_names/0","ref":"Plausible.Billing.Feature.html#list_short_names/0"},{"type":"callback","doc":"Returns the atom representing the feature name in the plan JSON file.","title":"Plausible.Billing.Feature.name/0","ref":"Plausible.Billing.Feature.html#c:name/0"},{"type":"callback","doc":"Returns whether the site explicitly opted out of the feature. This function\nis different from enabled/1, because enabled/1 returns false when the site\nowner does not have access to the feature.","title":"Plausible.Billing.Feature.opted_out?/1","ref":"Plausible.Billing.Feature.html#c:opted_out?/1"},{"type":"callback","doc":"Toggles the feature on and off for a site. Returns\n`{:error, :upgrade_required}` when toggling a feature the site owner does not\nhave access to.","title":"Plausible.Billing.Feature.toggle/2","ref":"Plausible.Billing.Feature.html#c:toggle/2"},{"type":"callback","doc":"Returns the %Plausible.Site{} field that toggles the feature on and off.","title":"Plausible.Billing.Feature.toggle_field/0","ref":"Plausible.Billing.Feature.html#c:toggle_field/0"},{"type":"type","doc":"","title":"Plausible.Billing.Feature.t/0","ref":"Plausible.Billing.Feature.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Billing.PaddleApi","ref":"Plausible.Billing.PaddleApi.html"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.checkout_domain/0","ref":"Plausible.Billing.PaddleApi.html#checkout_domain/0"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.fetch_prices/2","ref":"Plausible.Billing.PaddleApi.html#fetch_prices/2"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.get_invoices/1","ref":"Plausible.Billing.PaddleApi.html#get_invoices/1"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.get_subscription/1","ref":"Plausible.Billing.PaddleApi.html#get_subscription/1"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.update_subscription/2","ref":"Plausible.Billing.PaddleApi.html#update_subscription/2"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.update_subscription_preview/2","ref":"Plausible.Billing.PaddleApi.html#update_subscription_preview/2"},{"type":"function","doc":"","title":"Plausible.Billing.PaddleApi.vendors_domain/0","ref":"Plausible.Billing.PaddleApi.html#vendors_domain/0"},{"type":"module","doc":"","title":"Plausible.Billing.Plans","ref":"Plausible.Billing.Plans.html"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.all/0","ref":"Plausible.Billing.Plans.html#all/0"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.available_plans_for/2","ref":"Plausible.Billing.Plans.html#available_plans_for/2"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.business_plans_for/1","ref":"Plausible.Billing.Plans.html#business_plans_for/1"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.business_tier?/1","ref":"Plausible.Billing.Plans.html#business_tier?/1"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.business_tier_launch/0","ref":"Plausible.Billing.Plans.html#business_tier_launch/0"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.find/1","ref":"Plausible.Billing.Plans.html#find/1"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.get_price_for/2","ref":"Plausible.Billing.Plans.html#get_price_for/2"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.get_regular_plan/2","ref":"Plausible.Billing.Plans.html#get_regular_plan/2"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.get_subscription_plan/1","ref":"Plausible.Billing.Plans.html#get_subscription_plan/1"},{"type":"function","doc":"Returns a list of growth plans available for the user to choose.\n\nAs new versions of plans are introduced, users who were on old plans can\nstill choose from old plans.","title":"Plausible.Billing.Plans.growth_plans_for/1","ref":"Plausible.Billing.Plans.html#growth_plans_for/1"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.latest_enterprise_plan_with_price/2","ref":"Plausible.Billing.Plans.html#latest_enterprise_plan_with_price/2"},{"type":"function","doc":"","title":"Plausible.Billing.Plans.subscription_interval/1","ref":"Plausible.Billing.Plans.html#subscription_interval/1"},{"type":"function","doc":"Returns the most appropriate plan for a user based on their usage during a\ngiven cycle.\n\nIf the usage during the cycle exceeds the enterprise-level threshold, or if\nthe user already belongs to an enterprise plan, it suggests the :enterprise\nplan.\n\nOtherwise, it recommends the plan where the cycle usage falls just under the\nplan's limit from the available options for the user.","title":"Plausible.Billing.Plans.suggest/2","ref":"Plausible.Billing.Plans.html#suggest/2"},{"type":"function","doc":"This function takes a list of plans as an argument, gathers all product\nIDs in a single list, and makes an API call to Paddle. After a successful\nresponse, fills in the `monthly_cost` and `yearly_cost` fields for each\ngiven plan and returns the new list of plans with completed information.","title":"Plausible.Billing.Plans.with_prices/2","ref":"Plausible.Billing.Plans.html#with_prices/2"},{"type":"function","doc":"List yearly plans product IDs.","title":"Plausible.Billing.Plans.yearly_product_ids/0","ref":"Plausible.Billing.Plans.html#yearly_product_ids/0"},{"type":"module","doc":"This module provides functions to work with plans usage and limits.","title":"Plausible.Billing.Quota","ref":"Plausible.Billing.Quota.html"},{"type":"function","doc":"Returns whether the usage is below the limit or not.\nReturns false if usage is equal to the limit.","title":"Plausible.Billing.Quota.below_limit?/2","ref":"Plausible.Billing.Quota.html#below_limit?/2"},{"type":"function","doc":"","title":"Plausible.Billing.Quota.eligible_for_upgrade?/1","ref":"Plausible.Billing.Quota.html#eligible_for_upgrade?/1"},{"type":"function","doc":"Enterprise plans are always allowed to add more sites (even when\nover limit) to avoid service disruption. Their usage is checked\nin a background job instead (see `check_usage.ex`).","title":"Plausible.Billing.Quota.ensure_can_add_new_site/1","ref":"Plausible.Billing.Quota.html#ensure_can_add_new_site/1"},{"type":"function","doc":"","title":"Plausible.Billing.Quota.ensure_feature_access/2","ref":"Plausible.Billing.Quota.html#ensure_feature_access/2"},{"type":"function","doc":"Ensures that the given user (or the usage map) is within the limits\nof the given plan.\n\nAn `opts` argument can be passed with `ignore_pageview_limit: true`\nwhich bypasses the pageview limit check and returns `:ok` as long as\nthe other limits are not exceeded.","title":"Plausible.Billing.Quota.ensure_within_plan_limits/3","ref":"Plausible.Billing.Quota.html#ensure_within_plan_limits/3"},{"type":"function","doc":"","title":"Plausible.Billing.Quota.exceeded_cycles/2","ref":"Plausible.Billing.Quota.html#exceeded_cycles/2"},{"type":"function","doc":"","title":"Plausible.Billing.Quota.exceeds_last_two_usage_cycles?/2","ref":"Plausible.Billing.Quota.html#exceeds_last_two_usage_cycles?/2"},{"type":"function","doc":"Suggests a suitable tier (Growth or Business) for the given usage map.\n\nIf even the highest Business plan does not accommodate the usage, then\n`:custom` is returned. This means that this kind of usage should get on\na custom plan.\n\n`nil` is returned if the usage is not eligible for upgrade.","title":"Plausible.Billing.Quota.suggest_tier/3","ref":"Plausible.Billing.Quota.html#suggest_tier/3"},{"type":"function","doc":"Returns whether the usage is within the limit or not.\nReturns true if usage is equal to the limit.","title":"Plausible.Billing.Quota.within_limit?/2","ref":"Plausible.Billing.Quota.html#within_limit?/2"},{"type":"module","doc":"","title":"Plausible.Billing.SiteLocker","ref":"Plausible.Billing.SiteLocker.html"},{"type":"function","doc":"","title":"Plausible.Billing.SiteLocker.send_grace_period_end_email/1","ref":"Plausible.Billing.SiteLocker.html#send_grace_period_end_email/1"},{"type":"function","doc":"","title":"Plausible.Billing.SiteLocker.set_lock_status_for/2","ref":"Plausible.Billing.SiteLocker.html#set_lock_status_for/2"},{"type":"function","doc":"","title":"Plausible.Billing.SiteLocker.update_sites_for/2","ref":"Plausible.Billing.SiteLocker.html#update_sites_for/2"},{"type":"type","doc":"","title":"Plausible.Billing.SiteLocker.lock_reason/0","ref":"Plausible.Billing.SiteLocker.html#t:lock_reason/0"},{"type":"type","doc":"","title":"Plausible.Billing.SiteLocker.update_opt/0","ref":"Plausible.Billing.SiteLocker.html#t:update_opt/0"},{"type":"module","doc":"The subscription statuses are stored in Paddle. They can only be changed\nthrough Paddle webhooks, which always send the current subscription status\nvia the payload.\n\n* `active` - All good with the payments. Can access stats.\n\n* `past_due` - The payment has failed, but we're trying to charge the customer\n again. Access to stats is still granted. There will be three retries - after\n 3, 5, and 7 days have passed from the first failure. After a failure on the\n final retry, the subscription status will change to `paused`. As soon as the\n customer updates their billing details, Paddle will charge them again, and\n after a successful payment, the subscription will become `active` again.\n\n* `paused` - we've tried to charge the customer but all the retries have failed.\n Stats access restricted. As soon as the customer updates their billing details,\n Paddle will charge them again, and after a successful payment, the subscription\n will become `active` again.\n\n* `deleted` - The customer has triggered the cancel subscription action. Access\n to stats should be granted for the time the customer has already paid for. If\n they want to upgrade again, new billing details have to be provided.\n\nPaddle documentation links for reference:\n\n* Subscription statuses -\n https://developer.paddle.com/classic/reference/zg9joji1mzu0mdi2-subscription-status-reference\n\n* Payment failures -\n https://developer.paddle.com/classic/guides/zg9joji1mzu0mduy-payment-failures","title":"Plausible.Billing.Subscription.Status","ref":"Plausible.Billing.Subscription.Status.html"},{"type":"macro","doc":"","title":"Plausible.Billing.Subscription.Status.active/0","ref":"Plausible.Billing.Subscription.Status.html#active/0"},{"type":"function","doc":"","title":"Plausible.Billing.Subscription.Status.active?/1","ref":"Plausible.Billing.Subscription.Status.html#active?/1"},{"type":"macro","doc":"","title":"Plausible.Billing.Subscription.Status.deleted/0","ref":"Plausible.Billing.Subscription.Status.html#deleted/0"},{"type":"function","doc":"","title":"Plausible.Billing.Subscription.Status.deleted?/1","ref":"Plausible.Billing.Subscription.Status.html#deleted?/1"},{"type":"macro","doc":"","title":"Plausible.Billing.Subscription.Status.in?/2","ref":"Plausible.Billing.Subscription.Status.html#in?/2"},{"type":"macro","doc":"","title":"Plausible.Billing.Subscription.Status.past_due/0","ref":"Plausible.Billing.Subscription.Status.html#past_due/0"},{"type":"function","doc":"","title":"Plausible.Billing.Subscription.Status.past_due?/1","ref":"Plausible.Billing.Subscription.Status.html#past_due?/1"},{"type":"macro","doc":"","title":"Plausible.Billing.Subscription.Status.paused/0","ref":"Plausible.Billing.Subscription.Status.html#paused/0"},{"type":"function","doc":"","title":"Plausible.Billing.Subscription.Status.paused?/1","ref":"Plausible.Billing.Subscription.Status.html#paused?/1"},{"type":"function","doc":"","title":"Plausible.Billing.Subscription.Status.valid_statuses/0","ref":"Plausible.Billing.Subscription.Status.html#valid_statuses/0"},{"type":"type","doc":"","title":"Plausible.Billing.Subscription.Status.status/0","ref":"Plausible.Billing.Subscription.Status.html#t:status/0"},{"type":"behaviour","doc":"Caching interface specific for Plausible. Usage:\n\n use Plausible.Cache\n\n # - Implement the callbacks required\n # - Optionally override `unwrap_cache_keys/1`\n # - Populate the cache with `Plausible.Cache.Warmer`\n\nServes as a wrapper around `Plausible.Cache.Adapter`, where the underlying\nimplementation can be transparently swapped.\n\nEven though normally the relevant Adapter processes are started, cache access is disabled\nduring tests via the `:plausible, Elixir.Plausible.Cache, enabled: bool()` application env key.\nThis can be overridden on case by case basis, using the child specs options.\n\nThe `base_db_query/0` callback is used to generate the base query that is \nexecuted on every cache refresh. \n\nThere are two modes of refresh operation: `:all` and `:updated_recently`;\nthe former will invoke the query as is and clear all the existing entries,\nwhile the latter will attempt to limit the query to only the records that \nhave been updated in the last 15 minutes and try to merge the new results with\nexisting cache entries.\n\nBoth refresh modes are normally executed periodically from within a warmer process;\nsee: `Plausible.Cache.Warmer`. The reason for two modes is that the latter is lighter \non the database and can be executed more frequently.\n\nWhen Cache is disabled via application env, the `get/1` function\nfalls back to pure database lookups (implemented via `get_from_source/1` callback. \nThis should help with introducing cached lookups in existing code, \nso that no existing tests should break.\n\nRefreshing the cache emits telemetry event defined as per `telemetry_event_refresh/2`.","title":"Plausible.Cache","ref":"Plausible.Cache.html"},{"type":"callback","doc":"Returns the base Ecto query used to refresh the cache","title":"Plausible.Cache.base_db_query/0","ref":"Plausible.Cache.html#c:base_db_query/0"},{"type":"callback","doc":"Supervisor child id, must be unique within the supervision tree","title":"Plausible.Cache.child_id/0","ref":"Plausible.Cache.html#c:child_id/0"},{"type":"callback","doc":"Counts all items at the source, an aggregate query most likely","title":"Plausible.Cache.count_all/0","ref":"Plausible.Cache.html#c:count_all/0"},{"type":"function","doc":"Looks for application env value at `:plausible, Elixir.Plausible.Cache, enabled: bool()`","title":"Plausible.Cache.enabled?/0","ref":"Plausible.Cache.html#enabled?/0"},{"type":"callback","doc":"Retrieves the item from the source, in case the cache is disabled","title":"Plausible.Cache.get_from_source/1","ref":"Plausible.Cache.html#c:get_from_source/1"},{"type":"callback","doc":"Unique cache name, used by underlying implementation","title":"Plausible.Cache.name/0","ref":"Plausible.Cache.html#c:name/0"},{"type":"callback","doc":"Optionally unwraps the keys of the cache items, in case one item is stored under multiple keys","title":"Plausible.Cache.unwrap_cache_keys/1","ref":"Plausible.Cache.html#c:unwrap_cache_keys/1"},{"type":"module","doc":"Interface for the underlying cache implementation.\nCurrently: ConCache\n\nUsing the Adapter module directly, the user must ensure that the relevant\nprocesses are available to use, which is normally done via the child specification.","title":"Plausible.Cache.Adapter","ref":"Plausible.Cache.Adapter.html"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.child_spec/3","ref":"Plausible.Cache.Adapter.html#child_spec/3"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.delete/2","ref":"Plausible.Cache.Adapter.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.fetch/3","ref":"Plausible.Cache.Adapter.html#fetch/3"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.get/2","ref":"Plausible.Cache.Adapter.html#get/2"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.get/3","ref":"Plausible.Cache.Adapter.html#get/3"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.keys/1","ref":"Plausible.Cache.Adapter.html#keys/1"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.put/4","ref":"Plausible.Cache.Adapter.html#put/4"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.put_many/2","ref":"Plausible.Cache.Adapter.html#put_many/2"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.size/1","ref":"Plausible.Cache.Adapter.html#size/1"},{"type":"function","doc":"","title":"Plausible.Cache.Adapter.with_lock!/4","ref":"Plausible.Cache.Adapter.html#with_lock!/4"},{"type":"module","doc":"Keeps track of hit/miss ratio for various caches.","title":"Plausible.Cache.Stats","ref":"Plausible.Cache.Stats.html"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.bump/2","ref":"Plausible.Cache.Stats.html#bump/2"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.Cache.Stats.child_spec/1","ref":"Plausible.Cache.Stats.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.gather/1","ref":"Plausible.Cache.Stats.html#gather/1"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.handle_telemetry_event/4","ref":"Plausible.Cache.Stats.html#handle_telemetry_event/4"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.hit_rate/1","ref":"Plausible.Cache.Stats.html#hit_rate/1"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.init/1","ref":"Plausible.Cache.Stats.html#init/1"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.size/1","ref":"Plausible.Cache.Stats.html#size/1"},{"type":"function","doc":"","title":"Plausible.Cache.Stats.start_link/1","ref":"Plausible.Cache.Stats.html#start_link/1"},{"type":"module","doc":"A periodic cache warmer.\n\nChild specification options available:\n\n * `cache_impl` - module expected to implement `Plausible.Cache` behaviour\n * `interval` - the number of milliseconds for each warm-up cycle\n * `cache_name` - defaults to cache_impl.name() but can be overridden for testing\n * `force_start?` - enforcess process startup for testing, even if it's barred\n by `Plausible.Cache.enabled?`. This is useful for avoiding issues with DB ownership\n and async tests.\n * `warmer_fn` - by convention, either `:refresh_all` or `:refresh_updated_recently`,\n both are automatically provided by `cache_impl` module. Technically any exported\n or captured function will work, if need be.\n\nSee tests for more comprehensive examples.","title":"Plausible.Cache.Warmer","ref":"Plausible.Cache.Warmer.html"},{"type":"function","doc":"","title":"Plausible.Cache.Warmer.child_spec/1","ref":"Plausible.Cache.Warmer.html#child_spec/1"},{"type":"module","doc":"Helper function for working with Ecto changesets","title":"Plausible.ChangesetHelpers","ref":"Plausible.ChangesetHelpers.html"},{"type":"function","doc":"","title":"Plausible.ChangesetHelpers.traverse_errors/1","ref":"Plausible.ChangesetHelpers.html#traverse_errors/1"},{"type":"module","doc":"Provides the core functions to retrieve and manage\nthe CLDR data that supports formatting and localisation.\n\nIt provides the core functions to access formatted\nCLDR data, set and retrieve a current locale and validate\ncertain core data types such as locales, currencies and\nterritories.","title":"Plausible.Cldr","ref":"Plausible.Cldr.html"},{"type":"function","doc":"","title":"Plausible.Cldr.available_locale_name?/1","ref":"Plausible.Cldr.html#available_locale_name?/1"},{"type":"function","doc":"Returns the default `locale`.","title":"Plausible.Cldr.default_locale/0","ref":"Plausible.Cldr.html#default_locale/0"},{"type":"function","doc":"iex> Plausible.Cldr.default_locale()\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en-001\",\n cldr_locale_name: :\"en-001\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en-001\",\n script: :Latn,\n territory: :\"001\",\n transform: %{},\n language_variants: []\n }","title":"Example - Plausible.Cldr.default_locale/0","ref":"Plausible.Cldr.html#default_locale/0-example"},{"type":"function","doc":"Returns the default territory when a locale\ndoes not specify one and none can be inferred.","title":"Plausible.Cldr.default_territory/0","ref":"Plausible.Cldr.html#default_territory/0"},{"type":"function","doc":"iex> Plausible.Cldr.default_territory()\n :\"001\"","title":"Example - Plausible.Cldr.default_territory/0","ref":"Plausible.Cldr.html#default_territory/0-example"},{"type":"function","doc":"Add locale-specific ellipsis to a string.","title":"Plausible.Cldr.ellipsis/2","ref":"Plausible.Cldr.html#ellipsis/2"},{"type":"function","doc":"* `string` is any `String.t` or a 2-element list\n of `String.t` between which the ellipsis is inserted.\n\n* `backend` is any module that includes `use Cldr` and therefore\n is a `Cldr` backend module. The default is `Cldr.default_backend!/0`.\n Note that `Cldr.default_backend!/0` will raise an exception if\n no `:default_backend` is configured under the `:ex_cldr` key in\n `config.exs`.\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.ellipsis/2","ref":"Plausible.Cldr.html#ellipsis/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale name returned by `Cldr.known_locale_names/1`.\n The default is `Cldr.get_locale/0`.\n\n* `:location` determines where to place the ellipsis. The options are\n `:after` (the default for a single string argument), `:between`\n (the default and only valid location for an argument that is a list\n of two strings) and `:before`.\n\n* `:format` formats based upon whether the ellipsis\n is inserted between words or sentences. The valid options are\n `:word` or `:sentence`. The default is `:sentence`.","title":"Options - Plausible.Cldr.ellipsis/2","ref":"Plausible.Cldr.html#ellipsis/2-options"},{"type":"function","doc":"iex> Plausible.Cldr.ellipsis(\"And furthermore\")\n \"And furthermore…\"\n\n iex> Plausible.Cldr.ellipsis([\"And furthermore\", \"there is much to be done\"], locale: :ja)\n \"And furthermore…there is much to be done\"\n\n iex> Plausible.Cldr.ellipsis(\"And furthermore\", format: :word)\n \"And furthermore …\"\n\n iex> Plausible.Cldr.ellipsis([\"And furthermore\", \"there is much to be done\"], locale: :ja, format: :word)\n \"And furthermore … there is much to be done\"","title":"Examples - Plausible.Cldr.ellipsis/2","ref":"Plausible.Cldr.html#ellipsis/2-examples"},{"type":"function","doc":"Return the current locale to be used for `Cldr` functions that\ntake an optional locale parameter for which a locale is not supplied.","title":"Plausible.Cldr.get_locale/0","ref":"Plausible.Cldr.html#get_locale/0"},{"type":"function","doc":"iex> Plausible.Cldr.put_locale(\"pl\")\n iex> Plausible.Cldr.get_locale()\n %Cldr.LanguageTag{\n backend: Elixir.Plausible.Cldr,\n canonical_locale_name: \"pl\",\n cldr_locale_name: :pl,\n extensions: %{},\n language: \"pl\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :pl,\n territory: :PL,\n requested_locale_name: \"pl\",\n script: :Latn,\n transform: %{},\n language_variants: []\n }","title":"Example - Plausible.Cldr.get_locale/0","ref":"Plausible.Cldr.html#get_locale/0-example"},{"type":"function","doc":"","title":"Plausible.Cldr.known_calendars/0","ref":"Plausible.Cldr.html#known_calendars/0"},{"type":"function","doc":"","title":"Plausible.Cldr.known_currencies/0","ref":"Plausible.Cldr.html#known_currencies/0"},{"type":"function","doc":"Returns either the Gettext `locale_name` in Cldr format or\n`false` based upon whether the locale name is configured in\n`Gettext`.","title":"Plausible.Cldr.known_gettext_locale_name/1","ref":"Plausible.Cldr.html#known_gettext_locale_name/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_gettext_locale_names/0`","title":"Arguments - Plausible.Cldr.known_gettext_locale_name/1","ref":"Plausible.Cldr.html#known_gettext_locale_name/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.known_gettext_locale_name(\"en\")\n \"en\"\n\n iex> Plausible.Cldr.known_gettext_locale_name(\"en-SA\")\n false","title":"Examples - Plausible.Cldr.known_gettext_locale_name/1","ref":"Plausible.Cldr.html#known_gettext_locale_name/1-examples"},{"type":"function","doc":"Returns a boolean indicating if the specified locale\nname is configured and available in Gettext.","title":"Plausible.Cldr.known_gettext_locale_name?/1","ref":"Plausible.Cldr.html#known_gettext_locale_name?/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.known_gettext_locale_name?/1","ref":"Plausible.Cldr.html#known_gettext_locale_name?/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.known_gettext_locale_name?(\"en\")\n true\n\n iex> Plausible.Cldr.known_gettext_locale_name?(\"!!\")\n false","title":"Examples - Plausible.Cldr.known_gettext_locale_name?/1","ref":"Plausible.Cldr.html#known_gettext_locale_name?/1-examples"},{"type":"function","doc":"Returns a list of Gettext locale names but in CLDR format with\nunderscore replaced by hyphen in order to facilitate comparisons\nwith `Cldr` locale names.","title":"Plausible.Cldr.known_gettext_locale_names/0","ref":"Plausible.Cldr.html#known_gettext_locale_names/0"},{"type":"function","doc":"Returns either the `locale_name` or `false` based upon\nwhether the locale name is configured in `Cldr`.\n\nThis is helpful when building a list of `or` expressions\nto return the first known locale name from a list.","title":"Plausible.Cldr.known_locale_name/1","ref":"Plausible.Cldr.html#known_locale_name/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.known_locale_name/1","ref":"Plausible.Cldr.html#known_locale_name/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.known_locale_name(:\"en-AU\")\n :\"en-AU\"\n\n iex> Plausible.Cldr.known_locale_name(:\"en-SA\")\n false","title":"Examples - Plausible.Cldr.known_locale_name/1","ref":"Plausible.Cldr.html#known_locale_name/1-examples"},{"type":"function","doc":"Returns a boolean indicating if the specified locale\nname is configured and available in Cldr.","title":"Plausible.Cldr.known_locale_name?/1","ref":"Plausible.Cldr.html#known_locale_name?/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.known_locale_name?/1","ref":"Plausible.Cldr.html#known_locale_name?/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.known_locale_name?(:en)\n true\n\n iex> Plausible.Cldr.known_locale_name?(:\"!!\")\n false","title":"Examples - Plausible.Cldr.known_locale_name?/1","ref":"Plausible.Cldr.html#known_locale_name?/1-examples"},{"type":"function","doc":"Returns a list of the known locale names.\n\nKnown locales are those locales which\nare the subset of all CLDR locales that\nhave been configured for use either\nin this module or in `Gettext`.","title":"Plausible.Cldr.known_locale_names/0","ref":"Plausible.Cldr.html#known_locale_names/0"},{"type":"function","doc":"Returns a list of atoms representing the number systems types known to `Cldr`.","title":"Plausible.Cldr.known_number_system_types/0","ref":"Plausible.Cldr.html#known_number_system_types/0"},{"type":"function","doc":"iex> Plausible.Cldr.known_number_system_types()\n [:default, :finance, :native, :traditional]","title":"Example - Plausible.Cldr.known_number_system_types/0","ref":"Plausible.Cldr.html#known_number_system_types/0-example"},{"type":"function","doc":"","title":"Plausible.Cldr.known_number_systems/0","ref":"Plausible.Cldr.html#known_number_systems/0"},{"type":"function","doc":"Returns either the RBNF `locale_name` or `false` based upon\nwhether the locale name is configured in `Cldr`\nand has RBNF rules defined.","title":"Plausible.Cldr.known_rbnf_locale_name/1","ref":"Plausible.Cldr.html#known_rbnf_locale_name/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.known_rbnf_locale_name/1","ref":"Plausible.Cldr.html#known_rbnf_locale_name/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.known_rbnf_locale_name(:en)\n :en\n\n iex> Plausible.Cldr.known_rbnf_locale_name(:\"en-SA\")\n false","title":"Examples - Plausible.Cldr.known_rbnf_locale_name/1","ref":"Plausible.Cldr.html#known_rbnf_locale_name/1-examples"},{"type":"function","doc":"Returns a boolean indicating if the specified locale\nname is configured and available in Cldr and supports\nrules based number formats (RBNF).","title":"Plausible.Cldr.known_rbnf_locale_name?/1","ref":"Plausible.Cldr.html#known_rbnf_locale_name?/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.known_rbnf_locale_name?/1","ref":"Plausible.Cldr.html#known_rbnf_locale_name?/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.known_rbnf_locale_name?(:en)\n true\n\n iex> Plausible.Cldr.known_rbnf_locale_name?(:\"!!\")\n false","title":"Examples - Plausible.Cldr.known_rbnf_locale_name?/1","ref":"Plausible.Cldr.html#known_rbnf_locale_name?/1-examples"},{"type":"function","doc":"Returns a list of locale names which have rules-based number\nformats (RBNF).","title":"Plausible.Cldr.known_rbnf_locale_names/0","ref":"Plausible.Cldr.html#known_rbnf_locale_names/0"},{"type":"function","doc":"","title":"Plausible.Cldr.known_territories/0","ref":"Plausible.Cldr.html#known_territories/0"},{"type":"function","doc":"Normalizes a string by applying transliteration\nof common symbols in numbers, currencies and dates","title":"Plausible.Cldr.normalize_lenient_parse/3","ref":"Plausible.Cldr.html#normalize_lenient_parse/3"},{"type":"function","doc":"Set the current process's Gettext locale from a\n`t:Cldr.LanguageTag`.","title":"Plausible.Cldr.put_gettext_locale/1","ref":"Plausible.Cldr.html#put_gettext_locale/1"},{"type":"function","doc":"* `locale` is a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`.","title":"Arguments - Plausible.Cldr.put_gettext_locale/1","ref":"Plausible.Cldr.html#put_gettext_locale/1-arguments"},{"type":"function","doc":"* `{:ok, gettext_locale_name}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.put_gettext_locale/1","ref":"Plausible.Cldr.html#put_gettext_locale/1-returns"},{"type":"function","doc":"1. If the `locale.gettext_locale_name` is `nil` then an error\n is returned.\n\n2. The `gettext` locale for the `gettext_backend` configured for the\n CLDR backend defined by the `t:Cldr.LanguageTag` is set.","title":"Behaviour - Plausible.Cldr.put_gettext_locale/1","ref":"Plausible.Cldr.html#put_gettext_locale/1-behaviour"},{"type":"function","doc":"iex> import Cldr.LanguageTag.Sigil\n iex> Plausible.Cldr.put_gettext_locale(~l\"en\")\n {:ok, \"en\"}\n\n iex> import Cldr.LanguageTag.Sigil\n iex> Plausible.Cldr.put_gettext_locale(~l\"de\")\n {\n :error,\n {\n Cldr.UnknownLocaleError,\n \"Locale TestBackend.Cldr.Locale.new!(\\\"de-DE\\\") does not map to a known gettext locale name\"\n }\n }","title":"Examples - Plausible.Cldr.put_gettext_locale/1","ref":"Plausible.Cldr.html#put_gettext_locale/1-examples"},{"type":"function","doc":"Set the current locale to be used for `Cldr` functions that\ntake an optional locale parameter for which a locale is not supplied.","title":"Plausible.Cldr.put_locale/1","ref":"Plausible.Cldr.html#put_locale/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `t:Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`\n\nSee [rfc5646](https://tools.ietf.org/html/rfc5646) for the specification\nof a language tag.","title":"Arguments - Plausible.Cldr.put_locale/1","ref":"Plausible.Cldr.html#put_locale/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.put_locale(\"en\")\n {:ok,\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en\",\n cldr_locale_name: :en,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en\",\n script: :Latn,\n territory: :US,\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.put_locale(\"invalid-locale!\")\n {:error, {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"!\\\" starting at position 15\"}}","title":"Examples - Plausible.Cldr.put_locale/1","ref":"Plausible.Cldr.html#put_locale/1-examples"},{"type":"function","doc":"Add locale-specific quotation marks around a string.","title":"Plausible.Cldr.quote/2","ref":"Plausible.Cldr.html#quote/2"},{"type":"function","doc":"* `string` is any valid Elixir string\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.quote/2","ref":"Plausible.Cldr.html#quote/2-arguments"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Cldr.known_locale_names/1`.\n The default is `Cldr.get_locale/0`","title":"Options - Plausible.Cldr.quote/2","ref":"Plausible.Cldr.html#quote/2-options"},{"type":"function","doc":"iex> Plausible.Cldr.quote(\"Quoted String\")\n \"“Quoted String”\"\n\n iex> Plausible.Cldr.quote(\"Quoted String\", locale: :ja)\n \"「Quoted String」\"","title":"Examples - Plausible.Cldr.quote/2","ref":"Plausible.Cldr.html#quote/2-examples"},{"type":"function","doc":"Returns a list of the locales names that are configured,\nbut not known in CLDR.\n\nSince there is a compile-time exception raised if there are\nany unknown locales this function should always\nreturn an empty list.","title":"Plausible.Cldr.unknown_locale_names/0","ref":"Plausible.Cldr.html#unknown_locale_names/0"},{"type":"function","doc":"","title":"Plausible.Cldr.validate_calendar/1","ref":"Plausible.Cldr.html#validate_calendar/1"},{"type":"function","doc":"","title":"Plausible.Cldr.validate_currency/1","ref":"Plausible.Cldr.html#validate_currency/1"},{"type":"function","doc":"Normalise and validate a locale name.","title":"Plausible.Cldr.validate_locale/1","ref":"Plausible.Cldr.html#validate_locale/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`","title":"Arguments - Plausible.Cldr.validate_locale/1","ref":"Plausible.Cldr.html#validate_locale/1-arguments"},{"type":"function","doc":"* `{:ok, language_tag}`\n\n* `{:error, reason}`","title":"Returns - Plausible.Cldr.validate_locale/1","ref":"Plausible.Cldr.html#validate_locale/1-returns"},{"type":"function","doc":"See [rfc5646](https://tools.ietf.org/html/rfc5646) for the specification\nof a language tag.","title":"Notes - Plausible.Cldr.validate_locale/1","ref":"Plausible.Cldr.html#validate_locale/1-notes"},{"type":"function","doc":"iex> Plausible.Cldr.validate_locale(:en)\n {:ok,\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en\",\n cldr_locale_name: :en,\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en\",\n script: :Latn,\n territory: :US,\n transform: %{},\n language_variants: []\n }}\n\n\n iex> Plausible.Cldr.validate_locale Plausible.Cldr.default_locale()\n {:ok,\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en-001\",\n cldr_locale_name: :\"en-001\",\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en-001\",\n script: :Latn,\n territory: :\"001\",\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.validate_locale(\"zzz\")\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zzz\\\" is invalid\"}}","title":"Examples - Plausible.Cldr.validate_locale/1","ref":"Plausible.Cldr.html#validate_locale/1-examples"},{"type":"function","doc":"","title":"Plausible.Cldr.validate_number_system/1","ref":"Plausible.Cldr.html#validate_number_system/1"},{"type":"function","doc":"Normalise and validate a number system type.","title":"Plausible.Cldr.validate_number_system_type/1","ref":"Plausible.Cldr.html#validate_number_system_type/1"},{"type":"function","doc":"* `number_system_type` is any number system type returned by\n `Cldr.known_number_system_types/1`","title":"Arguments - Plausible.Cldr.validate_number_system_type/1","ref":"Plausible.Cldr.html#validate_number_system_type/1-arguments"},{"type":"function","doc":"* `{:ok, normalized_number_system_type}` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.validate_number_system_type/1","ref":"Plausible.Cldr.html#validate_number_system_type/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.validate_number_system_type(:default)\n {:ok, :default}\n\n iex> Plausible.Cldr.validate_number_system_type(:traditional)\n {:ok, :traditional}\n\n iex> Plausible.Cldr.validate_number_system_type(:latn)\n {\n :error,\n {Cldr.UnknownNumberSystemTypeError, \"The number system type :latn is unknown\"}\n }\n\n iex> Plausible.Cldr.validate_number_system_type(\"bork\")\n {\n :error,\n {Cldr.UnknownNumberSystemTypeError, \"The number system type \\\"bork\\\" is invalid\"}\n }","title":"Examples - Plausible.Cldr.validate_number_system_type/1","ref":"Plausible.Cldr.html#validate_number_system_type/1-examples"},{"type":"function","doc":"","title":"Plausible.Cldr.validate_territory/1","ref":"Plausible.Cldr.html#validate_territory/1"},{"type":"function","doc":"Execute a function with a locale ensuring that the\ncurrent locale is restored after the function.","title":"Plausible.Cldr.with_locale/2","ref":"Plausible.Cldr.html#with_locale/2"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Cldr.known_locale_names/1`.\n\n* `fun` is any 0-arity function or function capture.","title":"Arguments - Plausible.Cldr.with_locale/2","ref":"Plausible.Cldr.html#with_locale/2-arguments"},{"type":"function","doc":"* The value returned by the function `fun/0` or\n\n* `{:error, {exception, reason}}` if the locale is invalid or\n\n* raises an exception if the current locale cannot be\n identified.","title":"Returns - Plausible.Cldr.with_locale/2","ref":"Plausible.Cldr.html#with_locale/2-returns"},{"type":"module","doc":"Parses HTTP `Accept-Language` header values as defined in\n[rfc2616](https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4).\n\nThe Accept-Language request-header field is similar to Accept, but restricts\nthe set of natural languages that are preferred as a response to the request.\nLanguage tags function are provided in `Cldr.LanguageTag`.\n\nThe format of an `Accept-Language` header is as follows in `ABNF` format:\n\n Accept-Language = \"Accept-Language\" \":\"\n 1#( language-range [ \";\" \"q\" \"=\" qvalue ] )\n language-range = ( ( 1*8ALPHA *( \"-\" 1*8ALPHA ) ) | \"*\" )\n\nEach language-range MAY be given an associated quality value which represents an\nestimate of the user's preference for the languages specified by that range. The\nquality value defaults to \"q=1\". For example,\n\n Accept-Language: da, en-gb;q=0.8, en;q=0.7\n\nwould mean: \"I prefer Danish, but will accept British English and other types of English.\"","title":"Plausible.Cldr.AcceptLanguage","ref":"Plausible.Cldr.AcceptLanguage.html"},{"type":"function","doc":"Parse an `Accept-Language` string and return the best match for\na configured `Cldr` locale.","title":"Plausible.Cldr.AcceptLanguage.best_match/1","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1"},{"type":"function","doc":"* `accept_language` is a string representing an accept language header","title":"Arguments - Plausible.Cldr.AcceptLanguage.best_match/1","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1-arguments"},{"type":"function","doc":"* `{:ok, language_tag}` or\n\n* `{:error, reason}`","title":"Returns - Plausible.Cldr.AcceptLanguage.best_match/1","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.AcceptLanguage.best_match(\"da;q=0.1,zh-TW;q=0.3\", TestBackend.Cldr)\n {:ok,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.AcceptLanguage.best_match(\"da;q=0.1,zh-TW;q=0.3\", TestBackend.Cldr)\n {:ok,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.AcceptLanguage.best_match(\"xx,yy;q=0.3\")\n {:error,\n {Cldr.NoMatchingLocale,\n \"No configured locale could be matched to \\\"xx,yy;q=0.3\\\"\"}}\n\n iex> Plausible.Cldr.AcceptLanguage.best_match(\"invalid_tag\")\n {:error, {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}","title":"Examples - Plausible.Cldr.AcceptLanguage.best_match/1","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1-examples"},{"type":"function","doc":"Parses an `Accept-Language` header value in its string\nor tokenized form to return a tuple of the form\n`{:ok, [{quality, %Cldr.LanguageTag{}}, ...]}` sorted by quality.","title":"Plausible.Cldr.AcceptLanguage.parse/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1"},{"type":"function","doc":"* `accept-language` is any string in the format defined by\n [rfc2616](https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4)\n\n* `backend` is any module that includes `use Cldr` and therefore\n is a `Cldr` backend module","title":"Arguments - Plausible.Cldr.AcceptLanguage.parse/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1-arguments"},{"type":"function","doc":"* `{:ok, [{quality, language_tag}, ...]}` or\n\n* `{:error, {Cldr.AcceptLanguageError, String.t}}`\n\nIf at least one valid language tag is found but errors are also\ndetected on one more more tags, an `{ok, list}` tuple is returned\nwith an error tuple for each invalid tag added at the end of the list.","title":"Returns - Plausible.Cldr.AcceptLanguage.parse/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1-returns"},{"type":"function","doc":"iex> Cldr.AcceptLanguage.parse(\"da,zh-TW;q=0.3\", TestBackend.Cldr)\n {:ok,\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n ]}\n\n iex> Plausible.Cldr.AcceptLanguage.parse(\"invalid_tag\")\n {:error,\n {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}\n\n iex> Plausible.Cldr.AcceptLanguage.parse(\"da,zh-TW;q=0.3,invalid_tag\")\n {:ok,\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }},\n {:error,\n {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}\n ]}","title":"Example - Plausible.Cldr.AcceptLanguage.parse/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1-example"},{"type":"function","doc":"Parses an `Accept-Language` header value in its string\nor tokenized form to produce a list of tuples of the form\n`[{quality, %Cldr.LanguageTag{}}, ...]` sorted by quality\nin descending order.","title":"Plausible.Cldr.AcceptLanguage.parse!/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1"},{"type":"function","doc":"* `accept-language` is any string in the format defined by [rfc2616](https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4)","title":"Arguments - Plausible.Cldr.AcceptLanguage.parse!/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1-arguments"},{"type":"function","doc":"* `{:ok, [{quality, language_tag}, ...]}` or\n\n* raises a `Cldr.AcceptLanguageError` exception\n\nIf at least one valid language tag is found but errors are also\ndetected on one more more tags, an `{ok, list}` tuple is returned\nwith an error tuple for each invalid tag added at the end of the list.","title":"Returns - Plausible.Cldr.AcceptLanguage.parse!/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.AcceptLanguage.parse!(\"da,zh-TW;q=0.3\")\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n ]\n\n Plausible.Cldr.AcceptLanguage.parse! \"invalid_tag\"\n ** (Cldr.AcceptLanguageError) \"Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11\n (ex_cldr) lib/cldr/accept_language.ex:304: Cldr.AcceptLanguage.parse!/1\n\n iex> Plausible.Cldr.AcceptLanguage.parse!(\"da,zh-TW;q=0.3,invalid_tag\")\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }},\n {:error,\n {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}\n ]","title":"Example - Plausible.Cldr.AcceptLanguage.parse!/1","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1-example"},{"type":"module","doc":"","title":"Plausible.Cldr.Currency","ref":"Plausible.Cldr.Currency.html"},{"type":"function","doc":"Returns a map of the metadata for all currencies for\na given locale.","title":"Plausible.Cldr.Currency.currencies_for_locale/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","title":"Arguments - Plausible.Cldr.Currency.currencies_for_locale/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3-arguments"},{"type":"function","doc":"* `{:ok, currency_map}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Currency.currencies_for_locale/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3-returns"},{"type":"function","doc":"MyApp.Cldr.Currency.currencies_for_locale(\"en\")\n => {:ok,\n %{\n FJD: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"FJD\",\n count: %{one: \"Fijian dollar\", other: \"Fijian dollars\"},\n digits: 2,\n from: nil,\n iso_digits: 2,\n name: \"Fijian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"FJD\",\n tender: true,\n to: nil\n },\n SUR: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"SUR\",\n count: %{one: \"Soviet rouble\", other: \"Soviet roubles\"},\n digits: 2,\n from: nil,\n iso_digits: nil,\n name: \"Soviet Rouble\",\n narrow_symbol: nil,\n rounding: 0,\n symbol: \"SUR\",\n tender: true,\n to: nil\n },\n ...\n }}","title":"Example - Plausible.Cldr.Currency.currencies_for_locale/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3-example"},{"type":"function","doc":"Returns a map of the metadata for all currencies for\na given locale and raises on error.","title":"Plausible.Cldr.Currency.currencies_for_locale!/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","title":"Arguments - Plausible.Cldr.Currency.currencies_for_locale!/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3-arguments"},{"type":"function","doc":"* `{:ok, currency_map}` or\n\n* raises an exception","title":"Returns - Plausible.Cldr.Currency.currencies_for_locale!/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3-returns"},{"type":"function","doc":"MyApp.Cldr.Currency.currencies_for_locale!(\"en\")\n => %{\n FJD: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"FJD\",\n count: %{one: \"Fijian dollar\", other: \"Fijian dollars\"},\n digits: 2,\n from: nil,\n iso_digits: 2,\n name: \"Fijian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"FJD\",\n tender: true,\n to: nil\n },\n SUR: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"SUR\",\n count: %{one: \"Soviet rouble\", other: \"Soviet roubles\"},\n digits: 2,\n from: nil,\n iso_digits: nil,\n name: \"Soviet Rouble\",\n narrow_symbol: nil,\n rounding: 0,\n symbol: \"SUR\",\n tender: true,\n to: nil\n },\n ...\n }","title":"Example - Plausible.Cldr.Currency.currencies_for_locale!/3","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3-example"},{"type":"function","doc":"Returns the currency metadata for the requested currency code.","title":"Plausible.Cldr.Currency.currency_for_code/2","ref":"Plausible.Cldr.Currency.html#currency_for_code/2"},{"type":"function","doc":"* `currency_or_currency_code` is a `binary` or `atom` representation\n of an ISO 4217 currency code, or a `%Cldr.Currency{}` struct.","title":"Arguments - Plausible.Cldr.Currency.currency_for_code/2","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale name returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`","title":"Options - Plausible.Cldr.Currency.currency_for_code/2","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-options"},{"type":"function","doc":"* A `{:ok, currency}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Currency.currency_for_code/2","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.currency_for_code(\"AUD\")\n {:ok,\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"AUD\",\n count: %{one: \"Australian dollar\", other: \"Australian dollars\"},\n digits: 2,\n iso_digits: 2,\n name: \"Australian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"A$\",\n tender: true\n }}\n\n iex> Plausible.Cldr.Currency.currency_for_code(\"THB\")\n {:ok,\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"THB\",\n count: %{one: \"Thai baht\", other: \"Thai baht\"},\n digits: 2,\n iso_digits: 2,\n name: \"Thai Baht\",\n narrow_symbol: \"฿\",\n rounding: 0,\n symbol: \"THB\",\n tender: true\n }}","title":"Examples - Plausible.Cldr.Currency.currency_for_code/2","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-examples"},{"type":"function","doc":"Returns the currency metadata for the requested currency code.","title":"Plausible.Cldr.Currency.currency_for_code!/2","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2"},{"type":"function","doc":"* `currency_or_currency_code` is a `binary` or `atom` representation\n of an ISO 4217 currency code, or a `%Cldr.Currency{}` struct.","title":"Arguments - Plausible.Cldr.Currency.currency_for_code!/2","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale name returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`","title":"Options - Plausible.Cldr.Currency.currency_for_code!/2","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-options"},{"type":"function","doc":"* A `t:Cldr.Current.t/0` or\n\n* raises an exception","title":"Returns - Plausible.Cldr.Currency.currency_for_code!/2","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.currency_for_code!(\"AUD\")\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"AUD\",\n count: %{one: \"Australian dollar\", other: \"Australian dollars\"},\n digits: 2,\n iso_digits: 2,\n name: \"Australian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"A$\",\n tender: true\n }\n\n iex> Plausible.Cldr.Currency.currency_for_code!(\"THB\")\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"THB\",\n count: %{one: \"Thai baht\", other: \"Thai baht\"},\n digits: 2,\n iso_digits: 2,\n name: \"Thai Baht\",\n narrow_symbol: \"฿\",\n rounding: 0,\n symbol: \"THB\",\n tender: true\n }","title":"Examples - Plausible.Cldr.Currency.currency_for_code!/2","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-examples"},{"type":"function","doc":"Returns the effective currency for a given locale","title":"Plausible.Cldr.Currency.currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1"},{"type":"function","doc":"* `locale` is a `Cldr.LanguageTag` struct returned by\n `Cldr.Locale.new!/2`","title":"Arguments - Plausible.Cldr.Currency.currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1-arguments"},{"type":"function","doc":"* A ISO 4217 currency code as an upcased atom","title":"Returns - Plausible.Cldr.Currency.currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1-returns"},{"type":"function","doc":"iex> {:ok, locale} = Plausible.Cldr.validate_locale(\"en\")\n iex> Plausible.Cldr.Currency.currency_from_locale locale\n :USD\n\n iex> {:ok, locale} = Plausible.Cldr.validate_locale(\"en-AU\")\n iex> Plausible.Cldr.Currency.currency_from_locale locale\n :AUD\n\n iex> Plausible.Cldr.Currency.currency_from_locale(\"en-GB\")\n :GBP","title":"Examples - Plausible.Cldr.Currency.currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1-examples"},{"type":"function","doc":"Returns a list of historic and the current\ncurrency for a given locale.","title":"Plausible.Cldr.Currency.currency_history_for_locale/1","ref":"Plausible.Cldr.Currency.html#currency_history_for_locale/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`","title":"Arguments - Plausible.Cldr.Currency.currency_history_for_locale/1","ref":"Plausible.Cldr.Currency.html#currency_history_for_locale/1-arguments"},{"type":"function","doc":"iex> MyApp.Cldr.Currency.currency_history_for_locale(\"en\")\n {:ok,\n %{\n USD: %{from: ~D[1792-01-01], to: nil},\n USN: %{tender: false},\n USS: %{from: nil, tender: false, to: ~D[2014-03-01]}\n }\n }","title":"Example - Plausible.Cldr.Currency.currency_history_for_locale/1","ref":"Plausible.Cldr.Currency.html#currency_history_for_locale/1-example"},{"type":"function","doc":"Returns a map that matches a currency string to a\ncurrency code.\n\nA currency string is a localised name or symbol\nrepresenting a currency in a locale-specific manner.","title":"Plausible.Cldr.Currency.currency_strings/3","ref":"Plausible.Cldr.Currency.html#currency_strings/3"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","title":"Arguments - Plausible.Cldr.Currency.currency_strings/3","ref":"Plausible.Cldr.Currency.html#currency_strings/3-arguments"},{"type":"function","doc":"* `{:ok, currency_string_map}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Currency.currency_strings/3","ref":"Plausible.Cldr.Currency.html#currency_strings/3-returns"},{"type":"function","doc":"MyApp.Cldr.Currency.currency_strings(\"en\")\n => {:ok,\n %{\n \"mexican silver pesos\" => :MXP,\n \"sudanese dinar\" => :SDD,\n \"bad\" => :BAD,\n \"rsd\" => :RSD,\n \"swazi lilangeni\" => :SZL,\n \"zairean new zaire\" => :ZRN,\n \"guyanaese dollars\" => :GYD,\n \"equatorial guinean ekwele\" => :GQE,\n ...\n }}","title":"Example - Plausible.Cldr.Currency.currency_strings/3","ref":"Plausible.Cldr.Currency.html#currency_strings/3-example"},{"type":"function","doc":"Returns a map that matches a currency string to a\ncurrency code or raises an exception.\n\nA currency string is a localised name or symbol\nrepresenting a currency in a locale-specific manner.","title":"Plausible.Cldr.Currency.currency_strings!/3","ref":"Plausible.Cldr.Currency.html#currency_strings!/3"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","title":"Arguments - Plausible.Cldr.Currency.currency_strings!/3","ref":"Plausible.Cldr.Currency.html#currency_strings!/3-arguments"},{"type":"function","doc":"* `{:ok, currency_string_map}` or\n\n* raises an exception","title":"Returns - Plausible.Cldr.Currency.currency_strings!/3","ref":"Plausible.Cldr.Currency.html#currency_strings!/3-returns"},{"type":"function","doc":"MyApp.Cldr.Currency.currency_strings!(\"en\")\n => %{\n \"mexican silver pesos\" => :MXP,\n \"sudanese dinar\" => :SDD,\n \"bad\" => :BAD,\n \"rsd\" => :RSD,\n \"swazi lilangeni\" => :SZL,\n \"zairean new zaire\" => :ZRN,\n \"guyanaese dollars\" => :GYD,\n \"equatorial guinean ekwele\" => :GQE,\n ...\n }","title":"Example - Plausible.Cldr.Currency.currency_strings!/3","ref":"Plausible.Cldr.Currency.html#currency_strings!/3-example"},{"type":"function","doc":"Returns the current currency for a given locale.\n\nThis function does not consider the `U` extenion\nparameters `cu` or `rg`. It is recommended to us\n`Cldr.Currency.currency_from_locale/1` in most\ncircumstances.","title":"Plausible.Cldr.Currency.current_currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#current_currency_from_locale/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`","title":"Arguments - Plausible.Cldr.Currency.current_currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#current_currency_from_locale/1-arguments"},{"type":"function","doc":"iex> MyApp.Cldr.Currency.current_currency_from_locale(\"en\")\n :USD\n\n iex> MyApp.Cldr.Currency.current_currency_from_locale(\"en-AU\")\n :AUD","title":"Example - Plausible.Cldr.Currency.current_currency_from_locale/1","ref":"Plausible.Cldr.Currency.html#current_currency_from_locale/1-example"},{"type":"function","doc":"Returns a mapping from a territory code to its\ncurrent currency code.\n\nIf a territory has no current currency (like\nAntartica, territory code `:AQ`) then no\nmapping is returned for that territory.","title":"Plausible.Cldr.Currency.current_territory_currencies/0","ref":"Plausible.Cldr.Currency.html#current_territory_currencies/0"},{"type":"function","doc":"* A map of `{territory_code => Cldr.Currency.t}`","title":"Returns - Plausible.Cldr.Currency.current_territory_currencies/0","ref":"Plausible.Cldr.Currency.html#current_territory_currencies/0-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.current_territory_currencies()","title":"Example - Plausible.Cldr.Currency.current_territory_currencies/0","ref":"Plausible.Cldr.Currency.html#current_territory_currencies/0-example"},{"type":"function","doc":"","title":"Plausible.Cldr.Currency.known_currencies/0","ref":"Plausible.Cldr.Currency.html#known_currencies/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Currency.known_currency?/1","ref":"Plausible.Cldr.Currency.html#known_currency?/1"},{"type":"function","doc":"Returns a 2-tuple indicating if the supplied currency code is known.","title":"Plausible.Cldr.Currency.known_currency_code/1","ref":"Plausible.Cldr.Currency.html#known_currency_code/1"},{"type":"function","doc":"* `currency_code` is a `binary` or `atom` representing an ISO4217\n currency code","title":"Arguments - Plausible.Cldr.Currency.known_currency_code/1","ref":"Plausible.Cldr.Currency.html#known_currency_code/1-arguments"},{"type":"function","doc":"* `{:ok, currency_code}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Currency.known_currency_code/1","ref":"Plausible.Cldr.Currency.html#known_currency_code/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.known_currency_code(\"AUD\")\n {:ok, :AUD}\n\n iex> Plausible.Cldr.Currency.known_currency_code(\"GGG\")\n {:error, {Cldr.UnknownCurrencyError, \"The currency \\\"GGG\\\" is invalid\"}}","title":"Examples - Plausible.Cldr.Currency.known_currency_code/1","ref":"Plausible.Cldr.Currency.html#known_currency_code/1-examples"},{"type":"function","doc":"Returns a boolean indicating if the supplied currency code is known.","title":"Plausible.Cldr.Currency.known_currency_code?/1","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1"},{"type":"function","doc":"* `currency_code` is a `binary` or `atom` representing an ISO4217\n currency code","title":"Arguments - Plausible.Cldr.Currency.known_currency_code?/1","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1-arguments"},{"type":"function","doc":"* `true` or `false`","title":"Returns - Plausible.Cldr.Currency.known_currency_code?/1","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.known_currency_code?(\"AUD\")\n true\n\n iex> Plausible.Cldr.Currency.known_currency_code?(\"GGG\")\n false\n\n iex> Plausible.Cldr.Currency.known_currency_code?(:XCV)\n false","title":"Examples - Plausible.Cldr.Currency.known_currency_code?/1","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1-examples"},{"type":"function","doc":"Returns a list of all known currency codes.","title":"Plausible.Cldr.Currency.known_currency_codes/0","ref":"Plausible.Cldr.Currency.html#known_currency_codes/0"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.known_currency_codes()","title":"Example - Plausible.Cldr.Currency.known_currency_codes/0","ref":"Plausible.Cldr.Currency.html#known_currency_codes/0-example"},{"type":"function","doc":"Returns a `Currency` struct created from the arguments.","title":"Plausible.Cldr.Currency.new/2","ref":"Plausible.Cldr.Currency.html#new/2"},{"type":"function","doc":"* `currency` is a private use currency code in a format defined by\n [ISO4217](https://en.wikipedia.org/wiki/ISO_4217)\n which is `X` followed by two alphanumeric characters.\n\n* `options` is a map of options representing the optional elements of\n the `Cldr.Currency.t` struct.","title":"Arguments - Plausible.Cldr.Currency.new/2","ref":"Plausible.Cldr.Currency.html#new/2-arguments"},{"type":"function","doc":"* `:name` is the name of the currency. Required.\n* `:digits` is the precision of the currency. Required.\n* `:symbol` is the currency symbol. Optional.\n* `:narrow_symbol` is an alternative narrow symbol. Optional.\n* `:round_nearest` is the rounding precision such as `0.05`. Optional.\n* `:alt_code` is an alternative currency code for application use.\n* `:cash_digits` is the precision of the currency when used as cash. Optional.\n* `:cash_rounding_nearest` is the rounding precision when used as cash\n such as `0.05`. Optional.","title":"Options - Plausible.Cldr.Currency.new/2","ref":"Plausible.Cldr.Currency.html#new/2-options"},{"type":"function","doc":"* `{:ok, Cldr.Currency.t}` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Currency.new/2","ref":"Plausible.Cldr.Currency.html#new/2-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.new(:XAE, name: \"Custom Name\", digits: 0)\n {:ok,\n %Cldr.Currency{\n alt_code: :XAE,\n cash_digits: 0,\n cash_rounding: nil,\n code: :XAE,\n count: %{other: \"Custom Name\"},\n digits: 0,\n from: nil,\n iso_digits: 0,\n name: \"Custom Name\",\n narrow_symbol: nil,\n rounding: 0,\n symbol: \"XAE\",\n tender: false,\n to: nil\n }}\n iex> MyApp.Cldr.Currency.new(:XAH, name: \"Custom Name\")\n {:error, \"Required options are missing. Required options are [:name, :digits]\"}\n iex> Plausible.Cldr.Currency.new(:XAE, name: \"XAE\", digits: 0)\n {:error, {Cldr.CurrencyAlreadyDefined, \"Currency :XAE is already defined.\"}}","title":"Example - Plausible.Cldr.Currency.new/2","ref":"Plausible.Cldr.Currency.html#new/2-example"},{"type":"function","doc":"Returns the appropriate currency display name for the `currency`, based\non the plural rules in effect for the `locale`.","title":"Plausible.Cldr.Currency.pluralize/3","ref":"Plausible.Cldr.Currency.html#pluralize/3"},{"type":"function","doc":"* `number` is an integer, float or `Decimal`\n\n* `currency` is any currency returned by `Cldr.Currency.known_currencies/0`\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Currency.pluralize/3","ref":"Plausible.Cldr.Currency.html#pluralize/3-arguments"},{"type":"function","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`. The\n default is `Plausible.Cldr.get_locale/0`","title":"Options - Plausible.Cldr.Currency.pluralize/3","ref":"Plausible.Cldr.Currency.html#pluralize/3-options"},{"type":"function","doc":"* `{:ok, plural_string}` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Currency.pluralize/3","ref":"Plausible.Cldr.Currency.html#pluralize/3-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Currency.pluralize(1, :USD)\n {:ok, \"US dollar\"}\n\n iex> Plausible.Cldr.Currency.pluralize(3, :USD)\n {:ok, \"US dollars\"}\n\n iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: \"zh\")\n {:ok, \"美元\"}\n\n iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: \"fr\")\n {:ok, \"dollars des États-Unis\"}\n\n iex> Plausible.Cldr.Currency.pluralize(1, :USD, locale: \"fr\")\n {:ok, \"dollar des États-Unis\"}","title":"Examples - Plausible.Cldr.Currency.pluralize/3","ref":"Plausible.Cldr.Currency.html#pluralize/3-examples"},{"type":"function","doc":"Returns the strings associated with a currency\nin a given locale.","title":"Plausible.Cldr.Currency.strings_for_currency/2","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2"},{"type":"function","doc":"* `currency` is an ISO4217 currency code\n\n* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`","title":"Arguments - Plausible.Cldr.Currency.strings_for_currency/2","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2-arguments"},{"type":"function","doc":"* A list of strings or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Currency.strings_for_currency/2","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2-returns"},{"type":"function","doc":"iex> MyApp.Cldr.Currency.strings_for_currency :AUD,(\"en\")\n [\"a$\", \"australian dollars\", \"aud\", \"australian dollar\"]","title":"Example - Plausible.Cldr.Currency.strings_for_currency/2","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2-example"},{"type":"module","doc":"Backend module that provides functions\nto define new locales and display human-readable\nlocale names for presentation purposes.","title":"Plausible.Cldr.Locale","ref":"Plausible.Cldr.Locale.html"},{"type":"function","doc":"Returns the list of fallback locale names, starting\nwith the provided locale name.\n\nFallbacks are a list of locate names which can\nbe used to resolve translation or other localization\ndata if such localised data does not exist for\nthis specific locale..","title":"Plausible.Cldr.Locale.fallback_locale_names/1","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1"},{"type":"function","doc":"* `locale_name` is any locale name returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.Locale.fallback_locale_names/1","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1-arguments"},{"type":"function","doc":"* `{:ok, list_of_locale_names}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Locale.fallback_locale_names/1","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Locale.fallback_locale_names(:\"fr-CA\")\n {:ok, [:\"fr-CA\", :fr, :und]}\n\n # Fallbacks are typically formed by progressively\n # stripping variant, territory and script from the\n # given locale name. But not always - there are\n # certain fallbacks that take a different path.\n\n iex> Plausible.Cldr.Locale.fallback_locale_names(:nb)\n {:ok, [:nb, :no, :und]}","title":"Examples - Plausible.Cldr.Locale.fallback_locale_names/1","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1-examples"},{"type":"function","doc":"Returns the list of fallback locales, starting\nwith the provided locale name.\n\nFallbacks are a list of locate names which can\nbe used to resolve translation or other localization\ndata if such localised data does not exist for\nthis specific locale.","title":"Plausible.Cldr.Locale.fallback_locales/1","ref":"Plausible.Cldr.Locale.html#fallback_locales/1"},{"type":"function","doc":"* `locale_name` is any locale name returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.Locale.fallback_locales/1","ref":"Plausible.Cldr.Locale.html#fallback_locales/1-arguments"},{"type":"function","doc":"* `{:ok, list_of_locales}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Locale.fallback_locales/1","ref":"Plausible.Cldr.Locale.html#fallback_locales/1-returns"},{"type":"function","doc":"Plausible.Cldr.Locale.fallback_locales(:\"fr-CA\")\n => {:ok,\n [#Cldr.LanguageTag , #Cldr.LanguageTag ,\n #Cldr.LanguageTag ]}\n\n # Fallbacks are typically formed by progressively\n # stripping variant, territory and script from the\n # given locale name. But not always - there are\n # certain fallbacks that take a different path.\n\n Plausible.Cldr.Locale.fallback_locales(:nb))\n => {:ok,\n [#Cldr.LanguageTag , #Cldr.LanguageTag ,\n #Cldr.LanguageTag ]}","title":"Examples - Plausible.Cldr.Locale.fallback_locales/1","ref":"Plausible.Cldr.Locale.html#fallback_locales/1-examples"},{"type":"function","doc":"Returns the \"best fit\" locale for a given territory.\n\nUsing the population percentage data from CLDR, the\nlanguage most commonly spoken in the given territory\nis used to form a locale name which is then validated\nagainst the given backend.\n\nFirst a territory-specific locale is validated and if\nthat fails, the base language only is validate.\n\nFor example, if the territory is `AU` then then the\nlanguage most spoken is \"en\". First, the locale \"en-AU\"\nis validated and if that fails, \"en\" is validated.","title":"Plausible.Cldr.Locale.locale_for_territory/1","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1"},{"type":"function","doc":"* `territory` is any ISO 3166 Alpha-2 territory\n code that can be validated by `Cldr.validate_territory/1`","title":"Arguments - Plausible.Cldr.Locale.locale_for_territory/1","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1-arguments"},{"type":"function","doc":"* `{:ok, language_tag}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Locale.locale_for_territory/1","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Locale.locale_for_territory(:AU)\n Elixir.Plausible.Cldr.validate_locale(:\"en-AU\")\n\n iex> Plausible.Cldr.Locale.locale_for_territory(:US)\n Elixir.Plausible.Cldr.validate_locale(:\"en-US\")\n\n iex> Plausible.Cldr.Locale.locale_for_territory(:ZZ)\n {:error, {Cldr.UnknownTerritoryError, \"The territory :ZZ is unknown\"}}","title":"Examples - Plausible.Cldr.Locale.locale_for_territory/1","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1-examples"},{"type":"function","doc":"Returns a \"best fit\" locale for a host name.","title":"Plausible.Cldr.Locale.locale_from_host/2","ref":"Plausible.Cldr.Locale.html#locale_from_host/2"},{"type":"function","doc":"* `host` is any valid host name\n\n* `options` is a keyword list of options. The default\n is `[]`.","title":"Arguments - Plausible.Cldr.Locale.locale_from_host/2","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-arguments"},{"type":"function","doc":"* `:tlds` is a list of territory codes as upper-cased\n atoms that are to be considered as top-level domains.\n See `Cldr.Locale.locale_from_host/2` for the default\n list.","title":"Options - Plausible.Cldr.Locale.locale_from_host/2","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-options"},{"type":"function","doc":"* `{:ok, langauge_tag}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Locale.locale_from_host/2","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-returns"},{"type":"function","doc":"Certain top-level domains have become associated with content\nunderlated to the territory for who the domain is registered.\nTherefore Google (and perhaps others) do not associate these\nTLDs as belonging to the territory but rather are considered\ngeneric top-level domain names.","title":"Notes - Plausible.Cldr.Locale.locale_from_host/2","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-notes"},{"type":"function","doc":"iex> Plausible.Cldr.Locale.locale_from_host \"a.b.com.au\"\n Elixir.Plausible.Cldr.validate_locale(:\"en-AU\")\n\n iex> Plausible.Cldr.Locale.locale_from_host(\"a.b.com.tv\")\n {:error,\n {Cldr.UnknownLocaleError, \"No locale was identified for territory \\\"tv\\\"\"}}\n\n iex> Plausible.Cldr.Locale.locale_from_host(\"a.b.com\")\n {:error,\n {Cldr.UnknownLocaleError, \"No locale was identified for territory \\\"com\\\"\"}}","title":"Examples - Plausible.Cldr.Locale.locale_from_host/2","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-examples"},{"type":"function","doc":"","title":"Plausible.Cldr.Locale.new/1","ref":"Plausible.Cldr.Locale.html#new/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Locale.new!/1","ref":"Plausible.Cldr.Locale.html#new!/1"},{"type":"function","doc":"Returns the script direction for a locale.","title":"Plausible.Cldr.Locale.script_direction_from_locale/1","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1"},{"type":"function","doc":"* `language_tag` is any language tag returned by `Cldr.Locale.new/2`\n or any `locale_name` returned by `Cldr.known_locale_names/1`.","title":"Arguments - Plausible.Cldr.Locale.script_direction_from_locale/1","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1-arguments"},{"type":"function","doc":"* The script direction which is either `:ltr` (for left-to-right\n scripts) or `:rtl` (for right-to-left scripts).","title":"Returns - Plausible.Cldr.Locale.script_direction_from_locale/1","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Locale.script_direction_from_locale \"en-US\"\n :ltr\n\n iex> Plausible.Cldr.Locale.script_direction_from_locale :ar\n :rtl","title":"Examples - Plausible.Cldr.Locale.script_direction_from_locale/1","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1-examples"},{"type":"function","doc":"Returns the last segment of a host that might\nbe a territory.","title":"Plausible.Cldr.Locale.territory_from_host/1","ref":"Plausible.Cldr.Locale.html#territory_from_host/1"},{"type":"function","doc":"* `host` is any valid host name","title":"Arguments - Plausible.Cldr.Locale.territory_from_host/1","ref":"Plausible.Cldr.Locale.html#territory_from_host/1-arguments"},{"type":"function","doc":"* `{:ok, territory}` or\n\n* `{:error, {exception, reason}}`","title":"Returns - Plausible.Cldr.Locale.territory_from_host/1","ref":"Plausible.Cldr.Locale.html#territory_from_host/1-returns"},{"type":"function","doc":"iex> Cldr.Locale.territory_from_host(\"a.b.com.au\")\n {:ok, :AU}\n\n iex> Cldr.Locale.territory_from_host(\"a.b.com\")\n {:error,\n {Cldr.UnknownLocaleError, \"No locale was identified for territory \\\"com\\\"\"}}","title":"Examples - Plausible.Cldr.Locale.territory_from_host/1","ref":"Plausible.Cldr.Locale.html#territory_from_host/1-examples"},{"type":"function","doc":"Returns the territory from a language tag or\nlocale name.","title":"Plausible.Cldr.Locale.territory_from_locale/1","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1"},{"type":"function","doc":"* `locale` is any language tag returned by\n `Plausible.Cldr.Locale.new/1`\n or a locale name in the list returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.Locale.territory_from_locale/1","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1-arguments"},{"type":"function","doc":"* A territory code as an atom","title":"Returns - Plausible.Cldr.Locale.territory_from_locale/1","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Locale.territory_from_locale \"en-US\"\n :US\n\n iex> Plausible.Cldr.Locale.territory_from_locale \"en-US-u-rg-GBzzzz\"\n :GB","title":"Examples - Plausible.Cldr.Locale.territory_from_locale/1","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1-examples"},{"type":"function","doc":"Returns the time zone from a language tag or\nlocale name.","title":"Plausible.Cldr.Locale.timezone_from_locale/1","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1"},{"type":"function","doc":"* `locale` is any language tag returned by\n `Plausible.Cldr.Locale.new/1`\n or a locale name in the list returned by\n `Plausible.Cldr.known_locale_names/0`","title":"Arguments - Plausible.Cldr.Locale.timezone_from_locale/1","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1-arguments"},{"type":"function","doc":"* A time zone ID as a string or\n\n* `:error` if no time zone can be determined","title":"Returns - Plausible.Cldr.Locale.timezone_from_locale/1","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Locale.timezone_from_locale \"en-US-u-tz-ausyd\"\n \"Australia/Sydney\"","title":"Examples - Plausible.Cldr.Locale.timezone_from_locale/1","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1-examples"},{"type":"module","doc":"Formats numbers and currencies based upon CLDR's decimal formats specification.\n\nThe format specification is documentated in [Unicode TR35](http://unicode.org/reports/tr35/tr35-numbers.html#Number_Formats).\nThere are several classes of formatting including non-scientific, scientific,\nrules based (for spelling and ordinal formats), compact formats that display `1k`\nrather than `1,000` and so on. See `Cldr.Number.to_string/2` for specific formatting\noptions.\n\n#","title":"Plausible.Cldr.Number","ref":"Plausible.Cldr.Number.html"},{"type":"module","doc":"The following description applies to formats that do not use scientific\nnotation or significant digits:\n\n* If the number of actual integer digits exceeds the maximum integer digits,\n then only the least significant digits are shown. For example, 1997 is\n formatted as \"97\" if the maximum integer digits is set to 2.\n\n* If the number of actual integer digits is less than the minimum integer\n digits, then leading zeros are added. For example, 1997 is formatted as\n \"01997\" if the minimum integer digits is set to 5.\n\n* If the number of actual fraction digits exceeds the maximum fraction\n digits, then half-even rounding it performed to the maximum fraction\n digits. For example, 0.125 is formatted as \"0.12\" if the maximum fraction\n digits is 2. This behavior can be changed by specifying a rounding\n increment and a rounding mode.\n\n* If the number of actual fraction digits is less than the minimum fraction\n digits, then trailing zeros are added. For example, 0.125 is formatted as\n \"0.1250\" if the minimum fraction digits is set to 4.\n\n* Trailing fractional zeros are not displayed if they occur j positions after\n the decimal, where j is less than the maximum fraction digits. For example,\n 0.10004 is formatted as \"0.1\" if the maximum fraction digits is four or\n less.\n\n#","title":"Non-Scientific Notation Formatting - Plausible.Cldr.Number","ref":"Plausible.Cldr.Number.html#module-non-scientific-notation-formatting"},{"type":"module","doc":"Numbers in scientific notation are expressed as the product of a mantissa and\na power of ten, for example, 1234 can be expressed as 1.234 x 10^3. The\nmantissa is typically in the half-open interval [1.0, 10.0) or sometimes\n[0.0, 1.0), but it need not be. In a pattern, the exponent character\nimmediately followed by one or more digit characters indicates scientific\nnotation. Example: \"0.###E0\" formats the number 1234 as \"1.234E3\".\n\n* The number of digit characters after the exponent character gives the\n minimum exponent digit count. There is no maximum. Negative exponents are\n formatted using the localized minus sign, not the prefix and suffix from\n the pattern. This allows patterns such as \"0.###E0 m/s\". To prefix positive\n exponents with a localized plus sign, specify '+' between the exponent and\n the digits: \"0.###E+0\" will produce formats \"1E+1\", \"1E+0\", \"1E-1\", and so\n on. (In localized patterns, use the localized plus sign rather than '+'.)\n\n* The minimum number of integer digits is achieved by adjusting the exponent.\n Example: 0.00123 formatted with \"00.###E0\" yields \"12.3E-4\". This only\n happens if there is no maximum number of integer digits. If there is a\n maximum, then the minimum number of integer digits is fixed at one.\n\n* The maximum number of integer digits, if present, specifies the exponent\n grouping. The most common use of this is to generate engineering notation,\n in which the exponent is a multiple of three, for example, \"##0.###E0\". The\n number 12345 is formatted using \"##0.####E0\" as \"12.345E3\".\n\n* When using scientific notation, the formatter controls the digit counts\n using significant digits logic. The maximum number of significant digits\n limits the total number of integer and fraction digits that will be shown\n in the mantissa; it does not affect parsing. For example, 12345 formatted\n with \"##0.##E0\" is \"12.3E3\". Exponential patterns may not contain grouping\n separators.\n\n#","title":"Scientific Notation Formatting - Plausible.Cldr.Number","ref":"Plausible.Cldr.Number.html#module-scientific-notation-formatting"},{"type":"module","doc":"There are two ways of controlling how many digits are shows: (a)\nsignificant digits counts, or (b) integer and fraction digit counts. Integer\nand fraction digit counts are described above. When a formatter is using\nsignificant digits counts, it uses however many integer and fraction digits\nare required to display the specified number of significant digits. It may\nignore min/max integer/fraction digits, or it may use them to the extent\npossible.","title":"Significant Digits - Plausible.Cldr.Number","ref":"Plausible.Cldr.Number.html#module-significant-digits"},{"type":"function","doc":"Parse a string locale-aware manner and return\na number.","title":"Plausible.Cldr.Number.parse/2","ref":"Plausible.Cldr.Number.html#parse/2"},{"type":"function","doc":"* `string` is any `String.t`\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Number.parse/2","ref":"Plausible.Cldr.Number.html#parse/2-arguments"},{"type":"function","doc":"* `:number` is one of `:integer`, `:float`,\n `:decimal` or `nil`. The default is `nil`\n meaning that the type auto-detected as either\n an `integer` or a `float`.\n\n* `:locale` is any locale returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag.t`. The default is\n `Plausible.Cldr.get_locale/0`.","title":"Options - Plausible.Cldr.Number.parse/2","ref":"Plausible.Cldr.Number.html#parse/2-options"},{"type":"function","doc":"* A number of the requested or default type or\n\n* `{:error, {exception, error}}` if no number could be determined","title":"Returns - Plausible.Cldr.Number.parse/2","ref":"Plausible.Cldr.Number.html#parse/2-returns"},{"type":"function","doc":"This function parses a string to return a number but\nin a locale-aware manner. It will normalise grouping\ncharacters and decimal separators, different forms of\nthe `+` and `-` symbols that appear in Unicode and\nstrips any `_` characters that might be used for\nformatting in a string. It then parses the number\nusing the Elixir standard library functions.","title":"Notes - Plausible.Cldr.Number.parse/2","ref":"Plausible.Cldr.Number.html#parse/2-notes"},{"type":"function","doc":"iex> Plausible.Cldr.Number.parse(\"+1.000,34\", locale: \"de\")\n {:ok, 1000.34}\n\n iex> Plausible.Cldr.Number.parse(\"-1_000_000.34\")\n {:ok, -1000000.34}\n\n iex> Plausible.Cldr.Number.parse(\"1.000\", locale: \"de\", number: :integer)\n {:ok, 1000}\n\n iex> Plausible.Cldr.Number.parse(\"+1.000,34\", locale: \"de\", number: :integer)\n {:error,\n {Cldr.Number.ParseError,\n \"The string \\\"+1.000,34\\\" could not be parsed as a number\"}}","title":"Examples - Plausible.Cldr.Number.parse/2","ref":"Plausible.Cldr.Number.html#parse/2-examples"},{"type":"function","doc":"Resolve curencies from strings within\na list.","title":"Plausible.Cldr.Number.resolve_currencies/2","ref":"Plausible.Cldr.Number.html#resolve_currencies/2"},{"type":"function","doc":"* `list` is any list in which currency\n names and symbols are expected\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Number.resolve_currencies/2","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `Plausible.Cldr.get_locale()`\n\n* `:only` is an `atom` or list of `atoms` representing the\n currencies or currency types to be considered for a match.\n The equates to a list of acceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:except` is an `atom` or list of `atoms` representing the\n currencies or currency types to be not considered for a match.\n This equates to a list of unacceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:fuzzy` is a float greater than `0.0` and less than or\n equal to `1.0` which is used as input to\n `String.jaro_distance/2` to determine is the provided\n currency string is *close enough* to a known currency\n string for it to identify definitively a currency code.\n It is recommended to use numbers greater than `0.8` in\n order to reduce false positives.","title":"Options - Plausible.Cldr.Number.resolve_currencies/2","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-options"},{"type":"function","doc":"The `:only` and `:except` options accept a list of\ncurrency codes and/or currency types. The following\ntypes are recognised.\n\nIf both `:only` and `:except` are specified,\nthe `:except` entries take priority - that means\nany entries in `:except` are removed from the `:only`\nentries.\n\n * `:all`, the default, considers all currencies\n\n * `:current` considers those currencies that have a `:to`\n date of nil and which also is a known ISO4217 currency\n\n * `:historic` is the opposite of `:current`\n\n * `:tender` considers currencies that are legal tender\n\n * `:unannotated` considers currencies that don't have\n \"(some string)\" in their names. These are usually\n financial instruments.","title":"Notes - Plausible.Cldr.Number.resolve_currencies/2","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-notes"},{"type":"function","doc":"iex> Plausible.Cldr.Number.scan(\"100 US dollars\")\n ...> |> Plausible.Cldr.Number.resolve_currencies\n [100, :USD]\n\n iex> Plausible.Cldr.Number.scan(\"100 eurosports\")\n ...> |> Plausible.Cldr.Number.resolve_currencies(fuzzy: 0.75)\n [100, :EUR]\n\n iex> Plausible.Cldr.Number.scan(\"100 dollars des États-Unis\")\n ...> |> Plausible.Cldr.Number.resolve_currencies(locale: \"fr\")\n [100, :USD]","title":"Examples - Plausible.Cldr.Number.resolve_currencies/2","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-examples"},{"type":"function","doc":"Resolve a currency from a string","title":"Plausible.Cldr.Number.resolve_currency/2","ref":"Plausible.Cldr.Number.html#resolve_currency/2"},{"type":"function","doc":"* `list` is any list in which currency\n names and symbols are expected\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Number.resolve_currency/2","ref":"Plausible.Cldr.Number.html#resolve_currency/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `Plausible.Cldr.get_locale()`\n\n* `:only` is an `atom` or list of `atoms` representing the\n currencies or currency types to be considered for a match.\n The equates to a list of acceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:except` is an `atom` or list of `atoms` representing the\n currencies or currency types to be not considered for a match.\n This equates to a list of unacceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:fuzzy` is a float greater than `0.0` and less than or\n equal to `1.0` which is used as input to\n `String.jaro_distance/2` to determine is the provided\n currency string is *close enough* to a known currency\n string for it to identify definitively a currency code.\n It is recommended to use numbers greater than `0.8` in\n order to reduce false positives.","title":"Options - Plausible.Cldr.Number.resolve_currency/2","ref":"Plausible.Cldr.Number.html#resolve_currency/2-options"},{"type":"function","doc":"* An ISO4217 currency code as an atom or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Number.resolve_currency/2","ref":"Plausible.Cldr.Number.html#resolve_currency/2-returns"},{"type":"function","doc":"The `:only` and `:except` options accept a list of\ncurrency codes and/or currency types. The following\ntypes are recognised.\n\nIf both `:only` and `:except` are specified,\nthe `:except` entries take priority - that means\nany entries in `:except` are removed from the `:only`\nentries.\n\n * `:all`, the default, considers all currencies\n\n * `:current` considers those currencies that have a `:to`\n date of nil and which also is a known ISO4217 currency\n\n * `:historic` is the opposite of `:current`\n\n * `:tender` considers currencies that are legal tender\n\n * `:unannotated` considers currencies that don't have\n \"(some string)\" in their names. These are usually\n financial instruments.","title":"Notes - Plausible.Cldr.Number.resolve_currency/2","ref":"Plausible.Cldr.Number.html#resolve_currency/2-notes"},{"type":"function","doc":"iex> Plausible.Cldr.Number.resolve_currency(\"US dollars\")\n [:USD]\n\n iex> Plausible.Cldr.Number.resolve_currency(\"100 eurosports\", fuzzy: 0.75)\n [:EUR]\n\n iex> Plausible.Cldr.Number.resolve_currency(\"dollars des États-Unis\", locale: \"fr\")\n [:USD]\n\n iex> Plausible.Cldr.Number.resolve_currency(\"not a known currency\", locale: \"fr\")\n {:error,\n {Cldr.UnknownCurrencyError,\n \"The currency \\\"not a known currency\\\" is unknown or not supported\"}}","title":"Examples - Plausible.Cldr.Number.resolve_currency/2","ref":"Plausible.Cldr.Number.html#resolve_currency/2-examples"},{"type":"function","doc":"Resolve and tokenize percent or permille\nfrom the beginning and/or the end of a string","title":"Plausible.Cldr.Number.resolve_per/2","ref":"Plausible.Cldr.Number.html#resolve_per/2"},{"type":"function","doc":"* `list` is any list in which percent\n and permille symbols are expected\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Number.resolve_per/2","ref":"Plausible.Cldr.Number.html#resolve_per/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `options[:backend].get_locale()`","title":"Options - Plausible.Cldr.Number.resolve_per/2","ref":"Plausible.Cldr.Number.html#resolve_per/2-options"},{"type":"function","doc":"* An `:percent` or `permille` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Number.resolve_per/2","ref":"Plausible.Cldr.Number.html#resolve_per/2-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.resolve_per \"11%\"\n [\"11\", :percent]\n\n iex> Plausible.Cldr.Number.resolve_per \"% of linguists\"\n [:percent, \" of linguists\"]\n\n iex> Plausible.Cldr.Number.resolve_per \"% of linguists %\"\n [:percent, \" of linguists \", :percent]","title":"Examples - Plausible.Cldr.Number.resolve_per/2","ref":"Plausible.Cldr.Number.html#resolve_per/2-examples"},{"type":"function","doc":"Resolve and tokenize percent and permille\nsybols from strings within a list.\n\nPercent and permille symbols can be identified\nat the beginning and/or the end of a string.","title":"Plausible.Cldr.Number.resolve_pers/2","ref":"Plausible.Cldr.Number.html#resolve_pers/2"},{"type":"function","doc":"* `list` is any list in which percent and\n permille symbols are expected\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Number.resolve_pers/2","ref":"Plausible.Cldr.Number.html#resolve_pers/2-arguments"},{"type":"function","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `t:Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `options[:backend].get_locale()`","title":"Options - Plausible.Cldr.Number.resolve_pers/2","ref":"Plausible.Cldr.Number.html#resolve_pers/2-options"},{"type":"function","doc":"iex> Plausible.Cldr.Number.scan(\"100%\")\n ...> |> Plausible.Cldr.Number.resolve_pers()\n [100, :percent]","title":"Examples - Plausible.Cldr.Number.resolve_pers/2","ref":"Plausible.Cldr.Number.html#resolve_pers/2-examples"},{"type":"function","doc":"Scans a string locale-aware manner and returns\na list of strings and numbers.","title":"Plausible.Cldr.Number.scan/2","ref":"Plausible.Cldr.Number.html#scan/2"},{"type":"function","doc":"* `string` is any `String.t`\n\n* `options` is a keyword list of options","title":"Arguments - Plausible.Cldr.Number.scan/2","ref":"Plausible.Cldr.Number.html#scan/2-arguments"},{"type":"function","doc":"* `:number` is one of `:integer`, `:float`,\n `:decimal` or `nil`. The default is `nil`\n meaning that the type auto-detected as either\n an `integer` or a `float`.\n\n* `:locale` is any locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag.t`. The default is `Plausible.Cldr.get_locale/0`.","title":"Options - Plausible.Cldr.Number.scan/2","ref":"Plausible.Cldr.Number.html#scan/2-options"},{"type":"function","doc":"* A list of strings and numbers","title":"Returns - Plausible.Cldr.Number.scan/2","ref":"Plausible.Cldr.Number.html#scan/2-returns"},{"type":"function","doc":"Number parsing is performed by `Cldr.Number.Parser.parse/2`\nand any options provided are passed to that function.","title":"Notes - Plausible.Cldr.Number.scan/2","ref":"Plausible.Cldr.Number.html#scan/2-notes"},{"type":"function","doc":"iex> Plausible.Cldr.Number.scan(\"£1_000_000.34\")\n [\"£\", 1000000.34]\n\n iex> Plausible.Cldr.Number.scan(\"I want £1_000_000 dollars\")\n [\"I want £\", 1000000, \" dollars\"]\n\n iex> Plausible.Cldr.Number.scan(\"The prize is 23\")\n [\"The prize is \", 23]\n\n iex> Plausible.Cldr.Number.scan(\"The lottery number is 23 for the next draw\")\n [\"The lottery number is \", 23, \" for the next draw\"]\n\n iex> Plausible.Cldr.Number.scan(\"The loss is -1.000 euros\", locale: \"de\", number: :integer)\n [\"The loss is \", -1000, \" euros\"]","title":"Examples - Plausible.Cldr.Number.scan/2","ref":"Plausible.Cldr.Number.html#scan/2-examples"},{"type":"function","doc":"Formats a number and applies the `:approximately` format for\na locale and number system.","title":"Plausible.Cldr.Number.to_approx_string/2","ref":"Plausible.Cldr.Number.html#to_approx_string/2"},{"type":"function","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Cldr.Number.to_string/3` for a description of the available\n options.","title":"Arguments - Plausible.Cldr.Number.to_approx_string/2","ref":"Plausible.Cldr.Number.html#to_approx_string/2-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.to_approx_string 1234\n {:ok, \"~1,234\"}","title":"Example - Plausible.Cldr.Number.to_approx_string/2","ref":"Plausible.Cldr.Number.html#to_approx_string/2-example"},{"type":"function","doc":"Formats a number and applies the `:at_least` format for\na locale and number system.","title":"Plausible.Cldr.Number.to_at_least_string/2","ref":"Plausible.Cldr.Number.html#to_at_least_string/2"},{"type":"function","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Plausible.Cldr.Number.to_string/2` for a description of the available\n options.","title":"Arguments - Plausible.Cldr.Number.to_at_least_string/2","ref":"Plausible.Cldr.Number.html#to_at_least_string/2-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.to_at_least_string 1234\n {:ok, \"1,234+\"}","title":"Example - Plausible.Cldr.Number.to_at_least_string/2","ref":"Plausible.Cldr.Number.html#to_at_least_string/2-example"},{"type":"function","doc":"Formats a number and applies the `:at_most` format for\na locale and number system.","title":"Plausible.Cldr.Number.to_at_most_string/2","ref":"Plausible.Cldr.Number.html#to_at_most_string/2"},{"type":"function","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Cldr.Number.to_string/3` for a description of the available\n options.","title":"Arguments - Plausible.Cldr.Number.to_at_most_string/2","ref":"Plausible.Cldr.Number.html#to_at_most_string/2-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.to_at_most_string 1234\n {:ok, \"≤1,234\"}","title":"Example - Plausible.Cldr.Number.to_at_most_string/2","ref":"Plausible.Cldr.Number.html#to_at_most_string/2-example"},{"type":"function","doc":"Formats the first and last numbers of a range and applies\nthe `:range` format for a locale and number system.","title":"Plausible.Cldr.Number.to_range_string/2","ref":"Plausible.Cldr.Number.html#to_range_string/2"},{"type":"function","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Cldr.Number.to_string/3` for a description of the available\n options.","title":"Arguments - Plausible.Cldr.Number.to_range_string/2","ref":"Plausible.Cldr.Number.html#to_range_string/2-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.to_range_string 1234..5678\n {:ok, \"1,234–5,678\"}","title":"Example - Plausible.Cldr.Number.to_range_string/2","ref":"Plausible.Cldr.Number.html#to_range_string/2-example"},{"type":"function","doc":"Returns a number formatted into a string according to a format pattern and options.","title":"Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2"},{"type":"function","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.","title":"Arguments - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-arguments"},{"type":"function","doc":"* `format`: the format style or a format string defining how the number is\n formatted. See `Cldr.Number.Format` for how format strings can be constructed.\n See `Cldr.Number.Format.format_styles_for/3` to return available format styles\n for a locale. The default `format` is `:standard`.\n\n* If `:format` is set to `:long` or `:short` then the formatting depends on\n whether `:currency` is specified. If not specified then the number is\n formatted as `:decimal_long` or `:decimal_short`. If `:currency` is\n specified the number is formatted as `:currency_long` or\n `:currency_short` and `:fractional_digits` is set to 0 as a default.\n\n* `:format` may also be a format defined by CLDR's Rules Based Number\n Formats (RBNF). Further information is found in the module `Cldr.Rbnf`.\n The most commonly used formats in this category are to spell out the\n number in a the locales language. The applicable formats are `:spellout`,\n `:spellout_year`, `:ordinal`. A number can also be formatted as roman\n numbers by using the format `:roman` or `:roman_lower`.\n\n* `currency`: is the currency for which the number is formatted. For\n available currencies see `Cldr.Currency.known_currencies/0`. This option\n is required if `:format` is set to `:currency`. If `currency` is set\n and no `:format` is set, `:format` will be set to `:currency` as well.\n\n* `currency_symbol`: Allows overriding a currency symbol. The alternatives\n are:\n * `:iso` the ISO currency code will be used instead of the default\n currency symbol.\n * `:narrow` uses the narrow symbol defined for the locale. The same\n narrow symbol can be defined for more than one currency and therefore this\n should be used with care. If no narrow symbol is defined, the standard\n symbol is used.\n * `:symbol` uses the standard symbol defined in CLDR. A symbol is unique\n for each currency and can be safely used.\n * \"string\" uses `string` as the currency symbol\n * `:standard` (the default and recommended) uses the CLDR-defined symbol\n based upon the currency format for the locale.\n\n* `:cash`: a boolean which indicates whether a number being formatted as a\n `:currency` is to be considered a cash value or not. Currencies can be\n rounded differently depending on whether `:cash` is `true` or `false`.\n *This option is deprecated in favour of `currency_digits: :cash`.\n\n* `:currency_digits` indicates which of the rounding and digits should be\n used. The options are `:accounting` which is the default, `:cash` or\n `:iso`\n\n* `:rounding_mode`: determines how a number is rounded to meet the precision\n of the format requested. The available rounding modes are `:down`,\n :half_up, :half_even, :ceiling, :floor, :half_down, :up. The default is\n `:half_even`.\n\n* `:number_system`: determines which of the number systems for a locale\n should be used to define the separators and digits for the formatted\n number. If `number_system` is an `atom` then `number_system` is\n interpreted as a number system. If the `:number_system` is\n `binary` then it is interpreted as a number system name. See\n `Cldr.Number.System.number_system_names_for/2`. The default is `:default`.\n\n* `:locale`: determines the locale in which the number is formatted. See\n `Cldr.known_locale_names/0`. The default is`Cldr.get_locale/0` which is the\n locale currently in affect for this `Process` and which is set by\n `Cldr.put_locale/1`.\n\n* If `:fractional_digits` is set to a positive integer value then the number\n will be rounded to that number of digits and displayed accordingly - overriding\n settings that would be applied by default. For example, currencies have\n fractional digits defined reflecting each currencies minor unit. Setting\n `:fractional_digits` will override that setting.\n\n* If `:maximum_integer_digits` is set to a positive integer value then the\n number is left truncated before formatting. For example if the number `1234`\n is formatted with the option `maximum_integer_digits: 2`, the number is\n truncated to `34` and formatted.\n\n* If `:round_nearest` is set to a positive integer value then the number\n will be rounded to nearest increment of that value - overriding\n settings that would be applied by default.\n\n* `:minimum_grouping_digits` overrides the CLDR definition of minimum grouping\n digits. For example in the locale `es` the number `1234` is formatted by default\n as `1345` because the locale defines the `minimium_grouping_digits` as `2`. If\n `minimum_grouping_digits: 1` is set as an option the number is formatting as\n `1.345`. The `:minimum_grouping_digits` is added to the grouping defined by\n the number format. If the sum of these two digits is greater than the number\n of digits in the integer (or fractional) part of the number then no grouping\n is performed.\n\n* `:wrapper` is a 2-arity function that will be called for each number component\n with parameters `string` and `tag` where `tag` is one of `:number`,\n `:currency_symbol`, `:currency_space`, `:literal`, `:quote`, `:percent`,\n `:permille`, `:minus` or `:plus`. The function must return a string. The\n function can be used to wrap format elements in HTML or other tags.","title":"Options - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-options"},{"type":"function","doc":"A locale identifier can specify options that affect number formatting.\nThese options are:\n\n* `nu`: defines the number system to be used if none is specified by the `:number_system`\n option to `to_string/2`\n\nThis key is part of the [u extension](https://unicode.org/reports/tr35/#u_Extension) and\nthat document should be consulted for details on how to construct a locale identifier with these\nextensions.","title":"Locale extensions affecting formatting - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-locale-extensions-affecting-formatting"},{"type":"function","doc":"Wrapping elements is particularly useful when formatting a number with a\ncurrency symbol and the requirement is to have different HTML formatting\napplied to the symbol than the number. For example:\n\n iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn\n ...> string, :currency_symbol -> \" \" <> string <> \" \"\n ...> string, :number -> \" \" <> string <> \" \"\n ...> string, :currency_space -> \" \" <> string <> \" \"\n ...> string, _other -> string\n ...> end)\n {:ok, \" $ 100.00 \"}\n\nIt is also possible and recommended to use the `Phoenix.HTML.Tag.content_tag/3`\nfunction if wrapping HTML tags since these will ensure HTML entities are\ncorrectly encoded. For example:\n\n iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn\n ...> string, :currency_symbol -> Phoenix.HTML.Tag.content_tag(:span, string, class: \"symbol\")\n ...> string, :number -> Phoenix.HTML.Tag.content_tag(:span, string, class: \"number\")\n ...> string, :currency_space -> Phoenix.HTML.Tag.content_tag(:span, string)\n ...> string, _other -> string\n ...> end)\n {:ok, \" $ 100.00 \"}\n\nWhen formatting a number the format is parsed into format elements that might include\na currency symbol, a literal string, inserted text between a currency symbol and the\ncurrency amount, a percent sign, the number itself and several other elements. In\nsome cases it is helpful to be apply specific formatting to each element.\nThis can be achieved by specifying a `:wrapper` option. This option takes a 2-arity\nfunction as an argument. For each element of the format the wrapper function is called\nwith two parameters: the format element as a string and an atom representing the\nelement type. The wrapper function is required to return a string that is then\ninserted in the final formatted number.","title":"Wrapping format elements - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-wrapping-format-elements"},{"type":"function","doc":"* `{:ok, string}` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.to_string 12345\n {:ok, \"12,345\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, locale: \"fr\"\n {:ok, \"12 345\"}\n\n iex> Plausible.Cldr.Number.to_string 1345.32, currency: :EUR, locale: \"es\", minimum_grouping_digits: 1\n {:ok, \"1.345,32 €\"}\n\n iex> Plausible.Cldr.Number.to_string 1345.32, currency: :EUR, locale: \"es\"\n {:ok, \"1345,32 €\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, locale: \"fr\", currency: \"USD\"\n {:ok, \"12 345,00 $US\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: \"#E0\"\n {:ok, \"1.2345E4\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: \"THB\"\n {:ok, \"THB 12,345.00\"}\n\n iex> Plausible.Cldr.Number.to_string -12345, format: :accounting, currency: \"THB\"\n {:ok, \"(THB 12,345.00)\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: \"THB\",\n ...> locale: \"th\"\n {:ok, \"฿12,345.00\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: \"THB\",\n ...> locale: \"th\", number_system: :native\n {:ok, \"฿๑๒,๓๔๕.๐๐\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :long\n {:ok, \"1 thousand\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :long, currency: \"USD\"\n {:ok, \"1,244 US dollars\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :short\n {:ok, \"1K\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :short, currency: \"EUR\"\n {:ok, \"€1K\"}\n\n iex> Plausible.Cldr.Number.to_string 1234, format: :spellout\n {:ok, \"one thousand two hundred thirty-four\"}\n\n iex> Plausible.Cldr.Number.to_string 1234, format: :spellout_verbose\n {:ok, \"one thousand two hundred and thirty-four\"}\n\n iex> Plausible.Cldr.Number.to_string 1989, format: :spellout_year\n {:ok, \"nineteen eighty-nine\"}\n\n iex> Plausible.Cldr.Number.to_string 123, format: :ordinal\n {:ok, \"123rd\"}\n\n iex> Plausible.Cldr.Number.to_string 123, format: :roman\n {:ok, \"CXXIII\"}\n\n iex> Plausible.Cldr.Number.to_string 123, locale: \"th-u-nu-thai\"\n {:ok, \"๑๒๓\"}","title":"Examples - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-examples"},{"type":"function","doc":"An error tuple `{:error, reason}` will be returned if an error is detected.\nThe two most likely causes of an error return are:\n\n * A format cannot be compiled. In this case the error tuple will look like:\n\n```\n iex> Plausible.Cldr.Number.to_string(12345, format: \"0#\")\n {:error, {Cldr.FormatCompileError,\n \"Decimal format compiler: syntax error before: \\\"#\\\"\"}}\n```\n\n * The format style requested is not defined for the `locale` and\n `number_system`. This happens typically when the number system is\n `:algorithmic` rather than the more common `:numeric`. In this case the error\n return looks like:\n\n```\n iex> Plausible.Cldr.Number.to_string(1234, locale: \"he\", number_system: \"hebr\", format: :percent)\n {:error, {Cldr.UnknownFormatError,\n \"The locale :he with number system :hebr does not define a format :percent\"}}\n```","title":"Errors - Plausible.Cldr.Number.to_string/2","ref":"Plausible.Cldr.Number.html#to_string/2-errors"},{"type":"function","doc":"Same as the execution of `to_string/2` but raises an exception if an error would be\nreturned.","title":"Plausible.Cldr.Number.to_string!/2","ref":"Plausible.Cldr.Number.html#to_string!/2"},{"type":"function","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted. See\n `Plausible.Cldr.Number.to_string/2`","title":"Arguments - Plausible.Cldr.Number.to_string!/2","ref":"Plausible.Cldr.Number.html#to_string!/2-arguments"},{"type":"function","doc":"* a formatted number as a string or\n\n* raises an exception","title":"Returns - Plausible.Cldr.Number.to_string!/2","ref":"Plausible.Cldr.Number.html#to_string!/2-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.to_string! 12345\n \"12,345\"\n\n iex> Plausible.Cldr.Number.to_string! 12345, locale: \"fr\"\n \"12 345\"","title":"Examples - Plausible.Cldr.Number.to_string!/2","ref":"Plausible.Cldr.Number.html#to_string!/2-examples"},{"type":"function","doc":"Return a valid number system from a provided locale and number\nsystem name or type.\n\nThe number system or number system type must be valid for the\ngiven locale. If a number system type is provided, the\nunderlying number system is returned.","title":"Plausible.Cldr.Number.validate_number_system/2","ref":"Plausible.Cldr.Number.html#validate_number_system/2"},{"type":"function","doc":"* `locale` is any valid locale name returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n\n* `system_name` is any number system name returned by\n `Cldr.known_number_systems/0` or a number system type\n returned by `Cldr.known_number_system_types/0`","title":"Arguments - Plausible.Cldr.Number.validate_number_system/2","ref":"Plausible.Cldr.Number.html#validate_number_system/2-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.validate_number_system \"en\", :latn\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.validate_number_system \"en\", :default\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.validate_number_system \"en\", :unknown\n {:error,\n {Cldr.UnknownNumberSystemError, \"The number system :unknown is unknown\"}}\n\n iex> Plausible.Cldr.Number.validate_number_system \"zz\", :default\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zz\\\" is invalid\"}}","title":"Examples - Plausible.Cldr.Number.validate_number_system/2","ref":"Plausible.Cldr.Number.html#validate_number_system/2-examples"},{"type":"module","doc":"Implements cardinal plural rules for numbers.","title":"Plausible.Cldr.Number.Cardinal","ref":"Plausible.Cldr.Number.Cardinal.html"},{"type":"function","doc":"The locale names for which plural rules are defined.","title":"Plausible.Cldr.Number.Cardinal.available_locale_names/0","ref":"Plausible.Cldr.Number.Cardinal.html#available_locale_names/0"},{"type":"function","doc":"The configured locales for which plural rules are defined.\n\nReturns the intersection of `Plausible.Cldr.known_locale_names/0` and\nthe locales for which Cardinal plural rules are defined.\n\nThere are many `Cldr` locales which don't have their own plural\nrules so this list is the intersection of `Cldr`'s configured\nlocales and those that have rules.","title":"Plausible.Cldr.Number.Cardinal.known_locale_names/0","ref":"Plausible.Cldr.Number.Cardinal.html#known_locale_names/0"},{"type":"function","doc":"Return the plural key for a given number in a given locale\n\nReturns which plural key (`:zero`, `:one`, `:two`, `:few`,\n`:many` or `:other`) a given number fits into within the\ncontext of a given locale.\n\nNote that these key names should not be interpreted\nliterally. For example, the key returned from\n`Cldr.Number.Ordinal.plural_rule(0, \"en\")` is actually\n`:other`, not `:zero`.\n\nThis key can then be used to format a number, date, time, unit,\nlist or other content in a plural-sensitive way.","title":"Plausible.Cldr.Number.Cardinal.plural_rule/3","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rule/3"},{"type":"function","doc":"* `number` is any `integer`, `float` or `Decimal`\n\n* `locale` is any locale returned by `Cldr.Locale.new!/2` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `rounding` is one of `[:down, :up, :ceiling, :floor, :half_even, :half_up, :half_down]`. The\n default is `:half_even`.","title":"Arguments - Plausible.Cldr.Number.Cardinal.plural_rule/3","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rule/3-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Cardinal.plural_rule 0, \"fr\"\n :one\n\n iex> Plausible.Cldr.Number.Cardinal.plural_rule 0, \"en\"\n :other","title":"Examples - Plausible.Cldr.Number.Cardinal.plural_rule/3","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rule/3-examples"},{"type":"function","doc":"Returns all the plural rules defined in CLDR.","title":"Plausible.Cldr.Number.Cardinal.plural_rules/0","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rules/0"},{"type":"function","doc":"Return the plural rules for a locale.","title":"Plausible.Cldr.Number.Cardinal.plural_rules_for/1","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rules_for/1"},{"type":"function","doc":"* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\nThe rules are returned in AST form after parsing.","title":"Arguments - Plausible.Cldr.Number.Cardinal.plural_rules_for/1","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rules_for/1-arguments"},{"type":"function","doc":"Pluralize a number using cardinal plural rules\nand a substitution map.","title":"Plausible.Cldr.Number.Cardinal.pluralize/3","ref":"Plausible.Cldr.Number.Cardinal.html#pluralize/3"},{"type":"function","doc":"* `number` is an integer, float or Decimal\n\n* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `substitutions` is a map that maps plural keys to a string.\n The valid substitution keys are `:zero`, `:one`, `:two`,\n `:few`, `:many` and `:other`.\n\nSee also `Plausible.Cldr.Number.Cardinal.Cardinal.plural_rule/3`.","title":"Arguments - Plausible.Cldr.Number.Cardinal.pluralize/3","ref":"Plausible.Cldr.Number.Cardinal.html#pluralize/3-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Cardinal.pluralize 1, \"en\", %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 2, \"en\", %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 2, \"en\", %{one: \"one\", two: \"two\", other: \"other\"}\n \"other\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 22, \"en\", %{one: \"one\", two: \"two\", other: \"other\"}\n \"other\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(1), \"en\", %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), \"en\", %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), \"en\", %{one: \"one\", two: \"two\"}\n nil\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, \"ar\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"few\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, \"en\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"other\"","title":"Examples - Plausible.Cldr.Number.Cardinal.pluralize/3","ref":"Plausible.Cldr.Number.Cardinal.html#pluralize/3-examples"},{"type":"module","doc":"Functions to manage the collection of number patterns defined in Cldr.\n\nNumber patterns affect how numbers are interpreted in a localized context.\nHere are some examples, based on the French locale. The \".\" shows where the\ndecimal point should go. The \",\" shows where the thousands separator should\ngo. A \"0\" indicates zero-padding: if the number is too short, a zero (in the\nlocale's numeric set) will go there. A \"#\" indicates no padding: if the\nnumber is too short, nothing goes there. A \"¤\" shows where the currency sign\nwill go. The following illustrates the effects of different patterns for the\nFrench locale, with the number \"1234.567\". Notice how the pattern characters\n',' and '.' are replaced by the characters appropriate for the locale.","title":"Plausible.Cldr.Number.Format","ref":"Plausible.Cldr.Number.Format.html"},{"type":"module","doc":"| Pattern\t | Currency\t | Text |\n| ------------- | :-------------: | ----------: |\n| #,##0.##\t | n/a\t | 1 234,57 |\n| #,##0.###\t | n/a\t | 1 234,567 |\n| ###0.#####\t | n/a\t | 1234,567 |\n| ###0.0000#\t | n/a\t | 1234,5670 |\n| 00000.0000\t | n/a\t | 01234,5670 |\n| #,##0.00 ¤\t | EUR\t | 1 234,57 € |\n\nThe number of # placeholder characters before the decimal do not matter,\nsince no limit is placed on the maximum number of digits. There should,\nhowever, be at least one zero some place in the pattern. In currency formats,\nthe number of digits after the decimal also do not matter, since the\ninformation in the supplemental data (see Supplemental Currency Data) is used\nto override the number of decimal places — and the rounding — according to\nthe currency that is being formatted. That can be seen in the above chart,\nwith the difference between Yen and Euro formatting.\n\nDetails of the number formats are described in the\n[Unicode documentation](http://unicode.org/reports/tr35/tr35-numbers.html#Number_Format_Patterns)","title":"Number Pattern Examples - Plausible.Cldr.Number.Format","ref":"Plausible.Cldr.Number.Format.html#module-number-pattern-examples"},{"type":"function","doc":"Returns the decimal formats defined for a given locale.","title":"Plausible.Cldr.Number.Format.all_formats_for/1","ref":"Plausible.Cldr.Number.Format.html#all_formats_for/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Arguments - Plausible.Cldr.Number.Format.all_formats_for/1","ref":"Plausible.Cldr.Number.Format.html#all_formats_for/1-arguments"},{"type":"function","doc":"* `{:ok, map}` where map is a map of decimal formats\n keyed by number system or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Number.Format.all_formats_for/1","ref":"Plausible.Cldr.Number.Format.html#all_formats_for/1-returns"},{"type":"function","doc":"Returns the decimal formats defined for a given locale.","title":"Plausible.Cldr.Number.Format.all_formats_for!/1","ref":"Plausible.Cldr.Number.Format.html#all_formats_for!/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Arguments - Plausible.Cldr.Number.Format.all_formats_for!/1","ref":"Plausible.Cldr.Number.Format.html#all_formats_for!/1-arguments"},{"type":"function","doc":"* `{:ok, map}` where map is a map of decimal formats\n keyed by number system or\n\n* raises an exception.\n\nSee `Plausible.Cldr.Number.Format.Number.Format.all_formats_for/1` for further information.","title":"Returns - Plausible.Cldr.Number.Format.all_formats_for!/1","ref":"Plausible.Cldr.Number.Format.html#all_formats_for!/1-returns"},{"type":"function","doc":"Returns the currency space for a given locale and\nnumber system.","title":"Plausible.Cldr.Number.Format.currency_spacing/2","ref":"Plausible.Cldr.Number.Format.html#currency_spacing/2"},{"type":"function","doc":"Returns the list of decimal formats in the configured locales including\nthe list of locales configured for precompilation in `config.exs`.\n\nThis function exists to allow the decimal formatter\nto precompile all the known formats at compile time.","title":"Plausible.Cldr.Number.Format.decimal_format_list/0","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list/0"},{"type":"function","doc":"#=> Plausible.Cldr.Number.Format.Format.decimal_format_list\n [\"#\", \"#,##,##0%\",\n \"#,##,##0.###\", \"#,##,##0.00¤\", \"#,##,##0.00¤;(#,##,##0.00¤)\",\n \"#,##,##0 %\", \"#,##0%\", \"#,##0.###\", \"#,##0.00 ¤\",\n \"#,##0.00 ¤;(#,##0.00 ¤)\", \"#,##0.00¤\", \"#,##0.00¤;(#,##0.00¤)\",\n \"#,##0 %\", \"#0%\", \"#0.######\", \"#0.00 ¤\", \"#E0\", \"%#,##0\", \"% #,##0\",\n \"0\", \"0.000000E+000\", \"0000 M ¤\", \"0000¤\", \"000G ¤\", \"000K ¤\", \"000M ¤\",\n \"000T ¤\", \"000mM ¤\", \"000m ¤\", \"000 Bio'.' ¤\", \"000 Bln ¤\", \"000 Bn ¤\",\n \"000 B ¤\", \"000 E ¤\", \"000 K ¤\", \"000 MRD ¤\", \"000 Md ¤\", \"000 Mio'.' ¤\",\n \"000 Mio ¤\", \"000 Mld ¤\", \"000 Mln ¤\", \"000 Mn ¤\", \"000 Mrd'.' ¤\",\n \"000 Mrd ¤\", \"000 Mr ¤\", \"000 M ¤\", \"000 NT ¤\", \"000 N ¤\", \"000 Tn ¤\",\n \"000 Tr ¤\", ...]","title":"Example - Plausible.Cldr.Number.Format.decimal_format_list/0","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list/0-example"},{"type":"function","doc":"Returns the list of decimal formats for a configured locale.","title":"Plausible.Cldr.Number.Format.decimal_format_list_for/1","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list_for/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.\n\nThis function exists to allow the decimal formatter to precompile all\nthe known formats at compile time. Its use is not otherwise recommended.","title":"Arguments - Plausible.Cldr.Number.Format.decimal_format_list_for/1","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list_for/1-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Format.decimal_format_list_for(:en)\n {:ok, [\"#,##0%\", \"#,##0.###\", \"#,##0.00\", \"#,##0.00;(#,##0.00)\",\"#E0\",\n \"0 billion\", \"0 million\", \"0 thousand\",\n \"0 trillion\", \"00 billion\", \"00 million\", \"00 thousand\", \"00 trillion\",\n \"000 billion\", \"000 million\", \"000 thousand\", \"000 trillion\", \"000B\", \"000K\",\n \"000M\", \"000T\", \"00B\", \"00K\", \"00M\", \"00T\", \"0B\", \"0K\", \"0M\", \"0T\",\n \"¤#,##0.00\", \"¤#,##0.00;(¤#,##0.00)\", \"¤000B\", \"¤000K\", \"¤000M\",\n \"¤000T\", \"¤00B\", \"¤00K\", \"¤00M\", \"¤00T\", \"¤0B\", \"¤0K\", \"¤0M\", \"¤0T\",\n \"¤ #,##0.00\", \"¤ #,##0.00;(¤ #,##0.00)\", \"¤ 000B\", \"¤ 000K\", \"¤ 000M\",\n \"¤ 000T\", \"¤ 00B\", \"¤ 00K\", \"¤ 00M\", \"¤ 00T\", \"¤ 0B\", \"¤ 0K\", \"¤ 0M\", \"¤ 0T\"]}","title":"Example - Plausible.Cldr.Number.Format.decimal_format_list_for/1","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list_for/1-example"},{"type":"function","doc":"Returns the default grouping for a locale as a map.","title":"Plausible.Cldr.Number.Format.default_grouping_for/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Arguments - Plausible.Cldr.Number.Format.default_grouping_for/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1-arguments"},{"type":"function","doc":"* `{:ok, grouping}` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Number.Format.default_grouping_for/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Format.default_grouping_for(:en)\n {:ok, %{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}}","title":"Examples - Plausible.Cldr.Number.Format.default_grouping_for/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1-examples"},{"type":"function","doc":"Returns the default grouping for a locale\nor raises on error.","title":"Plausible.Cldr.Number.Format.default_grouping_for!/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Arguments - Plausible.Cldr.Number.Format.default_grouping_for!/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1-arguments"},{"type":"function","doc":"* `grouping` as a map or\n\n* raises an exception.","title":"Returns - Plausible.Cldr.Number.Format.default_grouping_for!/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Format.default_grouping_for!(:en)\n %{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}","title":"Examples - Plausible.Cldr.Number.Format.default_grouping_for!/1","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1-examples"},{"type":"function","doc":"Return the predfined formats for a given `locale` and `number_system`.","title":"Plausible.Cldr.Number.Format.formats_for/2","ref":"Plausible.Cldr.Number.Format.html#formats_for/2"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.\n\n* `number_system` is any valid number system or number system type returned\n by `Plausible.Cldr.Number.System.number_systems_for/1`.","title":"Arguments - Plausible.Cldr.Number.Format.formats_for/2","ref":"Plausible.Cldr.Number.Format.html#formats_for/2-arguments"},{"type":"function","doc":"Plausible.Cldr.Number.Format.formats_for :fr, :native\n #=> %Cldr.Number.Format{\n accounting: \"#,##0.00 ¤;(#,##0.00 ¤)\",\n currency: \"#,##0.00 ¤\",\n percent: \"#,##0 %\",\n scientific: \"#E0\",\n standard: \"#,##0.###\"\n currency_short: [{\"1000\", [one: \"0 k ¤\", other: \"0 k ¤\"]},\n {\"10000\", [one: \"00 k ¤\", other: \"00 k ¤\"]},\n {\"100000\", [one: \"000 k ¤\", other: \"000 k ¤\"]},\n {\"1000000\", [one: \"0 M ¤\", other: \"0 M ¤\"]},\n {\"10000000\", [one: \"00 M ¤\", other: \"00 M ¤\"]},\n {\"100000000\", [one: \"000 M ¤\", other: \"000 M ¤\"]},\n {\"1000000000\", [one: \"0 Md ¤\", other: \"0 Md ¤\"]},\n {\"10000000000\", [one: \"00 Md ¤\", other: \"00 Md ¤\"]},\n {\"100000000000\", [one: \"000 Md ¤\", other: \"000 Md ¤\"]},\n {\"1000000000000\", [one: \"0 Bn ¤\", other: \"0 Bn ¤\"]},\n {\"10000000000000\", [one: \"00 Bn ¤\", other: \"00 Bn ¤\"]},\n {\"100000000000000\", [one: \"000 Bn ¤\", other: \"000 Bn ¤\"]}],\n ...\n }","title":"Example - Plausible.Cldr.Number.Format.formats_for/2","ref":"Plausible.Cldr.Number.Format.html#formats_for/2-example"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.Format.formats_for!/2","ref":"Plausible.Cldr.Number.Format.html#formats_for!/2"},{"type":"function","doc":"Returns the minimum grouping digits for a locale.","title":"Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Arguments - Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1-arguments"},{"type":"function","doc":"* `{:ok, minumum_digits}` or\n\n* `{:error, {exception, message}}`","title":"Returns - Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for(\"en\")\n {:ok, 1}","title":"Examples - Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1-examples"},{"type":"function","doc":"Returns the minimum grouping digits for a locale\nor raises on error.","title":"Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Arguments - Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1-arguments"},{"type":"function","doc":"* `minumum_digits` or\n\n* raises an exception.","title":"Returns - Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for!(\"en\")\n 1","title":"Examples - Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1-examples"},{"type":"module","doc":"","title":"Plausible.Cldr.Number.Formatter.Decimal","ref":"Plausible.Cldr.Number.Formatter.Decimal.html"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.Formatter.Decimal.metadata!/1","ref":"Plausible.Cldr.Number.Formatter.Decimal.html#metadata!/1"},{"type":"function","doc":"Formats a number according to a decimal format string.","title":"Plausible.Cldr.Number.Formatter.Decimal.to_string/3","ref":"Plausible.Cldr.Number.Formatter.Decimal.html#to_string/3"},{"type":"function","doc":"* `number` is an integer, float or Decimal\n\n* `format` is a format string. See `Plausible.Cldr.Number` for further information.\n\n* `options` is a map of options. See `Plausible.Cldr.Number.to_string/2`\n for further information.","title":"Arguments - Plausible.Cldr.Number.Formatter.Decimal.to_string/3","ref":"Plausible.Cldr.Number.Formatter.Decimal.html#to_string/3-arguments"},{"type":"module","doc":"Implements ordinal plural rules for numbers.","title":"Plausible.Cldr.Number.Ordinal","ref":"Plausible.Cldr.Number.Ordinal.html"},{"type":"function","doc":"The locale names for which plural rules are defined.","title":"Plausible.Cldr.Number.Ordinal.available_locale_names/0","ref":"Plausible.Cldr.Number.Ordinal.html#available_locale_names/0"},{"type":"function","doc":"The configured locales for which plural rules are defined.\n\nReturns the intersection of `Plausible.Cldr.known_locale_names/0` and\nthe locales for which Ordinal plural rules are defined.\n\nThere are many `Cldr` locales which don't have their own plural\nrules so this list is the intersection of `Cldr`'s configured\nlocales and those that have rules.","title":"Plausible.Cldr.Number.Ordinal.known_locale_names/0","ref":"Plausible.Cldr.Number.Ordinal.html#known_locale_names/0"},{"type":"function","doc":"Return the plural key for a given number in a given locale\n\nReturns which plural key (`:zero`, `:one`, `:two`, `:few`,\n`:many` or `:other`) a given number fits into within the\ncontext of a given locale.\n\nNote that these key names should not be interpreted\nliterally. For example, the key returned from\n`Cldr.Number.Ordinal.plural_rule(0, \"en\")` is actually\n`:other`, not `:zero`.\n\nThis key can then be used to format a number, date, time, unit,\nlist or other content in a plural-sensitive way.","title":"Plausible.Cldr.Number.Ordinal.plural_rule/3","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rule/3"},{"type":"function","doc":"* `number` is any `integer`, `float` or `Decimal`\n\n* `locale` is any locale returned by `Cldr.Locale.new!/2` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `rounding` is one of `[:down, :up, :ceiling, :floor, :half_even, :half_up, :half_down]`. The\n default is `:half_even`.","title":"Arguments - Plausible.Cldr.Number.Ordinal.plural_rule/3","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rule/3-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Ordinal.plural_rule 0, \"fr\"\n :other\n\n iex> Plausible.Cldr.Number.Ordinal.plural_rule 1, \"en\"\n :one","title":"Examples - Plausible.Cldr.Number.Ordinal.plural_rule/3","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rule/3-examples"},{"type":"function","doc":"Returns all the plural rules defined in CLDR.","title":"Plausible.Cldr.Number.Ordinal.plural_rules/0","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rules/0"},{"type":"function","doc":"Return the plural rules for a locale.","title":"Plausible.Cldr.Number.Ordinal.plural_rules_for/1","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rules_for/1"},{"type":"function","doc":"* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\nThe rules are returned in AST form after parsing.","title":"Arguments - Plausible.Cldr.Number.Ordinal.plural_rules_for/1","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rules_for/1-arguments"},{"type":"function","doc":"Pluralize a number using ordinal plural rules\nand a substitution map.","title":"Plausible.Cldr.Number.Ordinal.pluralize/3","ref":"Plausible.Cldr.Number.Ordinal.html#pluralize/3"},{"type":"function","doc":"* `number` is an integer, float or Decimal or a `Range.t{}`. When a range, The\n is that in any usage, the start value is strictly less than the end value,\n and that no values are negative. Results for any cases that do not meet\n these criteria are undefined.\n\n* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `substitutions` is a map that maps plural keys to a string.\n The valid substitution keys are `:zero`, `:one`, `:two`,\n `:few`, `:many` and `:other`.\n\nSee also `Plausible.Cldr.Number.Ordinal.Ordinal.plural_rule/3`.","title":"Arguments - Plausible.Cldr.Number.Ordinal.pluralize/3","ref":"Plausible.Cldr.Number.Ordinal.html#pluralize/3-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Ordinal.pluralize 1, :en, %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: \"one\", two: \"two\"}\n \"two\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 22, :en, %{one: \"one\", two: \"two\", other: \"other\"}\n \"two\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(1), :en, %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: \"one\", two: \"two\"}\n \"two\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, \"ar\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"other\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, \"en\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"other\"","title":"Examples - Plausible.Cldr.Number.Ordinal.pluralize/3","ref":"Plausible.Cldr.Number.Ordinal.html#pluralize/3-examples"},{"type":"module","doc":"Implements plural rules for ranges","title":"Plausible.Cldr.Number.PluralRule.Range","ref":"Plausible.Cldr.Number.PluralRule.Range.html"},{"type":"function","doc":"Returns a final plural type for a start-of-range plural\ntype, an end-of-range plural type and a locale.","title":"Plausible.Cldr.Number.PluralRule.Range.plural_rule/3","ref":"Plausible.Cldr.Number.PluralRule.Range.html#plural_rule/3"},{"type":"function","doc":"* `first` is a plural type for the start of a range\n\n* `last` is a plural type for the end of a range\n\n* `locale` is any `Cldr.LanguageTag.t` or a language name\n (not locale name)","title":"Arguments - Plausible.Cldr.Number.PluralRule.Range.plural_rule/3","ref":"Plausible.Cldr.Number.PluralRule.Range.html#plural_rule/3-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.PluralRule.Range.plural_rule :other, :few, \"ar\"\n :few","title":"Example - Plausible.Cldr.Number.PluralRule.Range.plural_rule/3","ref":"Plausible.Cldr.Number.PluralRule.Range.html#plural_rule/3-example"},{"type":"module","doc":"","title":"Plausible.Cldr.Number.Symbol","ref":"Plausible.Cldr.Number.Symbol.html"},{"type":"function","doc":"Returns a list of all decimal symbols defined\nby the locales configured in this backend as\na list.","title":"Plausible.Cldr.Number.Symbol.all_decimal_symbols/0","ref":"Plausible.Cldr.Number.Symbol.html#all_decimal_symbols/0"},{"type":"function","doc":"Returns a list of all decimal symbols defined\nby the locales configured in this backend as\na string.\n\nThis string can be used as a character class\nwhen builing a regular expression.","title":"Plausible.Cldr.Number.Symbol.all_decimal_symbols_class/0","ref":"Plausible.Cldr.Number.Symbol.html#all_decimal_symbols_class/0"},{"type":"function","doc":"Returns a list of all grouping symbols defined\nby the locales configured in this backend as\na list.","title":"Plausible.Cldr.Number.Symbol.all_grouping_symbols/0","ref":"Plausible.Cldr.Number.Symbol.html#all_grouping_symbols/0"},{"type":"function","doc":"Returns a list of all grouping symbols defined\nby the locales configured in this backend as\na string.\n\nThis string can be used as a character class\nwhen builing a regular expression.","title":"Plausible.Cldr.Number.Symbol.all_grouping_symbols_class/0","ref":"Plausible.Cldr.Number.Symbol.html#all_grouping_symbols_class/0"},{"type":"function","doc":"Returns a map of `Cldr.Number.Symbol.t` structs of the number symbols for each\nof the number systems of a locale.","title":"Plausible.Cldr.Number.Symbol.number_symbols_for/1","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/1"},{"type":"function","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","title":"Options - Plausible.Cldr.Number.Symbol.number_symbols_for/1","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/1-options"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Symbol.number_symbols_for(:th)\n {:ok, %{\n latn: %Cldr.Number.Symbol{\n decimal: \".\",\n exponential: \"E\",\n group: \",\",\n infinity: \"∞\",\n list: \";\",\n minus_sign: \"-\",\n nan: \"NaN\",\n per_mille: \"‰\",\n percent_sign: \"%\",\n plus_sign: \"+\",\n superscripting_exponent: \"×\",\n time_separator: \":\"\n },\n thai: %Cldr.Number.Symbol{\n decimal: \".\",\n exponential: \"E\",\n group: \",\",\n infinity: \"∞\",\n list: \";\",\n minus_sign: \"-\",\n nan: \"NaN\",\n per_mille: \"‰\",\n percent_sign: \"%\",\n plus_sign: \"+\",\n superscripting_exponent: \"×\",\n time_separator: \":\"\n }\n }}","title":"Example: - Plausible.Cldr.Number.Symbol.number_symbols_for/1","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/1-example"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.Symbol.number_symbols_for/2","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/2"},{"type":"module","doc":"","title":"Plausible.Cldr.Number.System","ref":"Plausible.Cldr.Number.System.html"},{"type":"function","doc":"Returns the actual number system from a number system type.\n\n* `locale` is any valid locale name returned by `Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by ``Cldr.Locale.new!/2``\n\n* `system_name` is any number system name returned by\n `Cldr.known_number_systems/0` or a number system type\n returned by `Cldr.known_number_system_types/0`\n\nThis function will decode a number system type into the actual\nnumber system. If the number system provided can't be decoded\nit is returned as is.","title":"Plausible.Cldr.Number.System.number_system_for/2","ref":"Plausible.Cldr.Number.System.html#number_system_for/2"},{"type":"function","doc":"iex> Plausible.Cldr.Number.System.number_system_for \"th\", :latn\n {:ok, %{digits: \"0123456789\", type: :numeric}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"en\", :default\n {:ok, %{digits: \"0123456789\", type: :numeric}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"he\", :traditional\n {:ok, %{rules: \"hebrew\", type: :algorithmic}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"en\", :native\n {:ok, %{digits: \"0123456789\", type: :numeric}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"en\", :finance\n {\n :error,\n {Cldr.UnknownNumberSystemError,\n \"The number system :finance is unknown for the locale named :en. Valid number systems are %{default: :latn, native: :latn}\"}\n }","title":"Examples - Plausible.Cldr.Number.System.number_system_for/2","ref":"Plausible.Cldr.Number.System.html#number_system_for/2-examples"},{"type":"function","doc":"Returns the number system from a language tag or\nlocale name.","title":"Plausible.Cldr.Number.System.number_system_from_locale/1","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1"},{"type":"function","doc":"* `locale` is any language tag returned be `Cldr.Locale.new/2`\n or a locale name in the list returned by `Cldr.known_locale_names/1`","title":"Arguments - Plausible.Cldr.Number.System.number_system_from_locale/1","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1-arguments"},{"type":"function","doc":"* A number system name as an atom","title":"Returns - Plausible.Cldr.Number.System.number_system_from_locale/1","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1-returns"},{"type":"function","doc":"iex> Plausible.Cldr.Number.System.number_system_from_locale \"en-US-u-nu-thai\"\n :thai\n\n iex> Plausible.Cldr.Number.System.number_system_from_locale \"en-US\"\n :latn","title":"Examples - Plausible.Cldr.Number.System.number_system_from_locale/1","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1-examples"},{"type":"function","doc":"Returns the number systems available for a locale\nor `{:error, message}` if the locale is not known.\n\n* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`","title":"Plausible.Cldr.Number.System.number_system_names_for/1","ref":"Plausible.Cldr.Number.System.html#number_system_names_for/1"},{"type":"function","doc":"iex> Plausible.Cldr.Number.System.number_system_names_for \"en\"\n {:ok, [:latn]}\n\n iex> Plausible.Cldr.Number.System.number_system_names_for \"zz\"\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zz\\\" is invalid\"}}","title":"Examples - Plausible.Cldr.Number.System.number_system_names_for/1","ref":"Plausible.Cldr.Number.System.html#number_system_names_for/1-examples"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.System.number_system_names_for!/1","ref":"Plausible.Cldr.Number.System.html#number_system_names_for!/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.System.number_system_types_for/1","ref":"Plausible.Cldr.Number.System.html#number_system_types_for/1"},{"type":"function","doc":"Returns the number systems available for a locale\nor `{:error, message}` if the locale is not known.\n\n* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`","title":"Plausible.Cldr.Number.System.number_systems_for/1","ref":"Plausible.Cldr.Number.System.html#number_systems_for/1"},{"type":"function","doc":"iex> Plausible.Cldr.Number.System.number_systems_for \"en\"\n {:ok, %{default: :latn, native: :latn}}\n\n iex> Plausible.Cldr.Number.System.number_systems_for \"th\"\n {:ok, %{default: :latn, native: :thai}}\n\n iex> Plausible.Cldr.Number.System.number_systems_for \"zz\"\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zz\\\" is invalid\"}}","title":"Examples - Plausible.Cldr.Number.System.number_systems_for/1","ref":"Plausible.Cldr.Number.System.html#number_systems_for/1-examples"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.System.number_systems_for!/1","ref":"Plausible.Cldr.Number.System.html#number_systems_for!/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.System.number_systems_like/2","ref":"Plausible.Cldr.Number.System.html#number_systems_like/2"},{"type":"function","doc":"Returns a number system name for a given locale and number system reference.\n\n* `system_name` is any number system name returned by\n `Plausible.Cldr.known_number_systems/0` or a number system type\n returned by `Plausible.Cldr.known_number_system_types/0`\n\n* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`\n\nNumber systems can be references in one of two ways:\n\n* As a number system type such as :default, :native, :traditional and\n :finance. This allows references to a number system for a locale in a\n consistent fashion for a given use\n\n* WIth the number system name directly, such as :latn, :arab or any of the\n other 70 or so\n\nThis function dereferences the supplied `system_name` and returns the\nactual system name.","title":"Plausible.Cldr.Number.System.system_name_from/2","ref":"Plausible.Cldr.Number.System.html#system_name_from/2"},{"type":"function","doc":"ex> Plausible.Cldr.Number.System.system_name_from(:default, \"en\")\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.System.system_name_from(\"latn\", \"en\")\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.System.system_name_from(:native, \"en\")\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.System.system_name_from(:nope, \"en\")\n {\n :error,\n {Cldr.UnknownNumberSystemError, \"The number system :nope is unknown\"}\n }\n\nNote that return value is not guaranteed to be a valid\nnumber system for the given locale as demonstrated in the third example.","title":"Examples - Plausible.Cldr.Number.System.system_name_from/2","ref":"Plausible.Cldr.Number.System.html#system_name_from/2-examples"},{"type":"function","doc":"Converts a number into the representation of\na non-latin number system.\n\nThis function converts numbers to a known\nnumber system only, it does not provide number\nformatting.\n\n* `number` is a `float`, `integer` or `Decimal`\n\n* `system_name` is any number system name returned by\n `Cldr.known_number_systems/0` or a number system type\n returned by `Cldr.known_number_system_types/0`\n\nThere are two types of number systems in CLDR:\n\n* `:numeric` in which the number system defines\n a direct mapping between the latin digits `0..9`\n into a the number system equivalent. In this case,\n` to_system/2` invokes `Cldr.Number.Transliterate.transliterate_digits/3`\n for the given number.\n\n* `:algorithmic` in which the number system\n does not have the same structure as the `:latn`\n number system and therefore the conversion is\n done algorithmically. For CLDR the algorithm\n is implemented through `Cldr.Rbnf` rulesets.\n These rulesets are considered by CLDR to be\n less rigorous than the `:numeric` number systems\n and caution and testing for a specific use case\n is recommended.","title":"Plausible.Cldr.Number.System.to_system/2","ref":"Plausible.Cldr.Number.System.html#to_system/2"},{"type":"function","doc":"iex> Plausible.Cldr.Number.System.to_system 123456, :hebr\n {:ok, \"קכ״ג׳תנ״ו\"}\n\n iex> Plausible.Cldr.Number.System.to_system 123, :hans\n {:ok, \"一百二十三\"}\n\n iex> Plausible.Cldr.Number.System.to_system 123, :hant\n {:ok, \"一百二十三\"}\n\n iex> Plausible.Cldr.Number.System.to_system 123, :hansfin\n {:ok, \"壹佰贰拾叁\"}","title":"Examples - Plausible.Cldr.Number.System.to_system/2","ref":"Plausible.Cldr.Number.System.html#to_system/2-examples"},{"type":"function","doc":"Converts a number into the representation of\na non-latin number system. Returns a converted\nstring or raises on error.\n\n* `number` is a `float`, `integer` or `Decimal`\n\n* `system_name` is any number system name returned by\n `Plausible.Cldr.known_number_systems/0` or a number system type\n returned by `Plausible.Cldr.known_number_system_types/0`\n\nSee `Plausible.Cldr.Number.System.to_system/2` for further\ninformation.","title":"Plausible.Cldr.Number.System.to_system!/2","ref":"Plausible.Cldr.Number.System.html#to_system!/2"},{"type":"function","doc":"iex> Plausible.Cldr.Number.System.to_system! 123, :hans\n \"一百二十三\"\n\n iex> Plausible.Cldr.Number.System.to_system! 123, :hant\n \"一百二十三\"\n\n iex> Plausible.Cldr.Number.System.to_system! 123, :hansfin\n \"壹佰贰拾叁\"","title":"Examples - Plausible.Cldr.Number.System.to_system!/2","ref":"Plausible.Cldr.Number.System.html#to_system!/2-examples"},{"type":"module","doc":"Transliteration for digits and separators.\n\nTransliterating a string is an expensive business. First the string has to\nbe exploded into its component graphemes. Then for each grapheme we have\nto map to the equivalent in the other `{locale, number_system}`. Then we\nhave to reassemble the string.\n\nEffort is made to short circuit where possible. Transliteration is not\nrequired for any `{locale, number_system}` that is the same as `{\"en\",\n\"latn\"}` since the implementation uses this combination for the placeholders during\nformatting already. When short circuiting is possible (typically the en-*\nlocales with \"latn\" number_system - the total number of short circuited\nlocales is 211 of the 537 in CLDR) the overall number formatting is twice as\nfast than when formal transliteration is required.\n\n#","title":"Plausible.Cldr.Number.Transliterate","ref":"Plausible.Cldr.Number.Transliterate.html"},{"type":"module","doc":"This module includes `Cldr.Number.Transliterate.transliterate_digits/3` which transliterates\ndigits between number systems. For example from :arabic to :latn. Since generating a\ntransliteration map is slow, pairs of transliterations can be configured so that the\ntransliteration map is created at compile time and therefore speeding up transliteration at\nrun time.\n\nTo configure these transliteration pairs, add the to the `use Cldr` configuration\nin a backend module:\n\n defmodule MyApp.Cldr do\n use Cldr,\n locale: [\"en\", \"fr\", \"th\"],\n default_locale: \"en\",\n precompile_transliterations: [{:latn, :thai}, {:arab, :thai}]\n end\n\nWhere each tuple in the list configures one transliteration map. In this example, two maps are\nconfigured: from `:latn` to `:thai` and from `:arab` to `:thai`.\n\nA list of configurable number systems is returned by `Cldr.Number.System.numeric_systems/0`.\n\nIf a transliteration is requested between two number pairs that have not been configured for\nprecompilation, a warning is logged.","title":"Configuring precompilation of digit transliterations - Plausible.Cldr.Number.Transliterate","ref":"Plausible.Cldr.Number.Transliterate.html#module-configuring-precompilation-of-digit-transliterations"},{"type":"function","doc":"Transliterates from latin digits to another number system's digits.\n\nTransliterates the latin digits 0..9 to their equivalents in\nanother number system. Also transliterates the decimal and grouping\nseparators as well as the plus, minus and exponent symbols. Any other character\nin the string will be returned \"as is\".","title":"Plausible.Cldr.Number.Transliterate.transliterate/3","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate/3"},{"type":"function","doc":"* `sequence` is the string to be transliterated.\n\n* `locale` is any known locale, defaulting to `Plausible.Cldr.get_locale/0`.\n\n* `number_system` is any known number system. If expressed as a `string` it\n is the actual name of a known number system. If epressed as an `atom` it is\n used as a key to look up a number system for the locale (the usual keys are\n `:default` and `:native` but :traditional and :finance are also part of the\n standard). See `Plausible.Cldr.Number.System.number_systems_for/1` for a locale to\n see what number system types are defined. The default is `:default`.\n\nFor available number systems see `Cldr.Number.System.number_systems/0`\nand `Plausible.Cldr.Number.System.number_systems_for/1`. Also see\n`Plausible.Cldr.Number.Symbol.number_symbols_for/1`.","title":"Arguments - Plausible.Cldr.Number.Transliterate.transliterate/3","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate/3-arguments"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\")\n \"123556\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123,556.000\", \"fr\", :default)\n \"123 556,000\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\", \"th\", :default)\n \"123556\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\", \"th\", \"thai\")\n \"๑๒๓๕๕๖\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\", \"th\", :native)\n \"๑๒๓๕๕๖\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"Some number is: 123556\", \"th\", \"thai\")\n \"Some number is: ๑๒๓๕๕๖\"","title":"Examples - Plausible.Cldr.Number.Transliterate.transliterate/3","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate/3-examples"},{"type":"function","doc":"","title":"Plausible.Cldr.Number.Transliterate.transliterate!/3","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate!/3"},{"type":"function","doc":"Transliterates digits from one number system to another number system\n\n* `digits` is binary representation of a number\n\n* `from_system` and `to_system` are number system names in atom form. See\n`Cldr.Number.System.numeric_systems/0` for available number systems.","title":"Plausible.Cldr.Number.Transliterate.transliterate_digits/3","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate_digits/3"},{"type":"function","doc":"iex> Plausible.Cldr.Number.Transliterate.transliterate_digits \"٠١٢٣٤٥٦٧٨٩\", :arab, :latn\n \"0123456789\"","title":"Example - Plausible.Cldr.Number.Transliterate.transliterate_digits/3","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate_digits/3-example"},{"type":"module","doc":"Functions to implement the number system rule-based-number-format rules of CLDR.\n\nThese rules are defined only on the \"und\" locale and represent specialised\nnumber formatting.\n\nThe standard public API for RBNF is via the `Cldr.Number.to_string/2` function.\n\nThe functions on this module are defined at compile time based upon the RBNF rules\ndefined in the Unicode CLDR data repository. Available rules are identified by:\n\n iex> Plausible.Cldr.Rbnf.NumberSystem.rule_sets(:und)\n ...> |> Enum.sort()\n [\n :armenian_lower,\n :armenian_upper,\n :cyrillic_lower,\n :ethiopic,\n :georgian,\n :greek_lower,\n :greek_upper,\n :hebrew,\n :hebrew_item,\n :roman_lower,\n :roman_upper,\n :tamil,\n :zz_default\n ]\n\nA rule can then be invoked on an available rule_set. For example\n\n iex> Plausible.Cldr.Rbnf.NumberSystem.roman_upper(123, :und)\n \"CXXIII\"\n\nThis particular call is equivalent to the call through the public API of:\n\n iex> Plausible.Cldr.Number.to_string(123, format: :roman)\n {:ok, \"CXXIII\"}","title":"Plausible.Cldr.Rbnf.NumberSystem","ref":"Plausible.Cldr.Rbnf.NumberSystem.html"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.all_rule_sets/0","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#all_rule_sets/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_lower/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_lower/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_lower/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_lower/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_upper/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_upper/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_upper/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_upper/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_1_10/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_1_10/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_final/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_final/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_post/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_post/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_thousands/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_thousands/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p1/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p1/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p2/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p2/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p3/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p3/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.georgian/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#georgian/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.georgian/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#georgian/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_lower/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_lower/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_lower/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_lower/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_numeral_majuscules/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_numeral_majuscules/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_numeral_minuscules/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_numeral_minuscules/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_upper/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_upper/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_upper/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_upper/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_0_99/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_0_99/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_item/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_item/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_item/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_item/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_item_hundreds/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_item_hundreds/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_thousands/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_thousands/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_lower/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_lower/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_lower/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_lower/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_upper/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_upper/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_upper/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_upper/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.rule_sets/0","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#rule_sets/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.rule_sets/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#rule_sets/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.tamil/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#tamil/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.tamil/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#tamil/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.tamil_thousands/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#tamil_thousands/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.zz_default/1","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#zz_default/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.NumberSystem.zz_default/2","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#zz_default/2"},{"type":"module","doc":"Functions to implement the ordinal rule-based-number-format rules of CLDR.\n\nAs CLDR notes, the data is incomplete or non-existent for many languages. It\nis considered complete for English however.\n\nThe standard public API for RBNF is via the `Cldr.Number.to_string/2` function.\n\nThe functions on this module are defined at compile time based upon the RBNF rules\ndefined in the Unicode CLDR data repository. Available rules are identified by:\n\n iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets(:en)\n [:digits_ordinal]\n\n iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets(\"fr\")\n ...> |> Enum.sort()\n [\n :digits_ordinal,\n :digits_ordinal_feminine,\n :digits_ordinal_feminine_plural,\n :digits_ordinal_masculine,\n :digits_ordinal_masculine_plural\n ]\n\nA rule can then be invoked on an available rule_set. For example\n\n iex> Plausible.Cldr.Rbnf.Ordinal.digits_ordinal(123, :en)\n \"123rd\"\n\nThis call is equivalent to the call through the public API of:\n\n iex> Plausible.Cldr.Number.to_string(123, format: :ordinal)\n {:ok, \"123rd\"}","title":"Plausible.Cldr.Rbnf.Ordinal","ref":"Plausible.Cldr.Rbnf.Ordinal.html"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Ordinal.all_rule_sets/0","ref":"Plausible.Cldr.Rbnf.Ordinal.html#all_rule_sets/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Ordinal.digits_ordinal/2","ref":"Plausible.Cldr.Rbnf.Ordinal.html#digits_ordinal/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Ordinal.rule_sets/0","ref":"Plausible.Cldr.Rbnf.Ordinal.html#rule_sets/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Ordinal.rule_sets/1","ref":"Plausible.Cldr.Rbnf.Ordinal.html#rule_sets/1"},{"type":"module","doc":"Functions to implement the spellout rule-based-number-format rules of CLDR.\n\nAs CLDR notes, the data is incomplete or non-existent for many languages. It\nis considered complete for English however.\n\nThe standard public API for RBNF is via the `Cldr.Number.to_string/2` function.\n\nThe functions on this module are defined at compile time based upon the RBNF rules\ndefined in the Unicode CLDR data repository. Available rules are identified by:\n\n iex> Plausible.Cldr.Rbnf.Spellout.rule_sets(\"en\")\n ...> |> Enum.sort()\n [\n :spellout_cardinal,\n :spellout_cardinal_verbose,\n :spellout_numbering,\n :spellout_numbering_verbose,\n :spellout_numbering_year,\n :spellout_ordinal,\n :spellout_ordinal_verbose\n ]\n\nA rule can then be invoked on an available rule_set. For example:\n\n iex> Plausible.Cldr.Rbnf.Spellout.spellout_ordinal(123, \"en\")\n \"one hundred twenty-third\"\n\nThis call is equivalent to the call through the public API of:\n\n iex> Plausible.Cldr.Number.to_string(123, format: :spellout)\n {:ok, \"one hundred twenty-three\"}","title":"Plausible.Cldr.Rbnf.Spellout","ref":"Plausible.Cldr.Rbnf.Spellout.html"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.all_rule_sets/0","ref":"Plausible.Cldr.Rbnf.Spellout.html#all_rule_sets/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.and/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#and/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.and_o/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#and_o/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.commas/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#commas/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.commas_o/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#commas_o/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.r2d_year/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#r2d_year/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.rule_sets/0","ref":"Plausible.Cldr.Rbnf.Spellout.html#rule_sets/0"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.rule_sets/1","ref":"Plausible.Cldr.Rbnf.Spellout.html#rule_sets/1"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_cardinal/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_cardinal/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_cardinal_verbose/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_cardinal_verbose/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_numbering/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_numbering/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_numbering_verbose/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_numbering_verbose/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_numbering_year/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_numbering_year/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_ordinal/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_ordinal/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.spellout_ordinal_verbose/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_ordinal_verbose/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.th/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#th/2"},{"type":"function","doc":"","title":"Plausible.Cldr.Rbnf.Spellout.tieth/2","ref":"Plausible.Cldr.Rbnf.Spellout.html#tieth/2"},{"type":"module","doc":"Event schema for when NumericIDs migration is complete","title":"Plausible.ClickhouseEventV2","ref":"Plausible.ClickhouseEventV2.html"},{"type":"function","doc":"","title":"Plausible.ClickhouseEventV2.merge_session/2","ref":"Plausible.ClickhouseEventV2.html#merge_session/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseEventV2.new/1","ref":"Plausible.ClickhouseEventV2.html#new/1"},{"type":"module","doc":"Schema for storing location id <-> translation mappings in ClickHouse\n\nIndirectly read via dictionary `location_data_dictionary` in ALIAS columns in\n`events_v2`, `sessions_v2` and `imported_locations` table.","title":"Plausible.ClickhouseLocationData","ref":"Plausible.ClickhouseLocationData.html"},{"type":"module","doc":"","title":"Plausible.ClickhouseRepo","ref":"Plausible.ClickhouseRepo.html"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.aggregate/3","ref":"Plausible.ClickhouseRepo.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.aggregate/4","ref":"Plausible.ClickhouseRepo.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.all/2","ref":"Plausible.ClickhouseRepo.html#all/2"},{"type":"function","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","title":"Plausible.ClickhouseRepo.alter_update_all/3","ref":"Plausible.ClickhouseRepo.html#alter_update_all/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.checked_out?/0","ref":"Plausible.ClickhouseRepo.html#checked_out?/0"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.checkout/2","ref":"Plausible.ClickhouseRepo.html#checkout/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.child_spec/1","ref":"Plausible.ClickhouseRepo.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.config/0","ref":"Plausible.ClickhouseRepo.html#config/0"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.default_options/1","ref":"Plausible.ClickhouseRepo.html#default_options/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","title":"Plausible.ClickhouseRepo.disconnect_all/2","ref":"Plausible.ClickhouseRepo.html#disconnect_all/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.exists?/2","ref":"Plausible.ClickhouseRepo.html#exists?/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.get/3","ref":"Plausible.ClickhouseRepo.html#get/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.get!/3","ref":"Plausible.ClickhouseRepo.html#get!/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.get_by/3","ref":"Plausible.ClickhouseRepo.html#get_by/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.get_by!/3","ref":"Plausible.ClickhouseRepo.html#get_by!/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.get_dynamic_repo/0","ref":"Plausible.ClickhouseRepo.html#get_dynamic_repo/0"},{"type":"function","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","title":"Plausible.ClickhouseRepo.insert_stream/3","ref":"Plausible.ClickhouseRepo.html#insert_stream/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.load/2","ref":"Plausible.ClickhouseRepo.html#load/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.one/2","ref":"Plausible.ClickhouseRepo.html#one/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.one!/2","ref":"Plausible.ClickhouseRepo.html#one!/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.parallel_tasks/2","ref":"Plausible.ClickhouseRepo.html#parallel_tasks/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.preload/3","ref":"Plausible.ClickhouseRepo.html#preload/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.put_dynamic_repo/1","ref":"Plausible.ClickhouseRepo.html#put_dynamic_repo/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","title":"Plausible.ClickhouseRepo.query/3","ref":"Plausible.ClickhouseRepo.html#query/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","title":"Plausible.ClickhouseRepo.query!/3","ref":"Plausible.ClickhouseRepo.html#query!/3"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.reload/2","ref":"Plausible.ClickhouseRepo.html#reload/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.reload!/2","ref":"Plausible.ClickhouseRepo.html#reload!/2"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.start_link/1","ref":"Plausible.ClickhouseRepo.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.stop/1","ref":"Plausible.ClickhouseRepo.html#stop/1"},{"type":"function","doc":"","title":"Plausible.ClickhouseRepo.stream/2","ref":"Plausible.ClickhouseRepo.html#stream/2"},{"type":"function","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","title":"Plausible.ClickhouseRepo.to_inline_sql/2","ref":"Plausible.ClickhouseRepo.html#to_inline_sql/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","title":"Plausible.ClickhouseRepo.to_sql/2","ref":"Plausible.ClickhouseRepo.html#to_sql/2"},{"type":"module","doc":"Session schema for when NumericIDs migration is complete","title":"Plausible.ClickhouseSessionV2","ref":"Plausible.ClickhouseSessionV2.html"},{"type":"function","doc":"","title":"Plausible.ClickhouseSessionV2.random_uint64/0","ref":"Plausible.ClickhouseSessionV2.html#random_uint64/0"},{"type":"module","doc":"Custom type to cast Bool as UInt8","title":"Plausible.ClickhouseSessionV2.BoolUInt8","ref":"Plausible.ClickhouseSessionV2.BoolUInt8.html"},{"type":"function","doc":"","title":"Plausible.ClickhouseSessionV2.BoolUInt8.embed_as/1","ref":"Plausible.ClickhouseSessionV2.BoolUInt8.html#embed_as/1"},{"type":"function","doc":"","title":"Plausible.ClickhouseSessionV2.BoolUInt8.equal?/2","ref":"Plausible.ClickhouseSessionV2.BoolUInt8.html#equal?/2"},{"type":"module","doc":"","title":"Plausible.ConfigHelpers","ref":"Plausible.ConfigHelpers.html"},{"type":"function","doc":"","title":"Plausible.ConfigHelpers.get_int_from_path_or_env/3","ref":"Plausible.ConfigHelpers.html#get_int_from_path_or_env/3"},{"type":"function","doc":"","title":"Plausible.ConfigHelpers.get_var_from_path_or_env/3","ref":"Plausible.ConfigHelpers.html#get_var_from_path_or_env/3"},{"type":"module","doc":"Extensions for Kaffy CRM","title":"Plausible.CrmExtensions","ref":"Plausible.CrmExtensions.html"},{"type":"function","doc":"","title":"Plausible.CrmExtensions.javascripts/1","ref":"Plausible.CrmExtensions.html#javascripts/1"},{"type":"module","doc":"This module defines the setup for tests requiring\naccess to the application's data layer.\n\nYou may define functions here to be used as helpers in\nyour tests.\n\nFinally, if the test case interacts with the database,\nit cannot be async. For this reason, every test runs\ninside a transaction which is reset at the beginning\nof the test unless the test case is marked as async.","title":"Plausible.DataCase","ref":"Plausible.DataCase.html"},{"type":"module","doc":"Base module for coordinated Clickhouse data migrations\nrun via remote shell or otherwise (TBD).","title":"Plausible.DataMigration","ref":"Plausible.DataMigration.html"},{"type":"module","doc":"Clean up referrer_source entries for demo site with\n`Direct / None` for value populated by dogfooding\nPlausible stats.","title":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource","ref":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource.run/1","ref":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource.html#run/1"},{"type":"module","doc":"ClickHouse locations data migration for storing location names in ClickHouse.\n\nOnly run when `Location.version()` changes: either as a migration or in cron.\n\nThe migration:\n1. Truncates existing `location_data` table (if exists)\n2. Creates new table (if needed)\n3. Inserts new data from Location module\n4. (Re-)Creates dictionary to read location data from table\n5. Creates ALIAS columns in `events_v2`, `sessions_v2` and `imported_locations` table to make reading location names easy\n6. Updates table comment for `location_data` to indicate last version synced.\n\nNote that the dictionary is large enough to cache the whole dataset in memory, making lookups fast.\n\nThis migration is intended to be idempotent and rerunnable - if run multiple times, it should always set things to the same\nresult as if run once.\n\nSQL files available at: priv/data_migrations/LocationsSync/sql","title":"Plausible.DataMigration.LocationsSync","ref":"Plausible.DataMigration.LocationsSync.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.LocationsSync.confirm/3","ref":"Plausible.DataMigration.LocationsSync.html#confirm/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.LocationsSync.out_of_date?/0","ref":"Plausible.DataMigration.LocationsSync.html#out_of_date?/0"},{"type":"function","doc":"","title":"Plausible.DataMigration.LocationsSync.run/0","ref":"Plausible.DataMigration.LocationsSync.html#run/0"},{"type":"function","doc":"","title":"Plausible.DataMigration.LocationsSync.run_sql/3","ref":"Plausible.DataMigration.LocationsSync.html#run_sql/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.LocationsSync.run_sql_confirm/2","ref":"Plausible.DataMigration.LocationsSync.html#run_sql_confirm/2"},{"type":"module","doc":"Numeric IDs migration, SQL files available at:\npriv/data_migrations/NumericIDs/sql","title":"Plausible.DataMigration.NumericIDs","ref":"Plausible.DataMigration.NumericIDs.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.NumericIDs.confirm/3","ref":"Plausible.DataMigration.NumericIDs.html#confirm/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.NumericIDs.run/1","ref":"Plausible.DataMigration.NumericIDs.html#run/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.NumericIDs.run_sql/3","ref":"Plausible.DataMigration.NumericIDs.html#run_sql/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.NumericIDs.run_sql_confirm/2","ref":"Plausible.DataMigration.NumericIDs.html#run_sql_confirm/2"},{"type":"module","doc":"Populates event session columns with data from sessions table.\n\nRun via: ./bin/plausible rpc \"Plausible.DataMigration.PopulateEventSessionColumns.run\"\nKill via: ./bin/plausible rpc \"Plausible.DataMigration.PopulateEventSessionColumns.kill\"\nMonitor via ./bin/plausible rpc \"Plausible.DataMigration.PopulateEventSessionColumns.report_progress\"\n\nSuggested to run in a screen/tmux session to be able to easily monitor\n\nSQL files available at: priv/data_migrations/PopulateEventSessionColumns/sql","title":"Plausible.DataMigration.PopulateEventSessionColumns","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.confirm/3","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#confirm/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.kill/1","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#kill/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.report_progress/1","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#report_progress/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.run/1","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#run/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.run_sql/3","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#run_sql/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.run_sql_confirm/2","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#run_sql_confirm/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.PopulateEventSessionColumns.wait_until_mutations_complete/1","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#wait_until_mutations_complete/1"},{"type":"module","doc":"Ecto.Repo for Clickhouse data migrations, to be started manually,\noutside of the main application supervision tree.","title":"Plausible.DataMigration.Repo","ref":"Plausible.DataMigration.Repo.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.aggregate/3","ref":"Plausible.DataMigration.Repo.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.aggregate/4","ref":"Plausible.DataMigration.Repo.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.all/2","ref":"Plausible.DataMigration.Repo.html#all/2"},{"type":"function","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","title":"Plausible.DataMigration.Repo.alter_update_all/3","ref":"Plausible.DataMigration.Repo.html#alter_update_all/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.checked_out?/0","ref":"Plausible.DataMigration.Repo.html#checked_out?/0"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.checkout/2","ref":"Plausible.DataMigration.Repo.html#checkout/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.child_spec/1","ref":"Plausible.DataMigration.Repo.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.config/0","ref":"Plausible.DataMigration.Repo.html#config/0"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.default_options/1","ref":"Plausible.DataMigration.Repo.html#default_options/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.delete/2","ref":"Plausible.DataMigration.Repo.html#delete/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.delete!/2","ref":"Plausible.DataMigration.Repo.html#delete!/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.delete_all/2","ref":"Plausible.DataMigration.Repo.html#delete_all/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","title":"Plausible.DataMigration.Repo.disconnect_all/2","ref":"Plausible.DataMigration.Repo.html#disconnect_all/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.exists?/2","ref":"Plausible.DataMigration.Repo.html#exists?/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.get/3","ref":"Plausible.DataMigration.Repo.html#get/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.get!/3","ref":"Plausible.DataMigration.Repo.html#get!/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.get_by/3","ref":"Plausible.DataMigration.Repo.html#get_by/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.get_by!/3","ref":"Plausible.DataMigration.Repo.html#get_by!/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.get_dynamic_repo/0","ref":"Plausible.DataMigration.Repo.html#get_dynamic_repo/0"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.insert/2","ref":"Plausible.DataMigration.Repo.html#insert/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.insert!/2","ref":"Plausible.DataMigration.Repo.html#insert!/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.insert_all/3","ref":"Plausible.DataMigration.Repo.html#insert_all/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.insert_or_update/2","ref":"Plausible.DataMigration.Repo.html#insert_or_update/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.insert_or_update!/2","ref":"Plausible.DataMigration.Repo.html#insert_or_update!/2"},{"type":"function","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","title":"Plausible.DataMigration.Repo.insert_stream/3","ref":"Plausible.DataMigration.Repo.html#insert_stream/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.load/2","ref":"Plausible.DataMigration.Repo.html#load/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.one/2","ref":"Plausible.DataMigration.Repo.html#one/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.one!/2","ref":"Plausible.DataMigration.Repo.html#one!/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.preload/3","ref":"Plausible.DataMigration.Repo.html#preload/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.prepare_query/3","ref":"Plausible.DataMigration.Repo.html#prepare_query/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.put_dynamic_repo/1","ref":"Plausible.DataMigration.Repo.html#put_dynamic_repo/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","title":"Plausible.DataMigration.Repo.query/3","ref":"Plausible.DataMigration.Repo.html#query/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","title":"Plausible.DataMigration.Repo.query!/3","ref":"Plausible.DataMigration.Repo.html#query!/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.reload/2","ref":"Plausible.DataMigration.Repo.html#reload/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.reload!/2","ref":"Plausible.DataMigration.Repo.html#reload!/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.start/2","ref":"Plausible.DataMigration.Repo.html#start/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.start_link/1","ref":"Plausible.DataMigration.Repo.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.stop/1","ref":"Plausible.DataMigration.Repo.html#stop/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.stream/2","ref":"Plausible.DataMigration.Repo.html#stream/2"},{"type":"function","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","title":"Plausible.DataMigration.Repo.to_inline_sql/2","ref":"Plausible.DataMigration.Repo.html#to_inline_sql/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","title":"Plausible.DataMigration.Repo.to_sql/2","ref":"Plausible.DataMigration.Repo.html#to_sql/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.update/2","ref":"Plausible.DataMigration.Repo.html#update/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.update!/2","ref":"Plausible.DataMigration.Repo.html#update!/2"},{"type":"function","doc":"","title":"Plausible.DataMigration.Repo.update_all/3","ref":"Plausible.DataMigration.Repo.html#update_all/3"},{"type":"module","doc":"!!!WARNING!!!: This script is used in migrations. Please take special care\nwhen altering it.\n\nSite imports migration backfilling SiteImport entries for old imports\nand alters import end dates to match actual end date of respective import stats.","title":"Plausible.DataMigration.SiteImports","ref":"Plausible.DataMigration.SiteImports.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.SiteImports.run/1","ref":"Plausible.DataMigration.SiteImports.html#run/1"},{"type":"module","doc":"!!!WARNING!!!: This script is used in migrations. Please take special care\nwhen altering it.\n\nSessions CollapsingMergeTree -> VersionedCollapsingMergeTree migration,\nSQL files available at:\n\npriv/data_migrations/VersionedSessions/sql","title":"Plausible.DataMigration.VersionedSessions","ref":"Plausible.DataMigration.VersionedSessions.html"},{"type":"function","doc":"","title":"Plausible.DataMigration.VersionedSessions.confirm/3","ref":"Plausible.DataMigration.VersionedSessions.html#confirm/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.VersionedSessions.run/1","ref":"Plausible.DataMigration.VersionedSessions.html#run/1"},{"type":"function","doc":"","title":"Plausible.DataMigration.VersionedSessions.run_sql/3","ref":"Plausible.DataMigration.VersionedSessions.html#run_sql/3"},{"type":"function","doc":"","title":"Plausible.DataMigration.VersionedSessions.run_sql_confirm/2","ref":"Plausible.DataMigration.VersionedSessions.html#run_sql_confirm/2"},{"type":"module","doc":"Function execution context (with arguments) to Sentry reports.","title":"Plausible.DebugReplayInfo","ref":"Plausible.DebugReplayInfo.html"},{"type":"function","doc":"","title":"Plausible.DebugReplayInfo.deserialize/1","ref":"Plausible.DebugReplayInfo.html#deserialize/1"},{"type":"macro","doc":"","title":"Plausible.DebugReplayInfo.include_sentry_replay_info/0","ref":"Plausible.DebugReplayInfo.html#include_sentry_replay_info/0"},{"type":"module","doc":"Custom type for event name. Accepts Strings and Integers and stores them as String. Returns\n cast error if any other type is provided. Accepting integers is important for 404 tracking.","title":"Plausible.Ecto.EventName","ref":"Plausible.Ecto.EventName.html"},{"type":"function","doc":"","title":"Plausible.Ecto.EventName.cast/1","ref":"Plausible.Ecto.EventName.html#cast/1"},{"type":"function","doc":"","title":"Plausible.Ecto.EventName.dump/1","ref":"Plausible.Ecto.EventName.html#dump/1"},{"type":"function","doc":"","title":"Plausible.Ecto.EventName.embed_as/1","ref":"Plausible.Ecto.EventName.html#embed_as/1"},{"type":"function","doc":"","title":"Plausible.Ecto.EventName.equal?/2","ref":"Plausible.Ecto.EventName.html#equal?/2"},{"type":"function","doc":"","title":"Plausible.Ecto.EventName.load/1","ref":"Plausible.Ecto.EventName.html#load/1"},{"type":"function","doc":"","title":"Plausible.Ecto.EventName.type/0","ref":"Plausible.Ecto.EventName.html#type/0"},{"type":"module","doc":"Ensures that the regex is compiled on load","title":"Plausible.Ecto.Types.CompiledRegex","ref":"Plausible.Ecto.Types.CompiledRegex.html"},{"type":"function","doc":"","title":"Plausible.Ecto.Types.CompiledRegex.cast/1","ref":"Plausible.Ecto.Types.CompiledRegex.html#cast/1"},{"type":"function","doc":"","title":"Plausible.Ecto.Types.CompiledRegex.dump/1","ref":"Plausible.Ecto.Types.CompiledRegex.html#dump/1"},{"type":"function","doc":"","title":"Plausible.Ecto.Types.CompiledRegex.embed_as/1","ref":"Plausible.Ecto.Types.CompiledRegex.html#embed_as/1"},{"type":"function","doc":"","title":"Plausible.Ecto.Types.CompiledRegex.equal?/2","ref":"Plausible.Ecto.Types.CompiledRegex.html#equal?/2"},{"type":"function","doc":"","title":"Plausible.Ecto.Types.CompiledRegex.load/1","ref":"Plausible.Ecto.Types.CompiledRegex.html#load/1"},{"type":"function","doc":"","title":"Plausible.Ecto.Types.CompiledRegex.type/0","ref":"Plausible.Ecto.Types.CompiledRegex.html#type/0"},{"type":"module","doc":"Contains functions to export data for events and sessions as Zip archives.","title":"Plausible.Exports","ref":"Plausible.Exports.html"},{"type":"function","doc":"Renders export archive filename.\n\nExamples:\n\n iex> archive_filename(\"plausible.io\", _created_on = ~D[2024-12-31])\n \"plausible_io_20241231.zip\"","title":"Plausible.Exports.archive_filename/2","ref":"Plausible.Exports.html#archive_filename/2"},{"type":"function","doc":"Safely renders content disposition for an arbitrary export filename.\n\nExamples:\n\n iex> content_disposition(\"plausible_io_20241231.zip\")\n \"attachment; filename=\\\"plausible_io_20241231.zip\\\"\"\n\n iex> content_disposition(\"📊.zip\")\n \"attachment; filename=\\\"plausible-export.zip\\\"; filename*=utf-8''%F0%9F%93%8A.zip\"","title":"Plausible.Exports.content_disposition/1","ref":"Plausible.Exports.html#content_disposition/1"},{"type":"function","doc":"Returns the date range for the site's events data in site's timezone or `nil` if there is no data","title":"Plausible.Exports.date_range/2","ref":"Plausible.Exports.html#date_range/2"},{"type":"function","doc":"Deletes local export for a site","title":"Plausible.Exports.delete_local_export/1","ref":"Plausible.Exports.html#delete_local_export/1"},{"type":"function","doc":"Deletes S3 export for a site. Raises if object storage is unavailable.","title":"Plausible.Exports.delete_s3_export!/1","ref":"Plausible.Exports.html#delete_s3_export!/1"},{"type":"function","doc":"Builds Ecto queries to export data from `events_v2` and `sessions_v2`\ntables into the format of `imported_*` tables for a website.","title":"Plausible.Exports.export_queries/2","ref":"Plausible.Exports.html#export_queries/2"},{"type":"function","doc":"Gets last CSV export job for a site","title":"Plausible.Exports.get_last_export_job/1","ref":"Plausible.Exports.html#get_last_export_job/1"},{"type":"function","doc":"Gets local export for a site","title":"Plausible.Exports.get_local_export/3","ref":"Plausible.Exports.html#get_local_export/3"},{"type":"function","doc":"Gets S3 export for a site. Raises if object storage is unavailable.","title":"Plausible.Exports.get_s3_export!/2","ref":"Plausible.Exports.html#get_s3_export!/2"},{"type":"function","doc":"Subscribes to CSV export job notifications","title":"Plausible.Exports.oban_listen/0","ref":"Plausible.Exports.html#oban_listen/0"},{"type":"function","doc":"Schedules CSV export job to local storage","title":"Plausible.Exports.schedule_local_export/2","ref":"Plausible.Exports.html#schedule_local_export/2"},{"type":"function","doc":"Schedules CSV export job to S3 storage","title":"Plausible.Exports.schedule_s3_export/2","ref":"Plausible.Exports.html#schedule_s3_export/2"},{"type":"function","doc":"Creates a streamable Zip archive from the provided (named) Ecto queries.\n\nExample usage:\n\n {:ok, pool} = Ch.start_link(pool_size: 1)\n\n DBConnection.run(pool, fn conn ->\n conn\n |> stream_archive(export_queries(_site_id = 1), format: \"CSVWithNames\")\n |> Stream.into(File.stream!(\"export.zip\"))\n |> Stream.run()\n end)","title":"Plausible.Exports.stream_archive/3","ref":"Plausible.Exports.html#stream_archive/3"},{"type":"type","doc":"","title":"Plausible.Exports.export/0","ref":"Plausible.Exports.html#t:export/0"},{"type":"module","doc":"","title":"Plausible.Factory","ref":"Plausible.Factory.html"},{"type":"function","doc":"","title":"Plausible.Factory.api_key_factory/0","ref":"Plausible.Factory.html#api_key_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.build/2","ref":"Plausible.Factory.html#build/2"},{"type":"function","doc":"","title":"Plausible.Factory.build_list/3","ref":"Plausible.Factory.html#build_list/3"},{"type":"function","doc":"","title":"Plausible.Factory.build_pair/2","ref":"Plausible.Factory.html#build_pair/2"},{"type":"function","doc":"","title":"Plausible.Factory.business_subscription_factory/0","ref":"Plausible.Factory.html#business_subscription_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.ch_session_factory/0","ref":"Plausible.Factory.html#ch_session_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.country_rule_factory/0","ref":"Plausible.Factory.html#country_rule_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.create/1","ref":"Plausible.Factory.html#create/1"},{"type":"function","doc":"","title":"Plausible.Factory.create/2","ref":"Plausible.Factory.html#create/2"},{"type":"function","doc":"","title":"Plausible.Factory.create_list/3","ref":"Plausible.Factory.html#create_list/3"},{"type":"function","doc":"","title":"Plausible.Factory.create_pair/2","ref":"Plausible.Factory.html#create_pair/2"},{"type":"function","doc":"","title":"Plausible.Factory.drop_notification_factory/0","ref":"Plausible.Factory.html#drop_notification_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.enterprise_plan_factory/0","ref":"Plausible.Factory.html#enterprise_plan_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.event_factory/0","ref":"Plausible.Factory.html#event_factory/0"},{"type":"function","doc":"Raises a helpful error if no factory is defined.","title":"Plausible.Factory.factory/1","ref":"Plausible.Factory.html#factory/1"},{"type":"function","doc":"","title":"Plausible.Factory.goal_factory/1","ref":"Plausible.Factory.html#goal_factory/1"},{"type":"function","doc":"","title":"Plausible.Factory.google_auth_factory/0","ref":"Plausible.Factory.html#google_auth_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.growth_subscription_factory/0","ref":"Plausible.Factory.html#growth_subscription_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_browsers_factory/0","ref":"Plausible.Factory.html#imported_browsers_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_custom_events_factory/0","ref":"Plausible.Factory.html#imported_custom_events_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_devices_factory/0","ref":"Plausible.Factory.html#imported_devices_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_entry_pages_factory/0","ref":"Plausible.Factory.html#imported_entry_pages_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_exit_pages_factory/0","ref":"Plausible.Factory.html#imported_exit_pages_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_locations_factory/0","ref":"Plausible.Factory.html#imported_locations_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_operating_systems_factory/0","ref":"Plausible.Factory.html#imported_operating_systems_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_pages_factory/0","ref":"Plausible.Factory.html#imported_pages_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_sources_factory/0","ref":"Plausible.Factory.html#imported_sources_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.imported_visitors_factory/0","ref":"Plausible.Factory.html#imported_visitors_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.insert/1","ref":"Plausible.Factory.html#insert/1"},{"type":"function","doc":"","title":"Plausible.Factory.insert/2","ref":"Plausible.Factory.html#insert/2"},{"type":"function","doc":"","title":"Plausible.Factory.insert/3","ref":"Plausible.Factory.html#insert/3"},{"type":"function","doc":"","title":"Plausible.Factory.insert_list/3","ref":"Plausible.Factory.html#insert_list/3"},{"type":"function","doc":"","title":"Plausible.Factory.insert_list/4","ref":"Plausible.Factory.html#insert_list/4"},{"type":"function","doc":"","title":"Plausible.Factory.insert_pair/2","ref":"Plausible.Factory.html#insert_pair/2"},{"type":"function","doc":"","title":"Plausible.Factory.insert_pair/3","ref":"Plausible.Factory.html#insert_pair/3"},{"type":"function","doc":"","title":"Plausible.Factory.invitation_factory/0","ref":"Plausible.Factory.html#invitation_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.ip_rule_factory/0","ref":"Plausible.Factory.html#ip_rule_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.monthly_report_factory/0","ref":"Plausible.Factory.html#monthly_report_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.pageview_factory/0","ref":"Plausible.Factory.html#pageview_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.params_for/2","ref":"Plausible.Factory.html#params_for/2"},{"type":"function","doc":"","title":"Plausible.Factory.params_with_assocs/2","ref":"Plausible.Factory.html#params_with_assocs/2"},{"type":"function","doc":"","title":"Plausible.Factory.shared_link_factory/0","ref":"Plausible.Factory.html#shared_link_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.site_factory/1","ref":"Plausible.Factory.html#site_factory/1"},{"type":"function","doc":"","title":"Plausible.Factory.site_import_factory/0","ref":"Plausible.Factory.html#site_import_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.site_membership_factory/0","ref":"Plausible.Factory.html#site_membership_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.spike_notification_factory/0","ref":"Plausible.Factory.html#spike_notification_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.string_params_for/2","ref":"Plausible.Factory.html#string_params_for/2"},{"type":"function","doc":"","title":"Plausible.Factory.string_params_with_assocs/2","ref":"Plausible.Factory.html#string_params_with_assocs/2"},{"type":"function","doc":"","title":"Plausible.Factory.subscription_factory/0","ref":"Plausible.Factory.html#subscription_factory/0"},{"type":"function","doc":"","title":"Plausible.Factory.user_factory/1","ref":"Plausible.Factory.html#user_factory/1"},{"type":"function","doc":"","title":"Plausible.Factory.weekly_report_factory/0","ref":"Plausible.Factory.html#weekly_report_factory/0"},{"type":"module","doc":"A funnel is a marketing term used to capture and describe the journey\nthat users go through, from initial step to conversion.\nA funnel consists of several steps (here: 2..8).\n\nThis module defines the database schema for storing funnels\nand changeset helpers for enumerating the steps within.\n\nEach step references a goal (either a Custom Event or Visit)\n- see: `Plausible.Goal`.","title":"Plausible.Funnel","ref":"Plausible.Funnel.html"},{"type":"function","doc":"","title":"Plausible.Funnel.changeset/2","ref":"Plausible.Funnel.html#changeset/2"},{"type":"macro","doc":"","title":"Plausible.Funnel.max_steps/0","ref":"Plausible.Funnel.html#max_steps/0"},{"type":"macro","doc":"","title":"Plausible.Funnel.min_steps/0","ref":"Plausible.Funnel.html#min_steps/0"},{"type":"function","doc":"","title":"Plausible.Funnel.put_steps/2","ref":"Plausible.Funnel.html#put_steps/2"},{"type":"type","doc":"","title":"Plausible.Funnel.t/0","ref":"Plausible.Funnel.html#t:t/0"},{"type":"module","doc":"Compile-time convenience constants for funnel characteristics.","title":"Plausible.Funnel.Const","ref":"Plausible.Funnel.Const.html"},{"type":"macro","doc":"","title":"Plausible.Funnel.Const.max_steps/0","ref":"Plausible.Funnel.Const.html#max_steps/0"},{"type":"macro","doc":"","title":"Plausible.Funnel.Const.min_steps/0","ref":"Plausible.Funnel.Const.html#min_steps/0"},{"type":"module","doc":"This module defines the database schema for a single Funnel step.\nSee: `Plausible.Funnel` for more information.","title":"Plausible.Funnel.Step","ref":"Plausible.Funnel.Step.html"},{"type":"function","doc":"","title":"Plausible.Funnel.Step.changeset/2","ref":"Plausible.Funnel.Step.html#changeset/2"},{"type":"type","doc":"","title":"Plausible.Funnel.Step.t/0","ref":"Plausible.Funnel.Step.html#t:t/0"},{"type":"module","doc":"This module implements contextual Funnel interface, allowing listing,\ncreating and deleting funnel definitions.\n\nFor brief explanation of what a Funnel is, please see `Plausible.Funnel` schema.\nSee `Plausible.Stats.Funnel` for the evaluation logic.","title":"Plausible.Funnels","ref":"Plausible.Funnels.html"},{"type":"function","doc":"","title":"Plausible.Funnels.create/3","ref":"Plausible.Funnels.html#create/3"},{"type":"function","doc":"","title":"Plausible.Funnels.create_changeset/3","ref":"Plausible.Funnels.html#create_changeset/3"},{"type":"function","doc":"","title":"Plausible.Funnels.delete/2","ref":"Plausible.Funnels.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Funnels.edit_changeset/3","ref":"Plausible.Funnels.html#edit_changeset/3"},{"type":"function","doc":"","title":"Plausible.Funnels.ephemeral_definition/3","ref":"Plausible.Funnels.html#ephemeral_definition/3"},{"type":"function","doc":"","title":"Plausible.Funnels.get/2","ref":"Plausible.Funnels.html#get/2"},{"type":"function","doc":"","title":"Plausible.Funnels.list/1","ref":"Plausible.Funnels.html#list/1"},{"type":"function","doc":"","title":"Plausible.Funnels.update/3","ref":"Plausible.Funnels.html#update/3"},{"type":"function","doc":"","title":"Plausible.Funnels.with_goals_query/1","ref":"Plausible.Funnels.html#with_goals_query/1"},{"type":"module","doc":"This module provides an API for fetching IP geolocation.","title":"Plausible.Geo","ref":"Plausible.Geo.html"},{"type":"function","doc":"Waits for the database to start after calling `load_db/1` with the async option.","title":"Plausible.Geo.await_loader/0","ref":"Plausible.Geo.html#await_loader/0"},{"type":"function","doc":"Returns geodatabase type.\n\nUsed for deciding whether to show the DB-IP disclaimer or not.","title":"Plausible.Geo.database_type/0","ref":"Plausible.Geo.html#database_type/0"},{"type":"function","doc":"In the case of a DB-IP database:\n\n iex> database_type()\n \"DBIP-City-Lite\"\n\n In the case of a MaxMind database:\n\n iex> database_type()\n \"GeoLite2-City\"","title":"Examples - Plausible.Geo.database_type/0","ref":"Plausible.Geo.html#database_type/0-examples"},{"type":"function","doc":"Starts the geodatabase loading process. Two modes are supported: local file\nand MaxMind license key.","title":"Plausible.Geo.load_db/1","ref":"Plausible.Geo.html#load_db/1"},{"type":"function","doc":"* `:path` - the path to the .mmdb database local file. When present,\n `:license_key` and `:edition` are not required.\n\n * `:license_key` - the [license key](https://support.maxmind.com/hc/en-us/articles/4407111582235-Generate-a-License-Key)\n from MaxMind to authenticate requests to MaxMind.\n\n * `:edition` - the name of the MaxMind database to be downloaded from MaxMind\n servers. Defaults to `GeoLite2-City`.\n\n * `:cache_dir` - if set, the downloaded .mmdb files are cached there across\n restarts.\n\n * `:async` - when used, configures the database loading to run\n asynchronously.","title":"Options - Plausible.Geo.load_db/1","ref":"Plausible.Geo.html#load_db/1-options"},{"type":"function","doc":"Loading from a local file:\n\n iex> load_db(path: \"/etc/plausible/dbip-city.mmdb\")\n :ok\n\n Downloading a MaxMind DB (this license key is no longer active):\n\n iex> load_db(license_key: \"LNpsJCCKPis6XvBP\", edition: \"GeoLite2-City\", async: true)\n :ok","title":"Examples - Plausible.Geo.load_db/1","ref":"Plausible.Geo.html#load_db/1-examples"},{"type":"function","doc":"Looks up geo info about an IP address.","title":"Plausible.Geo.lookup/1","ref":"Plausible.Geo.html#lookup/1"},{"type":"function","doc":"iex> lookup(\"8.7.6.5\")\n %{\n \"city\" => %{\n \"geoname_id\" => 5349755,\n \"names\" => %{\n \"de\" => \"Fontana\",\n \"en\" => \"Fontana\",\n \"ja\" => \"フォンタナ\",\n \"ru\" => \"Фонтана\"\n }\n },\n \"continent\" => %{\n \"code\" => \"NA\",\n \"geoname_id\" => 6255149,\n \"names\" => %{\n \"de\" => \"Nordamerika\",\n \"en\" => \"North America\",\n \"es\" => \"Norteamérica\",\n \"fr\" => \"Amérique du Nord\",\n \"ja\" => \"北アメリカ\",\n \"pt-BR\" => \"América do Norte\",\n \"ru\" => \"Северная Америка\",\n \"zh-CN\" => \"北美洲\"\n }\n },\n \"country\" => %{\n \"geoname_id\" => 6252001,\n \"iso_code\" => \"US\",\n \"names\" => %{\n \"de\" => \"Vereinigte Staaten\",\n \"en\" => \"United States\",\n \"es\" => \"Estados Unidos\",\n \"fr\" => \"États Unis\",\n \"ja\" => \"アメリカ\",\n \"pt-BR\" => \"EUA\",\n \"ru\" => \"США\",\n \"zh-CN\" => \"美国\"\n }\n },\n \"location\" => %{\n \"accuracy_radius\" => 50,\n \"latitude\" => 34.1211,\n \"longitude\" => -117.4362,\n \"metro_code\" => 803,\n \"time_zone\" => \"America/Los_Angeles\"\n },\n \"postal\" => %{\"code\" => \"92336\"},\n \"registered_country\" => %{\n \"geoname_id\" => 6252001,\n \"iso_code\" => \"US\",\n \"names\" => %{\n \"de\" => \"Vereinigte Staaten\",\n \"en\" => \"United States\",\n \"es\" => \"Estados Unidos\",\n \"fr\" => \"États Unis\",\n \"ja\" => \"アメリカ\",\n \"pt-BR\" => \"EUA\",\n \"ru\" => \"США\",\n \"zh-CN\" => \"美国\"\n }\n },\n \"subdivisions\" => [\n %{\n \"geoname_id\" => 5332921,\n \"iso_code\" => \"CA\",\n \"names\" => %{\n \"de\" => \"Kalifornien\",\n \"en\" => \"California\",\n \"es\" => \"California\",\n \"fr\" => \"Californie\",\n \"ja\" => \"カリフォルニア州\",\n \"pt-BR\" => \"Califórnia\",\n \"ru\" => \"Калифорния\",\n \"zh-CN\" => \"加州\"\n }\n }\n ]\n }","title":"Examples - Plausible.Geo.lookup/1","ref":"Plausible.Geo.html#lookup/1-examples"},{"type":"module","doc":"","title":"Plausible.Goal","ref":"Plausible.Goal.html"},{"type":"function","doc":"","title":"Plausible.Goal.changeset/2","ref":"Plausible.Goal.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Goal.display_name/1","ref":"Plausible.Goal.html#display_name/1"},{"type":"function","doc":"","title":"Plausible.Goal.max_event_name_length/0","ref":"Plausible.Goal.html#max_event_name_length/0"},{"type":"function","doc":"","title":"Plausible.Goal.type/1","ref":"Plausible.Goal.html#type/1"},{"type":"type","doc":"","title":"Plausible.Goal.t/0","ref":"Plausible.Goal.html#t:t/0"},{"type":"module","doc":"Currency specific functions for revenue goals","title":"Plausible.Goal.Revenue","ref":"Plausible.Goal.Revenue.html"},{"type":"function","doc":"","title":"Plausible.Goal.Revenue.currency_option/1","ref":"Plausible.Goal.Revenue.html#currency_option/1"},{"type":"function","doc":"","title":"Plausible.Goal.Revenue.currency_options/0","ref":"Plausible.Goal.Revenue.html#currency_options/0"},{"type":"function","doc":"","title":"Plausible.Goal.Revenue.display/1","ref":"Plausible.Goal.Revenue.html#display/1"},{"type":"function","doc":"","title":"Plausible.Goal.Revenue.revenue?/1","ref":"Plausible.Goal.Revenue.html#revenue?/1"},{"type":"function","doc":"","title":"Plausible.Goal.Revenue.valid_currencies/0","ref":"Plausible.Goal.Revenue.html#valid_currencies/0"},{"type":"module","doc":"","title":"Plausible.Goals","ref":"Plausible.Goals.html"},{"type":"function","doc":"","title":"Plausible.Goals.batch_create_event_goals/2","ref":"Plausible.Goals.html#batch_create_event_goals/2"},{"type":"function","doc":"","title":"Plausible.Goals.count/1","ref":"Plausible.Goals.html#count/1"},{"type":"function","doc":"Creates a Goal for a site.\n\nIf the created goal is a revenue goal, it sets site.updated_at to be\nrefreshed by the sites cache, as revenue goals are used during ingestion.","title":"Plausible.Goals.create/3","ref":"Plausible.Goals.html#create/3"},{"type":"function","doc":"","title":"Plausible.Goals.create_404/1","ref":"Plausible.Goals.html#create_404/1"},{"type":"function","doc":"","title":"Plausible.Goals.create_file_downloads/1","ref":"Plausible.Goals.html#create_file_downloads/1"},{"type":"function","doc":"","title":"Plausible.Goals.create_outbound_links/1","ref":"Plausible.Goals.html#create_outbound_links/1"},{"type":"function","doc":"If a goal belongs to funnel(s), we need to inspect their number of steps.\n\nIf it exceeds the minimum allowed (defined via `Plausible.Funnel.min_steps/0`),\nthe funnel will be reduced (i.e. a step associated with the goal to be deleted\nis removed), so that the minimum number of steps is preserved. This is done\nimplicitly, by postgres, as per on_delete: :delete_all.\n\nOtherwise, for associated funnel(s) consisting of minimum number steps only,\nfunnel record(s) are removed completely along with the targeted goal.","title":"Plausible.Goals.delete/2","ref":"Plausible.Goals.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Goals.delete_404/1","ref":"Plausible.Goals.html#delete_404/1"},{"type":"function","doc":"","title":"Plausible.Goals.delete_file_downloads/1","ref":"Plausible.Goals.html#delete_file_downloads/1"},{"type":"function","doc":"","title":"Plausible.Goals.delete_outbound_links/1","ref":"Plausible.Goals.html#delete_outbound_links/1"},{"type":"function","doc":"","title":"Plausible.Goals.find_or_create/2","ref":"Plausible.Goals.html#find_or_create/2"},{"type":"function","doc":"","title":"Plausible.Goals.for_site/2","ref":"Plausible.Goals.html#for_site/2"},{"type":"function","doc":"","title":"Plausible.Goals.for_site_query/2","ref":"Plausible.Goals.html#for_site_query/2"},{"type":"function","doc":"","title":"Plausible.Goals.get/2","ref":"Plausible.Goals.html#get/2"},{"type":"function","doc":"","title":"Plausible.Goals.list_revenue_goals/1","ref":"Plausible.Goals.html#list_revenue_goals/1"},{"type":"function","doc":"","title":"Plausible.Goals.update/2","ref":"Plausible.Goals.html#update/2"},{"type":"module","doc":"API to Google services.","title":"Plausible.Google.API","ref":"Plausible.Google.API.html"},{"type":"function","doc":"","title":"Plausible.Google.API.fetch_access_token!/1","ref":"Plausible.Google.API.html#fetch_access_token!/1"},{"type":"function","doc":"","title":"Plausible.Google.API.fetch_stats/4","ref":"Plausible.Google.API.html#fetch_stats/4"},{"type":"function","doc":"","title":"Plausible.Google.API.fetch_verified_properties/1","ref":"Plausible.Google.API.html#fetch_verified_properties/1"},{"type":"function","doc":"","title":"Plausible.Google.API.get_analytics_end_date/2","ref":"Plausible.Google.API.html#get_analytics_end_date/2"},{"type":"function","doc":"","title":"Plausible.Google.API.get_analytics_start_date/2","ref":"Plausible.Google.API.html#get_analytics_start_date/2"},{"type":"function","doc":"","title":"Plausible.Google.API.get_property/2","ref":"Plausible.Google.API.html#get_property/2"},{"type":"function","doc":"","title":"Plausible.Google.API.import_authorize_url/1","ref":"Plausible.Google.API.html#import_authorize_url/1"},{"type":"function","doc":"","title":"Plausible.Google.API.list_properties/1","ref":"Plausible.Google.API.html#list_properties/1"},{"type":"function","doc":"","title":"Plausible.Google.API.maybe_refresh_token/1","ref":"Plausible.Google.API.html#maybe_refresh_token/1"},{"type":"function","doc":"","title":"Plausible.Google.API.property?/1","ref":"Plausible.Google.API.html#property?/1"},{"type":"function","doc":"","title":"Plausible.Google.API.search_console_authorize_url/1","ref":"Plausible.Google.API.html#search_console_authorize_url/1"},{"type":"module","doc":"Mock of API to Google services.","title":"Plausible.Google.API.Mock","ref":"Plausible.Google.API.Mock.html"},{"type":"function","doc":"","title":"Plausible.Google.API.Mock.fetch_stats/4","ref":"Plausible.Google.API.Mock.html#fetch_stats/4"},{"type":"module","doc":"API for Google Analytics 4.","title":"Plausible.Google.GA4.API","ref":"Plausible.Google.GA4.API.html"},{"type":"function","doc":"","title":"Plausible.Google.GA4.API.fetch_and_persist/2","ref":"Plausible.Google.GA4.API.html#fetch_and_persist/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.API.get_analytics_end_date/2","ref":"Plausible.Google.GA4.API.html#get_analytics_end_date/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.API.get_analytics_start_date/2","ref":"Plausible.Google.GA4.API.html#get_analytics_start_date/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.API.get_property/2","ref":"Plausible.Google.GA4.API.html#get_property/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.API.import_analytics/4","ref":"Plausible.Google.GA4.API.html#import_analytics/4"},{"type":"function","doc":"","title":"Plausible.Google.GA4.API.list_properties/1","ref":"Plausible.Google.GA4.API.html#list_properties/1"},{"type":"type","doc":"","title":"Plausible.Google.GA4.API.import_auth/0","ref":"Plausible.Google.GA4.API.html#t:import_auth/0"},{"type":"module","doc":"HTTP client implementation for Google Analytics 4 API.","title":"Plausible.Google.GA4.HTTP","ref":"Plausible.Google.GA4.HTTP.html"},{"type":"function","doc":"","title":"Plausible.Google.GA4.HTTP.get_analytics_end_date/2","ref":"Plausible.Google.GA4.HTTP.html#get_analytics_end_date/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.HTTP.get_analytics_start_date/2","ref":"Plausible.Google.GA4.HTTP.html#get_analytics_start_date/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.HTTP.get_property/2","ref":"Plausible.Google.GA4.HTTP.html#get_property/2"},{"type":"function","doc":"","title":"Plausible.Google.GA4.HTTP.get_report/1","ref":"Plausible.Google.GA4.HTTP.html#get_report/1"},{"type":"function","doc":"","title":"Plausible.Google.GA4.HTTP.list_accounts_for_user/1","ref":"Plausible.Google.GA4.HTTP.html#list_accounts_for_user/1"},{"type":"module","doc":"Report request struct for Google Analytics 4 API","title":"Plausible.Google.GA4.ReportRequest","ref":"Plausible.Google.GA4.ReportRequest.html"},{"type":"function","doc":"","title":"Plausible.Google.GA4.ReportRequest.full_report/0","ref":"Plausible.Google.GA4.ReportRequest.html#full_report/0"},{"type":"type","doc":"","title":"Plausible.Google.GA4.ReportRequest.t/0","ref":"Plausible.Google.GA4.ReportRequest.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Google.HTTP","ref":"Plausible.Google.HTTP.html"},{"type":"function","doc":"","title":"Plausible.Google.HTTP.fetch_access_token!/1","ref":"Plausible.Google.HTTP.html#fetch_access_token!/1"},{"type":"function","doc":"","title":"Plausible.Google.HTTP.list_sites/1","ref":"Plausible.Google.HTTP.html#list_sites/1"},{"type":"function","doc":"","title":"Plausible.Google.HTTP.list_stats/5","ref":"Plausible.Google.HTTP.html#list_stats/5"},{"type":"function","doc":"","title":"Plausible.Google.HTTP.refresh_auth_token/1","ref":"Plausible.Google.HTTP.html#refresh_auth_token/1"},{"type":"module","doc":"HTTP Client built on top of Finch.\n\nBy default, request parameters are json-encoded.\n\nIf a raw binary value is supplied, no encoding is performed.\nIf x-www-form-urlencoded content-type is set in headers,\nURL encoding is invoked.","title":"Plausible.HTTPClient","ref":"Plausible.HTTPClient.html"},{"type":"function","doc":"Make a GET request","title":"Plausible.HTTPClient.get/3","ref":"Plausible.HTTPClient.html#get/3"},{"type":"function","doc":"","title":"Plausible.HTTPClient.impl/0","ref":"Plausible.HTTPClient.html#impl/0"},{"type":"function","doc":"Make a POST request","title":"Plausible.HTTPClient.post/4","ref":"Plausible.HTTPClient.html#post/4"},{"type":"behaviour","doc":"","title":"Plausible.HTTPClient.Interface","ref":"Plausible.HTTPClient.Interface.html"},{"type":"callback","doc":"","title":"Plausible.HTTPClient.Interface.get/1","ref":"Plausible.HTTPClient.Interface.html#c:get/1"},{"type":"callback","doc":"","title":"Plausible.HTTPClient.Interface.get/2","ref":"Plausible.HTTPClient.Interface.html#c:get/2"},{"type":"callback","doc":"","title":"Plausible.HTTPClient.Interface.get/3","ref":"Plausible.HTTPClient.Interface.html#c:get/3"},{"type":"callback","doc":"","title":"Plausible.HTTPClient.Interface.post/3","ref":"Plausible.HTTPClient.Interface.html#c:post/3"},{"type":"callback","doc":"","title":"Plausible.HTTPClient.Interface.post/4","ref":"Plausible.HTTPClient.Interface.html#c:post/4"},{"type":"type","doc":"","title":"Plausible.HTTPClient.Interface.finch_request_opts/0","ref":"Plausible.HTTPClient.Interface.html#t:finch_request_opts/0"},{"type":"type","doc":"","title":"Plausible.HTTPClient.Interface.headers/0","ref":"Plausible.HTTPClient.Interface.html#t:headers/0"},{"type":"type","doc":"","title":"Plausible.HTTPClient.Interface.params/0","ref":"Plausible.HTTPClient.Interface.html#t:params/0"},{"type":"type","doc":"","title":"Plausible.HTTPClient.Interface.response/0","ref":"Plausible.HTTPClient.Interface.html#t:response/0"},{"type":"type","doc":"","title":"Plausible.HTTPClient.Interface.url/0","ref":"Plausible.HTTPClient.Interface.html#t:url/0"},{"type":"module","doc":"","title":"Plausible.HTTPClient.Non200Error","ref":"Plausible.HTTPClient.Non200Error.html"},{"type":"function","doc":"","title":"Plausible.HTTPClient.Non200Error.new/1","ref":"Plausible.HTTPClient.Non200Error.html#new/1"},{"type":"type","doc":"","title":"Plausible.HTTPClient.Non200Error.t/0","ref":"Plausible.HTTPClient.Non200Error.html#t:t/0"},{"type":"module","doc":"HelpScout callback API logic.","title":"Plausible.HelpScout","ref":"Plausible.HelpScout.html"},{"type":"function","doc":"","title":"Plausible.HelpScout.get_details_for_customer/1","ref":"Plausible.HelpScout.html#get_details_for_customer/1"},{"type":"function","doc":"","title":"Plausible.HelpScout.get_details_for_emails/2","ref":"Plausible.HelpScout.html#get_details_for_emails/2"},{"type":"function","doc":"","title":"Plausible.HelpScout.search_users/2","ref":"Plausible.HelpScout.html#search_users/2"},{"type":"function","doc":"","title":"Plausible.HelpScout.signature_errors/0","ref":"Plausible.HelpScout.html#signature_errors/0"},{"type":"function","doc":"Validates signature against secret key configured for the\nHelpScout application.\n\nNOTE: HelpScout signature generation procedure at\nhttps://developer.helpscout.com/apps/guides/signature-validation/\nfails to mention that it's implicitly dependent on request params\norder getting preserved. PHP arrays are ordered maps, so they provide\nthis guarantee. Here, on the other hand, we have to determine the original\norder of the keys directly from the query string and serialize\nparams to JSON using wrapper struct, informing Jason to put the values\nin the serialized object in this particular order matching query string.","title":"Plausible.HelpScout.validate_signature/1","ref":"Plausible.HelpScout.html#validate_signature/1"},{"type":"type","doc":"","title":"Plausible.HelpScout.signature_error/0","ref":"Plausible.HelpScout.html#t:signature_error/0"},{"type":"module","doc":"Provides a vault that will be used to encrypt/decrypt the stored HelpScout API access tokens.","title":"Plausible.HelpScout.Vault","ref":"Plausible.HelpScout.Vault.html"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.HelpScout.Vault.child_spec/1","ref":"Plausible.HelpScout.Vault.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.HelpScout.Vault.start_link/1","ref":"Plausible.HelpScout.Vault.html#start_link/1"},{"type":"module","doc":"Common helpers for JSON handling","title":"Plausible.Helpers.JSON","ref":"Plausible.Helpers.JSON.html"},{"type":"function","doc":"","title":"Plausible.Helpers.JSON.decode_or_fallback/1","ref":"Plausible.Helpers.JSON.html#decode_or_fallback/1"},{"type":"module","doc":"A dedicated repo for import related mutations","title":"Plausible.ImportDeletionRepo","ref":"Plausible.ImportDeletionRepo.html"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.aggregate/3","ref":"Plausible.ImportDeletionRepo.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.aggregate/4","ref":"Plausible.ImportDeletionRepo.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.all/2","ref":"Plausible.ImportDeletionRepo.html#all/2"},{"type":"function","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","title":"Plausible.ImportDeletionRepo.alter_update_all/3","ref":"Plausible.ImportDeletionRepo.html#alter_update_all/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.checked_out?/0","ref":"Plausible.ImportDeletionRepo.html#checked_out?/0"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.checkout/2","ref":"Plausible.ImportDeletionRepo.html#checkout/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.child_spec/1","ref":"Plausible.ImportDeletionRepo.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.config/0","ref":"Plausible.ImportDeletionRepo.html#config/0"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.default_options/1","ref":"Plausible.ImportDeletionRepo.html#default_options/1"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.delete/2","ref":"Plausible.ImportDeletionRepo.html#delete/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.delete!/2","ref":"Plausible.ImportDeletionRepo.html#delete!/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.delete_all/2","ref":"Plausible.ImportDeletionRepo.html#delete_all/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","title":"Plausible.ImportDeletionRepo.disconnect_all/2","ref":"Plausible.ImportDeletionRepo.html#disconnect_all/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.exists?/2","ref":"Plausible.ImportDeletionRepo.html#exists?/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.get/3","ref":"Plausible.ImportDeletionRepo.html#get/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.get!/3","ref":"Plausible.ImportDeletionRepo.html#get!/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.get_by/3","ref":"Plausible.ImportDeletionRepo.html#get_by/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.get_by!/3","ref":"Plausible.ImportDeletionRepo.html#get_by!/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.get_dynamic_repo/0","ref":"Plausible.ImportDeletionRepo.html#get_dynamic_repo/0"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.insert/2","ref":"Plausible.ImportDeletionRepo.html#insert/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.insert!/2","ref":"Plausible.ImportDeletionRepo.html#insert!/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.insert_all/3","ref":"Plausible.ImportDeletionRepo.html#insert_all/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.insert_or_update/2","ref":"Plausible.ImportDeletionRepo.html#insert_or_update/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.insert_or_update!/2","ref":"Plausible.ImportDeletionRepo.html#insert_or_update!/2"},{"type":"function","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","title":"Plausible.ImportDeletionRepo.insert_stream/3","ref":"Plausible.ImportDeletionRepo.html#insert_stream/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.load/2","ref":"Plausible.ImportDeletionRepo.html#load/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.one/2","ref":"Plausible.ImportDeletionRepo.html#one/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.one!/2","ref":"Plausible.ImportDeletionRepo.html#one!/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.preload/3","ref":"Plausible.ImportDeletionRepo.html#preload/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.prepare_query/3","ref":"Plausible.ImportDeletionRepo.html#prepare_query/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.put_dynamic_repo/1","ref":"Plausible.ImportDeletionRepo.html#put_dynamic_repo/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","title":"Plausible.ImportDeletionRepo.query/3","ref":"Plausible.ImportDeletionRepo.html#query/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","title":"Plausible.ImportDeletionRepo.query!/3","ref":"Plausible.ImportDeletionRepo.html#query!/3"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.reload/2","ref":"Plausible.ImportDeletionRepo.html#reload/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.reload!/2","ref":"Plausible.ImportDeletionRepo.html#reload!/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.start_link/1","ref":"Plausible.ImportDeletionRepo.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.stop/1","ref":"Plausible.ImportDeletionRepo.html#stop/1"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.stream/2","ref":"Plausible.ImportDeletionRepo.html#stream/2"},{"type":"function","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","title":"Plausible.ImportDeletionRepo.to_inline_sql/2","ref":"Plausible.ImportDeletionRepo.html#to_inline_sql/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","title":"Plausible.ImportDeletionRepo.to_sql/2","ref":"Plausible.ImportDeletionRepo.html#to_sql/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.update/2","ref":"Plausible.ImportDeletionRepo.html#update/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.update!/2","ref":"Plausible.ImportDeletionRepo.html#update!/2"},{"type":"function","doc":"","title":"Plausible.ImportDeletionRepo.update_all/3","ref":"Plausible.ImportDeletionRepo.html#update_all/3"},{"type":"module","doc":"Context for managing site statistics imports.\n\nFor list of currently supported import sources see `Plausible.Imported.ImportSources`.\n\nFor more information on implementing importers, see `Plausible.Imported.Importer`.","title":"Plausible.Imported","ref":"Plausible.Imported.html"},{"type":"function","doc":"","title":"Plausible.Imported.clamp_dates/3","ref":"Plausible.Imported.html#clamp_dates/3"},{"type":"function","doc":"","title":"Plausible.Imported.clamp_dates/4","ref":"Plausible.Imported.html#clamp_dates/4"},{"type":"function","doc":"","title":"Plausible.Imported.delete_imports_for_site/1","ref":"Plausible.Imported.html#delete_imports_for_site/1"},{"type":"function","doc":"","title":"Plausible.Imported.get_cutoff_date/1","ref":"Plausible.Imported.html#get_cutoff_date/1"},{"type":"function","doc":"","title":"Plausible.Imported.get_import/2","ref":"Plausible.Imported.html#get_import/2"},{"type":"function","doc":"","title":"Plausible.Imported.get_imports_date_range/1","ref":"Plausible.Imported.html#get_imports_date_range/1"},{"type":"function","doc":"","title":"Plausible.Imported.get_legacy_import/1","ref":"Plausible.Imported.html#get_legacy_import/1"},{"type":"function","doc":"","title":"Plausible.Imported.get_occupied_date_ranges/1","ref":"Plausible.Imported.html#get_occupied_date_ranges/1"},{"type":"function","doc":"","title":"Plausible.Imported.goals_with_path/0","ref":"Plausible.Imported.html#goals_with_path/0"},{"type":"function","doc":"","title":"Plausible.Imported.goals_with_url/0","ref":"Plausible.Imported.html#goals_with_url/0"},{"type":"function","doc":"","title":"Plausible.Imported.imported_custom_props/0","ref":"Plausible.Imported.html#imported_custom_props/0"},{"type":"function","doc":"","title":"Plausible.Imported.list_all_imports/2","ref":"Plausible.Imported.html#list_all_imports/2"},{"type":"function","doc":"","title":"Plausible.Imported.list_complete_import_ids/1","ref":"Plausible.Imported.html#list_complete_import_ids/1"},{"type":"function","doc":"","title":"Plausible.Imported.listen/0","ref":"Plausible.Imported.html#listen/0"},{"type":"function","doc":"","title":"Plausible.Imported.load_import_data/1","ref":"Plausible.Imported.html#load_import_data/1"},{"type":"function","doc":"","title":"Plausible.Imported.max_complete_imports/0","ref":"Plausible.Imported.html#max_complete_imports/0"},{"type":"function","doc":"","title":"Plausible.Imported.other_imports_in_progress?/1","ref":"Plausible.Imported.html#other_imports_in_progress?/1"},{"type":"function","doc":"","title":"Plausible.Imported.schemas/0","ref":"Plausible.Imported.html#schemas/0"},{"type":"function","doc":"","title":"Plausible.Imported.tables/0","ref":"Plausible.Imported.html#tables/0"},{"type":"module","doc":"This GenServer inserts records into Clickhouse `imported_*` tables. Multiple buffers are\nautomatically created for each table. Records are flushed when the table buffer reaches the\nmaximum size, defined by `max_buffer_size/0`.","title":"Plausible.Imported.Buffer","ref":"Plausible.Imported.Buffer.html"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.Imported.Buffer.child_spec/1","ref":"Plausible.Imported.Buffer.html#child_spec/1"},{"type":"function","doc":"Flushes all table buffers to Clickhouse.","title":"Plausible.Imported.Buffer.flush/2","ref":"Plausible.Imported.Buffer.html#flush/2"},{"type":"function","doc":"","title":"Plausible.Imported.Buffer.handle_continue/2","ref":"Plausible.Imported.Buffer.html#handle_continue/2"},{"type":"function","doc":"","title":"Plausible.Imported.Buffer.init/1","ref":"Plausible.Imported.Buffer.html#init/1"},{"type":"function","doc":"Puts the given records into the table buffer.","title":"Plausible.Imported.Buffer.insert_many/3","ref":"Plausible.Imported.Buffer.html#insert_many/3"},{"type":"function","doc":"Returns the total count of items in the given table buffer.","title":"Plausible.Imported.Buffer.size/2","ref":"Plausible.Imported.Buffer.html#size/2"},{"type":"function","doc":"","title":"Plausible.Imported.Buffer.start_link/1","ref":"Plausible.Imported.Buffer.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.Imported.Buffer.stop/1","ref":"Plausible.Imported.Buffer.html#stop/1"},{"type":"module","doc":"CSV importer from either S3 for which it uses ClickHouse [s3 table function](https://clickhouse.com/docs/en/sql-reference/table-functions/s3)\nor from local storage for which it uses [input function.](https://clickhouse.com/docs/en/sql-reference/table-functions/input)","title":"Plausible.Imported.CSVImporter","ref":"Plausible.Imported.CSVImporter.html"},{"type":"function","doc":"Extracts min/max date range from a list of uploads.\n\nExamples:\n\n iex> date_range([\n ...> %{\"filename\" => \"imported_devices_20190101_20210101.csv\"},\n ...> \"pages_20200101_20220101.csv\"\n ...> ])\n Date.range(~D[2019-01-01], ~D[2022-01-01])\n\n iex> date_range([])\n nil","title":"Plausible.Imported.CSVImporter.date_range/1","ref":"Plausible.Imported.CSVImporter.html#date_range/1"},{"type":"function","doc":"Extracts the table name from the provided filename.\n\nRaises if the filename doesn't conform to the expected format.\n\nExamples:\n\n iex> extract_table(\"my_data.csv\")\n ** (ArgumentError) invalid filename\n\n iex> extract_table(\"imported_devices_00010101_20250101.csv\")\n \"imported_devices\"\n\n iex> extract_table(\"devices_00010101_20250101.csv\")\n \"imported_devices\"","title":"Plausible.Imported.CSVImporter.extract_table/1","ref":"Plausible.Imported.CSVImporter.html#extract_table/1"},{"type":"function","doc":"Returns local directory for CSV imports storage.\n\nBuilds upon `$DATA_DIR`, `$PERSISTENT_CACHE_DIR` or `$DEFAULT_DATA_DIR` (if set) and falls back to /tmp.\n\n`$DEFAULT_DATA_DIR` is set to `/var/lib/plausible` in container images.\n\nExamples:\n\n iex> local_dir = local_dir(_site_id = 37)\n iex> String.ends_with?(local_dir, \"/plausible-imports/37\")\n true","title":"Plausible.Imported.CSVImporter.local_dir/1","ref":"Plausible.Imported.CSVImporter.html#local_dir/1"},{"type":"function","doc":"","title":"Plausible.Imported.CSVImporter.new_import/3","ref":"Plausible.Imported.CSVImporter.html#new_import/3"},{"type":"function","doc":"Extracts table name and min/max dates from the filename.\n\nExamples:\n\n iex> parse_filename!(\"my_data.csv\")\n ** (ArgumentError) invalid filename\n\n iex> parse_filename!(\"imported_devices_00010101_20250101.csv\")\n {\"imported_devices\", ~D[0001-01-01], ~D[2025-01-01]}\n\n iex> parse_filename!(\"devices_00010101_20250101.csv\")\n {\"imported_devices\", ~D[0001-01-01], ~D[2025-01-01]}","title":"Plausible.Imported.CSVImporter.parse_filename!/1","ref":"Plausible.Imported.CSVImporter.html#parse_filename!/1"},{"type":"function","doc":"Checks if the provided filename conforms to the expected format.\n\nExamples:\n\n iex> valid_filename?(\"my_data.csv\")\n false\n\n iex> valid_filename?(\"imported_devices_00010101_20250101.csv\")\n true\n\n iex> valid_filename?(\"devices_00010101_20250101.csv\")\n true","title":"Plausible.Imported.CSVImporter.valid_filename?/1","ref":"Plausible.Imported.CSVImporter.html#valid_filename?/1"},{"type":"module","doc":"Import implementation for Google Analytics 4.","title":"Plausible.Imported.GoogleAnalytics4","ref":"Plausible.Imported.GoogleAnalytics4.html"},{"type":"function","doc":"","title":"Plausible.Imported.GoogleAnalytics4.from_report/4","ref":"Plausible.Imported.GoogleAnalytics4.html#from_report/4"},{"type":"function","doc":"Imports stats from a Google Analytics 4 property to a Plausible site.\n\nThis function fetches Google Analytics 4 reports which are then passed in batches\nto Clickhouse by the `Plausible.Imported.Buffer` process.","title":"Plausible.Imported.GoogleAnalytics4.import_data/2","ref":"Plausible.Imported.GoogleAnalytics4.html#import_data/2"},{"type":"function","doc":"","title":"Plausible.Imported.GoogleAnalytics4.new_import/3","ref":"Plausible.Imported.GoogleAnalytics4.html#new_import/3"},{"type":"module","doc":"Definitions of import sources.","title":"Plausible.Imported.ImportSources","ref":"Plausible.Imported.ImportSources.html"},{"type":"function","doc":"","title":"Plausible.Imported.ImportSources.by_name/1","ref":"Plausible.Imported.ImportSources.html#by_name/1"},{"type":"function","doc":"","title":"Plausible.Imported.ImportSources.names/0","ref":"Plausible.Imported.ImportSources.html#names/0"},{"type":"behaviour","doc":"Behaviour that should be implemented for each import source.\n\nAll imports are executed as background jobs run via `Plausible.Workers.ImportAnalytics`\nOban worker. Each import source must define a module conforming `Importer` behaviour.\n\nThe callbacks that need to be implemented:\n\n* `name/0` - Returns import source name as an atom. Example: `:universal_analytics`.\n* `label/0` - Descriptive, display friendly name of the source.\n Example: \"Google Analytics\".\n* `email_template/0` - Name of the email template to use for notifications in\n `PlausibleWeb.Email` (`import_success` and `import_failure`). The template\n should have content customized for a particular source.\n* `parse_args/1` - Receives Oban job arguments coming from `new_import/3`. Whatever\n options were passed to `new_import/3` will be present in the input map with string\n keys and values serialized to primitives. If, for instance `start_date: ~D[2024-01-03]`\n is passed as an option, `parse_args/1` receives `%{..., \"start_date\" => \"2024-01-03\"}`.\n The expectation is parsing the map values producing a keyword list of options to\n pass to `import_data/2`.\n* `import_data/2` - Receives site import struct and options produced by `parse_args/1`.\n This is where all the import processing is done. The way the import is implemented\n is entirely arbitrary except the requirement that the process as a whole must\n by synchronous. The callback is expected to return either `:ok` or `{:ok, %{...}}`\n on successful import or `{:error, ...}` on failure. The map in success tuple is\n used for updating site import struct and is passed to `on_success/2` callback.\n Please note that error tuple should be only returned on errors that can't be\n recovered from. For transient errors, the import should throw an exception or\n simply crash. The error tuple has an alternative `{error, reason, opts}` form,\n where `opts` allow to skip purging imported data so far via `skip_purge?` flag\n and skip marking the import as failed and notifying the user via `skip_mark_failed?`\n flag. Both flags are booleans.\n* `before_start/2` - Optional callback run right before scheduling import job. It's\n expected to either return `{:ok, site_import}` for the import to proceed\n or `{:error, ...}` tuple, which will be returned from `new_import/3` call.\n The `site_import` can be altered or replaced at this stage. The second argument\n are opts passed to `new_import/3`.\n* `on_success/2` - Optional callback run once site import is completed. Receives map\n returned from `import_data/2`. Expected to always return `:ok`.\n* `on_failure/1` - Optional callback run when import job fails permanently.\n\nAll sources must be added to the list in `Plausible.Imported.ImportSources`.\n\nIn order to schedule a new import job using a given source, respective importer's\n`new_import/3` function must be called. It accepts site, user who is doing the import\nand any options necessary to carry out the import.\n\nThere's an expectation that `start_date` and `end_date` are provided either as options\npassed to `new_import/3` or data in map returned from `import_data/2`. If these parameters\nare not provided, the import will eventually crash. These parameters define time range\nof imported data which is in turn used for efficient querying.\n\nLogic running inside `import_data/2` is expected to populated all `imported_*` tables\nin ClickHouse with `import_id` column set to site import's ID.\n\nManaging any configuration or authentication prior to running import is outside of\nscope of importer logic and is expected to be implemented separately.","title":"Plausible.Imported.Importer","ref":"Plausible.Imported.Importer.html"},{"type":"behaviour","doc":"In case it's necessary to run the whole import job fully synchronously, the\n`Plausible.Workers.ImportAnalytics` worker sends an `Oban.Notifier` message\non completion, failure or transient failure of the import.\n\nA basic usage scenario looks like this:\n\n```elixir\n{:ok, job} = Plausible.Imported.NoopImporter.new_import(\n site,\n user,\n start_date: ~D[2005-01-01],\n end_date: Date.utc_today(),\n # this option is necessary to setup the calling process as listener\n listen?: true\n)\n\nimport_id = job.args[:import_id]\n\nreceive do\n {:notification, :analytics_imports_jobs, %{\"event\" => \"complete\", \"import_id\" => ^import_id}} ->\n IO.puts(\"Job completed\")\n\n {:notification, :analytics_imports_jobs, %{\"event\" => \"transient_fail\", \"import_id\" => ^import_id}} ->\n IO.puts(\"Job failed transiently\")\n\n {:notification, :analytics_imports_jobs, %{\"event\" => \"fail\", \"import_id\" => ^import_id}} ->\n IO.puts(\"Job failed permanently\")\nafter\n 15_000 ->\n IO.puts(\"Job didn't finish in 15 seconds\")\nend\n```\n\nIn a more realistic scenario, job scheduling will be done inside a GenServer process\nlike LiveView, where notifications can be listened for via `handle_info/2`.","title":"Running import fully synchronously - Plausible.Imported.Importer","ref":"Plausible.Imported.Importer.html#module-running-import-fully-synchronously"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.before_start/2","ref":"Plausible.Imported.Importer.html#c:before_start/2"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.email_template/0","ref":"Plausible.Imported.Importer.html#c:email_template/0"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.import_data/2","ref":"Plausible.Imported.Importer.html#c:import_data/2"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.label/0","ref":"Plausible.Imported.Importer.html#c:label/0"},{"type":"function","doc":"Allows to explicitly start listening for importer job notifications.\n\nListener must explicitly filter out a subset of imports that apply to the given context.","title":"Plausible.Imported.Importer.listen/0","ref":"Plausible.Imported.Importer.html#listen/0"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.name/0","ref":"Plausible.Imported.Importer.html#c:name/0"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.on_failure/1","ref":"Plausible.Imported.Importer.html#c:on_failure/1"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.on_success/2","ref":"Plausible.Imported.Importer.html#c:on_success/2"},{"type":"callback","doc":"","title":"Plausible.Imported.Importer.parse_args/1","ref":"Plausible.Imported.Importer.html#c:parse_args/1"},{"type":"module","doc":"Stub import implementation.","title":"Plausible.Imported.NoopImporter","ref":"Plausible.Imported.NoopImporter.html"},{"type":"function","doc":"","title":"Plausible.Imported.NoopImporter.new_import/3","ref":"Plausible.Imported.NoopImporter.html#new_import/3"},{"type":"module","doc":"Site import schema.","title":"Plausible.Imported.SiteImport","ref":"Plausible.Imported.SiteImport.html"},{"type":"function","doc":"","title":"Plausible.Imported.SiteImport.complete_changeset/2","ref":"Plausible.Imported.SiteImport.html#complete_changeset/2"},{"type":"macro","doc":"","title":"Plausible.Imported.SiteImport.completed/0","ref":"Plausible.Imported.SiteImport.html#completed/0"},{"type":"function","doc":"","title":"Plausible.Imported.SiteImport.create_changeset/3","ref":"Plausible.Imported.SiteImport.html#create_changeset/3"},{"type":"function","doc":"","title":"Plausible.Imported.SiteImport.fail_changeset/1","ref":"Plausible.Imported.SiteImport.html#fail_changeset/1"},{"type":"macro","doc":"","title":"Plausible.Imported.SiteImport.failed/0","ref":"Plausible.Imported.SiteImport.html#failed/0"},{"type":"macro","doc":"","title":"Plausible.Imported.SiteImport.importing/0","ref":"Plausible.Imported.SiteImport.html#importing/0"},{"type":"function","doc":"","title":"Plausible.Imported.SiteImport.label/1","ref":"Plausible.Imported.SiteImport.html#label/1"},{"type":"macro","doc":"","title":"Plausible.Imported.SiteImport.pending/0","ref":"Plausible.Imported.SiteImport.html#pending/0"},{"type":"function","doc":"","title":"Plausible.Imported.SiteImport.start_changeset/1","ref":"Plausible.Imported.SiteImport.html#start_changeset/1"},{"type":"type","doc":"","title":"Plausible.Imported.SiteImport.t/0","ref":"Plausible.Imported.SiteImport.html#t:t/0"},{"type":"module","doc":"Import implementation for Universal Analytics.\n\nNOTE: As importing from UA is no longer supported, this module\nis only used to support rendering existing imports.","title":"Plausible.Imported.UniversalAnalytics","ref":"Plausible.Imported.UniversalAnalytics.html"},{"type":"function","doc":"","title":"Plausible.Imported.UniversalAnalytics.new_import/3","ref":"Plausible.Imported.UniversalAnalytics.html#new_import/3"},{"type":"module","doc":"Write-centric Clickhouse access interface","title":"Plausible.IngestRepo","ref":"Plausible.IngestRepo.html"},{"type":"function","doc":"","title":"Plausible.IngestRepo.aggregate/3","ref":"Plausible.IngestRepo.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.aggregate/4","ref":"Plausible.IngestRepo.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.IngestRepo.all/2","ref":"Plausible.IngestRepo.html#all/2"},{"type":"function","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","title":"Plausible.IngestRepo.alter_update_all/3","ref":"Plausible.IngestRepo.html#alter_update_all/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.checked_out?/0","ref":"Plausible.IngestRepo.html#checked_out?/0"},{"type":"function","doc":"","title":"Plausible.IngestRepo.checkout/2","ref":"Plausible.IngestRepo.html#checkout/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.child_spec/1","ref":"Plausible.IngestRepo.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.IngestRepo.clustered_table?/1","ref":"Plausible.IngestRepo.html#clustered_table?/1"},{"type":"function","doc":"","title":"Plausible.IngestRepo.config/0","ref":"Plausible.IngestRepo.html#config/0"},{"type":"function","doc":"","title":"Plausible.IngestRepo.default_options/1","ref":"Plausible.IngestRepo.html#default_options/1"},{"type":"function","doc":"","title":"Plausible.IngestRepo.delete/2","ref":"Plausible.IngestRepo.html#delete/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.delete!/2","ref":"Plausible.IngestRepo.html#delete!/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.delete_all/2","ref":"Plausible.IngestRepo.html#delete_all/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","title":"Plausible.IngestRepo.disconnect_all/2","ref":"Plausible.IngestRepo.html#disconnect_all/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.exists?/2","ref":"Plausible.IngestRepo.html#exists?/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.get/3","ref":"Plausible.IngestRepo.html#get/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.get!/3","ref":"Plausible.IngestRepo.html#get!/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.get_by/3","ref":"Plausible.IngestRepo.html#get_by/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.get_by!/3","ref":"Plausible.IngestRepo.html#get_by!/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.get_dynamic_repo/0","ref":"Plausible.IngestRepo.html#get_dynamic_repo/0"},{"type":"function","doc":"","title":"Plausible.IngestRepo.insert/2","ref":"Plausible.IngestRepo.html#insert/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.insert!/2","ref":"Plausible.IngestRepo.html#insert!/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.insert_all/3","ref":"Plausible.IngestRepo.html#insert_all/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.insert_or_update/2","ref":"Plausible.IngestRepo.html#insert_or_update/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.insert_or_update!/2","ref":"Plausible.IngestRepo.html#insert_or_update!/2"},{"type":"function","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","title":"Plausible.IngestRepo.insert_stream/3","ref":"Plausible.IngestRepo.html#insert_stream/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.load/2","ref":"Plausible.IngestRepo.html#load/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.one/2","ref":"Plausible.IngestRepo.html#one/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.one!/2","ref":"Plausible.IngestRepo.html#one!/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.preload/3","ref":"Plausible.IngestRepo.html#preload/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.prepare_query/3","ref":"Plausible.IngestRepo.html#prepare_query/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.put_dynamic_repo/1","ref":"Plausible.IngestRepo.html#put_dynamic_repo/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","title":"Plausible.IngestRepo.query/3","ref":"Plausible.IngestRepo.html#query/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","title":"Plausible.IngestRepo.query!/3","ref":"Plausible.IngestRepo.html#query!/3"},{"type":"function","doc":"","title":"Plausible.IngestRepo.reload/2","ref":"Plausible.IngestRepo.html#reload/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.reload!/2","ref":"Plausible.IngestRepo.html#reload!/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.start_link/1","ref":"Plausible.IngestRepo.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.IngestRepo.stop/1","ref":"Plausible.IngestRepo.html#stop/1"},{"type":"function","doc":"","title":"Plausible.IngestRepo.stream/2","ref":"Plausible.IngestRepo.html#stream/2"},{"type":"function","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","title":"Plausible.IngestRepo.to_inline_sql/2","ref":"Plausible.IngestRepo.html#to_inline_sql/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","title":"Plausible.IngestRepo.to_sql/2","ref":"Plausible.IngestRepo.html#to_sql/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.update/2","ref":"Plausible.IngestRepo.html#update/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.update!/2","ref":"Plausible.IngestRepo.html#update!/2"},{"type":"function","doc":"","title":"Plausible.IngestRepo.update_all/3","ref":"Plausible.IngestRepo.html#update_all/3"},{"type":"module","doc":"This is instrumentation necessary for keeping track of per-domain\ninternal metrics. Due to metric labels cardinality (domain x metric_name),\nthese statistics are not suitable for prometheus/grafana exposure,\nhence an internal storage is used.\n\nThe module installs `Counters.TelemetryHandler` and periodically\nflushes the internal counter aggregates via `Counters.Buffer` interface.\n\nThe underlying database schema is running `SummingMergeTree` engine.\nTo take advantage of automatic roll-ups it provides, upon dispatching the\nbuffered records to Clickhouse this module transforms each `event_timebucket`\naggregate into a 1-minute resolution.\n\nClickhouse connection is set to insert counters asynchronously every time\na pool checkout is made. Those properties are reverted once the insert is done\n(or naturally, if the connection crashes).","title":"Plausible.Ingestion.Counters","ref":"Plausible.Ingestion.Counters.html"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.child_spec/1","ref":"Plausible.Ingestion.Counters.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.enabled?/0","ref":"Plausible.Ingestion.Counters.html#enabled?/0"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.stop/1","ref":"Plausible.Ingestion.Counters.html#stop/1"},{"type":"module","doc":"A buffer aggregating counters for internal metrics, within 10 seconds time buckets.\n\nSee `Plausible.Ingestion.Counters` for integration.\n\nFlushing is by default possible only once the 10s bucket is complete\n(its window has moved). This is to avoid race conditions \nwhen clearing up the buffer on dequeue - because there is no atomic \"get and delete\",\nand items are buffered concurrently, there is a gap between get and delete\nin which items written may disappear otherwise.\n\n`aggregate_bucket_fn` and `flush_boundary_fn` control that semantics and\nare configurable only for test purposes.","title":"Plausible.Ingestion.Counters.Buffer","ref":"Plausible.Ingestion.Counters.Buffer.html"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.Buffer.aggregate/4","ref":"Plausible.Ingestion.Counters.Buffer.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.Buffer.bucket_10s/1","ref":"Plausible.Ingestion.Counters.Buffer.html#bucket_10s/1"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.Buffer.flush/2","ref":"Plausible.Ingestion.Counters.Buffer.html#flush/2"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.Buffer.new/2","ref":"Plausible.Ingestion.Counters.Buffer.html#new/2"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.Buffer.previous_10s/1","ref":"Plausible.Ingestion.Counters.Buffer.html#previous_10s/1"},{"type":"type","doc":"","title":"Plausible.Ingestion.Counters.Buffer.bucket_fn_opt/0","ref":"Plausible.Ingestion.Counters.Buffer.html#t:bucket_fn_opt/0"},{"type":"type","doc":"","title":"Plausible.Ingestion.Counters.Buffer.t/0","ref":"Plausible.Ingestion.Counters.Buffer.html#t:t/0"},{"type":"type","doc":"","title":"Plausible.Ingestion.Counters.Buffer.unix_timestamp/0","ref":"Plausible.Ingestion.Counters.Buffer.html#t:unix_timestamp/0"},{"type":"module","doc":"Clickhouse schema for storing ingest counter metrics","title":"Plausible.Ingestion.Counters.Record","ref":"Plausible.Ingestion.Counters.Record.html"},{"type":"type","doc":"","title":"Plausible.Ingestion.Counters.Record.t/0","ref":"Plausible.Ingestion.Counters.Record.html#t:t/0"},{"type":"module","doc":"Subscribes to telemetry events emitted by `Plausible.Ingestion.Event`.\nEvery time a request derived event is either dispatched to clickhouse or dropped,\na telemetry event is emitted respectively. That event is captured here,\nits metadata is extracted and sent for internal stats aggregation via\n`Counters.Buffer` interface.","title":"Plausible.Ingestion.Counters.TelemetryHandler","ref":"Plausible.Ingestion.Counters.TelemetryHandler.html"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.TelemetryHandler.handle_event/4","ref":"Plausible.Ingestion.Counters.TelemetryHandler.html#handle_event/4"},{"type":"function","doc":"","title":"Plausible.Ingestion.Counters.TelemetryHandler.install/1","ref":"Plausible.Ingestion.Counters.TelemetryHandler.html#install/1"},{"type":"module","doc":"This module exposes the `build_and_buffer/1` function capable of\nturning %Plausible.Ingestion.Request{} into a series of events that in turn\nare uniformly either buffered in batches (to Clickhouse) or dropped\n(e.g. due to spam blocklist) from the processing pipeline.","title":"Plausible.Ingestion.Event","ref":"Plausible.Ingestion.Event.html"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.build_and_buffer/1","ref":"Plausible.Ingestion.Event.html#build_and_buffer/1"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.emit_telemetry_buffered/1","ref":"Plausible.Ingestion.Event.html#emit_telemetry_buffered/1"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.emit_telemetry_dropped/2","ref":"Plausible.Ingestion.Event.html#emit_telemetry_dropped/2"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.telemetry_event_buffered/0","ref":"Plausible.Ingestion.Event.html#telemetry_event_buffered/0"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.telemetry_event_dropped/0","ref":"Plausible.Ingestion.Event.html#telemetry_event_dropped/0"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.telemetry_pipeline_step_duration/0","ref":"Plausible.Ingestion.Event.html#telemetry_pipeline_step_duration/0"},{"type":"type","doc":"","title":"Plausible.Ingestion.Event.drop_reason/0","ref":"Plausible.Ingestion.Event.html#t:drop_reason/0"},{"type":"type","doc":"","title":"Plausible.Ingestion.Event.t/0","ref":"Plausible.Ingestion.Event.html#t:t/0"},{"type":"module","doc":"Revenue specific functions for the ingestion scope","title":"Plausible.Ingestion.Event.Revenue","ref":"Plausible.Ingestion.Event.Revenue.html"},{"type":"function","doc":"","title":"Plausible.Ingestion.Event.Revenue.get_revenue_attrs/1","ref":"Plausible.Ingestion.Event.Revenue.html#get_revenue_attrs/1"},{"type":"module","doc":"The %Plausible.Ingestion.Request{} struct stores all needed fields\nto create an event downstream. Pre-eliminary validation is made\nto detect user errors early.","title":"Plausible.Ingestion.Request","ref":"Plausible.Ingestion.Request.html"},{"type":"function","doc":"Builds and initially validates %Plausible.Ingestion.Request{} struct from %Plug.Conn{}.","title":"Plausible.Ingestion.Request.build/2","ref":"Plausible.Ingestion.Request.html#build/2"},{"type":"function","doc":"Removes the \"www\" part of a hostname.","title":"Plausible.Ingestion.Request.sanitize_hostname/1","ref":"Plausible.Ingestion.Request.html#sanitize_hostname/1"},{"type":"type","doc":"","title":"Plausible.Ingestion.Request.t/0","ref":"Plausible.Ingestion.Request.html#t:t/0"},{"type":"module","doc":"Revenue specific functions for the ingestion scope","title":"Plausible.Ingestion.Request.Revenue","ref":"Plausible.Ingestion.Request.Revenue.html"},{"type":"function","doc":"","title":"Plausible.Ingestion.Request.Revenue.put_revenue_source/2","ref":"Plausible.Ingestion.Request.Revenue.html#put_revenue_source/2"},{"type":"module","doc":"This module ensures that you cannot run Plausible Analytics Enterprise Edition without a valid license key.\n The software contained within the ee/ and assets/js/dashboard/ee directories are Copyright © Plausible Insights OÜ.\n We have made this code available solely for informational and transparency purposes. No rights are granted to use,\n distribute, or exploit this software in any form.\n\n Any attempt to disable or modify the behavior of this module will be considered a violation of copyright.\n If you wish to use the Plausible Analytics Enterprise Edition for your own requirements, please contact us\n at hello@plausible.io to discuss obtaining a license.","title":"Plausible.License","ref":"Plausible.License.html"},{"type":"function","doc":"","title":"Plausible.License.ensure_valid_license/0","ref":"Plausible.License.html#ensure_valid_license/0"},{"type":"module","doc":"Temporary fix for `Phoenix.LiveViewTest.render_component/2` failing CI with warnings.\n\nThis module can be removed once Plausible switches to `phoenix_live_view ~> 1.0.0`","title":"Plausible.LiveViewTest","ref":"Plausible.LiveViewTest.html"},{"type":"macro","doc":"Same as `Phoenix.LiveViewTest.render_component/2` but with backported fixes from\nhttps://github.com/phoenixframework/phoenix_live_view/commit/489e8de024e03976e9ae38138eec517fbd456d27","title":"Plausible.LiveViewTest.render_component/3","ref":"Plausible.LiveViewTest.html#render_component/3"},{"type":"module","doc":"","title":"Plausible.Mailer","ref":"Plausible.Mailer.html"},{"type":"function","doc":"","title":"Plausible.Mailer.deliver/1","ref":"Plausible.Mailer.html#deliver/1"},{"type":"function","doc":"","title":"Plausible.Mailer.deliver_later/2","ref":"Plausible.Mailer.html#deliver_later/2"},{"type":"function","doc":"","title":"Plausible.Mailer.deliver_later!/2","ref":"Plausible.Mailer.html#deliver_later!/2"},{"type":"function","doc":"","title":"Plausible.Mailer.deliver_now/2","ref":"Plausible.Mailer.html#deliver_now/2"},{"type":"function","doc":"","title":"Plausible.Mailer.deliver_now!/2","ref":"Plausible.Mailer.html#deliver_now!/2"},{"type":"function","doc":"","title":"Plausible.Mailer.send/1","ref":"Plausible.Mailer.html#send/1"},{"type":"type","doc":"","title":"Plausible.Mailer.result/0","ref":"Plausible.Mailer.html#t:result/0"},{"type":"module","doc":"Base module for to use in Clickhouse migrations","title":"Plausible.MigrationUtils","ref":"Plausible.MigrationUtils.html"},{"type":"function","doc":"","title":"Plausible.MigrationUtils.dictionary_connection_params/0","ref":"Plausible.MigrationUtils.html#dictionary_connection_params/0"},{"type":"function","doc":"","title":"Plausible.MigrationUtils.on_cluster_statement/1","ref":"Plausible.MigrationUtils.html#on_cluster_statement/1"},{"type":"module","doc":"[Custom OpenTelemetry sampler](https://hexdocs.pm/opentelemetry/readme.html#samplers)\nimplementation that ignores particular traces to reduce noise. Ingestion\nHTTP requests and queries to Oban tables are ignored, for example.\n\nFor non-ignored traces, implements trace ID ratio-based sampling following the method\nfrom [built-in sampler](https://github.com/open-telemetry/opentelemetry-erlang/blob/main/apps/opentelemetry/src/otel_sampler_trace_id_ratio_based.erl).","title":"Plausible.OpenTelemetry.Sampler","ref":"Plausible.OpenTelemetry.Sampler.html"},{"type":"module","doc":"","title":"Plausible.PaddleApi.Mock","ref":"Plausible.PaddleApi.Mock.html"},{"type":"function","doc":"","title":"Plausible.PaddleApi.Mock.fetch_prices/2","ref":"Plausible.PaddleApi.Mock.html#fetch_prices/2"},{"type":"function","doc":"","title":"Plausible.PaddleApi.Mock.get_invoices/1","ref":"Plausible.PaddleApi.Mock.html#get_invoices/1"},{"type":"function","doc":"","title":"Plausible.PaddleApi.Mock.get_subscription/1","ref":"Plausible.PaddleApi.Mock.html#get_subscription/1"},{"type":"function","doc":"","title":"Plausible.PaddleApi.Mock.update_subscription/2","ref":"Plausible.PaddleApi.Mock.html#update_subscription/2"},{"type":"function","doc":"","title":"Plausible.PaddleApi.Mock.update_subscription_preview/2","ref":"Plausible.PaddleApi.Mock.html#update_subscription_preview/2"},{"type":"module","doc":"Cursor-based pagination.","title":"Plausible.Pagination","ref":"Plausible.Pagination.html"},{"type":"function","doc":"","title":"Plausible.Pagination.paginate/4","ref":"Plausible.Pagination.html#paginate/4"},{"type":"module","doc":"Context module for querying API capabilities","title":"Plausible.Plugins.API.Capabilities","ref":"Plausible.Plugins.API.Capabilities.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Capabilities.get/1","ref":"Plausible.Plugins.API.Capabilities.html#get/1"},{"type":"module","doc":"Plugins API context module for Custom Props.\nAll high level Custom Props operations should be implemented here.","title":"Plausible.Plugins.API.CustomProps","ref":"Plausible.Plugins.API.CustomProps.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.CustomProps.disable/2","ref":"Plausible.Plugins.API.CustomProps.html#disable/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.CustomProps.enable/2","ref":"Plausible.Plugins.API.CustomProps.html#enable/2"},{"type":"module","doc":"Plugins API context module for Funnels.\nAll high level Funnel operations should be implemented here.","title":"Plausible.Plugins.API.Funnels","ref":"Plausible.Plugins.API.Funnels.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Funnels.create/2","ref":"Plausible.Plugins.API.Funnels.html#create/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Funnels.get/2","ref":"Plausible.Plugins.API.Funnels.html#get/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Funnels.get_funnels/2","ref":"Plausible.Plugins.API.Funnels.html#get_funnels/2"},{"type":"type","doc":"","title":"Plausible.Plugins.API.Funnels.create_request/0","ref":"Plausible.Plugins.API.Funnels.html#t:create_request/0"},{"type":"module","doc":"Plugins API context module for Goals.\nAll high level Goal operations should be implemented here.","title":"Plausible.Plugins.API.Goals","ref":"Plausible.Plugins.API.Goals.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Goals.create/2","ref":"Plausible.Plugins.API.Goals.html#create/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Goals.delete/2","ref":"Plausible.Plugins.API.Goals.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Goals.get/2","ref":"Plausible.Plugins.API.Goals.html#get/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Goals.get_goals/2","ref":"Plausible.Plugins.API.Goals.html#get_goals/2"},{"type":"type","doc":"","title":"Plausible.Plugins.API.Goals.create_request/0","ref":"Plausible.Plugins.API.Goals.html#t:create_request/0"},{"type":"module","doc":"Plugins API context module for Shared Links.\nAll high level Shared Links operations should be implemented here.","title":"Plausible.Plugins.API.SharedLinks","ref":"Plausible.Plugins.API.SharedLinks.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.SharedLinks.get/2","ref":"Plausible.Plugins.API.SharedLinks.html#get/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.SharedLinks.get_or_create/3","ref":"Plausible.Plugins.API.SharedLinks.html#get_or_create/3"},{"type":"function","doc":"","title":"Plausible.Plugins.API.SharedLinks.get_shared_links/2","ref":"Plausible.Plugins.API.SharedLinks.html#get_shared_links/2"},{"type":"module","doc":"Ecto schema for Plugins API Tokens.\nTokens are stored hashed and require a description.\n\nTokens are considered secret, although the Plugins API\nby nature will expose very little, if any, destructive/insecure operations.\n\nThe raw token version is meant to be presented to the user upon creation.\nIt is prefixed with a plain text identifier allowing source scanning\nfor leaked secrets.","title":"Plausible.Plugins.API.Token","ref":"Plausible.Plugins.API.Token.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Token.generate/1","ref":"Plausible.Plugins.API.Token.html#generate/1"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Token.hash/1","ref":"Plausible.Plugins.API.Token.html#hash/1"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Token.insert_changeset/3","ref":"Plausible.Plugins.API.Token.html#insert_changeset/3"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Token.last_used_humanize/1","ref":"Plausible.Plugins.API.Token.html#last_used_humanize/1"},{"type":"function","doc":"Raw tokens are prefixed so that tools like\nhttps://docs.github.com/en/code-security/secret-scanning/about-secret-scanning\ncan scan repositories for accidental secret commits.","title":"Plausible.Plugins.API.Token.prefix/0","ref":"Plausible.Plugins.API.Token.html#prefix/0"},{"type":"type","doc":"","title":"Plausible.Plugins.API.Token.t/0","ref":"Plausible.Plugins.API.Token.html#t:t/0"},{"type":"module","doc":"Context module for Plugins API Tokens.\nExposes high-level operation for token-based authentication flows.","title":"Plausible.Plugins.API.Tokens","ref":"Plausible.Plugins.API.Tokens.html"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Tokens.any?/1","ref":"Plausible.Plugins.API.Tokens.html#any?/1"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Tokens.create/3","ref":"Plausible.Plugins.API.Tokens.html#create/3"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Tokens.delete/2","ref":"Plausible.Plugins.API.Tokens.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Tokens.find/1","ref":"Plausible.Plugins.API.Tokens.html#find/1"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Tokens.list/1","ref":"Plausible.Plugins.API.Tokens.html#list/1"},{"type":"function","doc":"","title":"Plausible.Plugins.API.Tokens.update_last_seen/2","ref":"Plausible.Plugins.API.Tokens.html#update_last_seen/2"},{"type":"module","doc":"","title":"Plausible.PromEx","ref":"Plausible.PromEx.html"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.PromEx.child_spec/1","ref":"Plausible.PromEx.html#child_spec/1"},{"type":"module","doc":"Custom PromEx plugin for instrumenting code within Plausible app.","title":"Plausible.PromEx.Plugins.PlausibleMetrics","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html"},{"type":"function","doc":"Fire telemetry events for various caches","title":"Plausible.PromEx.Plugins.PlausibleMetrics.execute_cache_metrics/0","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html#execute_cache_metrics/0"},{"type":"function","doc":"Add telemetry events for Session and Event write buffers","title":"Plausible.PromEx.Plugins.PlausibleMetrics.execute_write_buffer_metrics/0","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html#execute_write_buffer_metrics/0"},{"type":"function","doc":"","title":"Plausible.PromEx.Plugins.PlausibleMetrics.measure_duration/3","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html#measure_duration/3"},{"type":"module","doc":"Context module for handling custom event props.","title":"Plausible.Props","ref":"Plausible.Props.html"},{"type":"function","doc":"Allows a prop key or a list of props keys to be included in ClickHouse\nqueries. Allowing prop keys does not affect ingestion, as we don't want any\ndata to be dropped or lost.","title":"Plausible.Props.allow/2","ref":"Plausible.Props.html#allow/2"},{"type":"function","doc":"","title":"Plausible.Props.allow_changeset/2","ref":"Plausible.Props.html#allow_changeset/2"},{"type":"function","doc":"Allows the 300 most frequent props keys for a specific site over\nthe past 6 months.","title":"Plausible.Props.allow_existing_props/1","ref":"Plausible.Props.html#allow_existing_props/1"},{"type":"function","doc":"Returns the custom props allowed in queries for the given site. There are\ntwo factors deciding whether a custom property is allowed for a site.\n\n#","title":"Plausible.Props.allowed_for/2","ref":"Plausible.Props.html#allowed_for/2"},{"type":"function","doc":"Internally used keys (i.e. `[\"url\", \"path\", \"search_query\"]`) are always allowed,\neven for plans that don't include props. For any other props, access to the\nCustom Properties feature is required.\n\n#","title":"1. Subscription plan including the props feature. - Plausible.Props.allowed_for/2","ref":"Plausible.Props.html#allowed_for/2-1-subscription-plan-including-the-props-feature"},{"type":"function","doc":"For customers with a configured `allowed_event_props` list, this function\nreturns that list (+ internally used keys). That helps to filter out garbage\nprops which people might not want to see in their dashboards.\n\nWith the `bypass_setup?` boolean option you can override the requirement of\nthe site having set up props in the `allowed_event_props` list. For example,\nthis is currently used for fetching allowed properties in Stats API queries\nin order to ensure the props feature access.\n\nSince `allowed_event_props` was added after the props feature had already\nbeen used for a while, there are sites with `allowed_event_props = nil`. For\nthose sites, all custom properties that exist in the database are allowed to\nbe queried.","title":"2. The site having an `allowed_event_props` list configured. - Plausible.Props.allowed_for/2","ref":"Plausible.Props.html#allowed_for/2-2-the-site-having-an-allowed_event_props-list-configured"},{"type":"function","doc":"Returns whether the site has configured custom props or not.","title":"Plausible.Props.configured?/1","ref":"Plausible.Props.html#configured?/1"},{"type":"function","doc":"Removes previously allowed prop key(s) from the allow list. This means this\nprop key won't be included in ClickHouse queries. This doesn't drop any\nClickHouse data, nor affects ingestion.","title":"Plausible.Props.disallow/2","ref":"Plausible.Props.html#disallow/2"},{"type":"function","doc":"","title":"Plausible.Props.ensure_prop_key_accessible/2","ref":"Plausible.Props.html#ensure_prop_key_accessible/2"},{"type":"function","doc":"Lists prop keys used internally.\n\nThese props should be allowed by default, and should not be displayed in the\nprops settings page. For example, `url` is a special prop key used for file\ndownloads and outbound links. It doesn't make sense to remove this prop key\nfrom the allow list, or to suggest users to add this prop key.","title":"Plausible.Props.internal_keys/0","ref":"Plausible.Props.html#internal_keys/0"},{"type":"function","doc":"","title":"Plausible.Props.max_prop_key_length/0","ref":"Plausible.Props.html#max_prop_key_length/0"},{"type":"function","doc":"","title":"Plausible.Props.max_prop_value_length/0","ref":"Plausible.Props.html#max_prop_value_length/0"},{"type":"function","doc":"","title":"Plausible.Props.max_props/0","ref":"Plausible.Props.html#max_props/0"},{"type":"function","doc":"Queries the events table to fetch the 300 most frequent prop keys\nfor a specific site over the past 6 months, excluding keys that are already\nallowed.","title":"Plausible.Props.suggest_keys_to_allow/2","ref":"Plausible.Props.html#suggest_keys_to_allow/2"},{"type":"type","doc":"","title":"Plausible.Props.prop/0","ref":"Plausible.Props.html#t:prop/0"},{"type":"module","doc":"Deletes data from a site.\n\nStats are stored on Clickhouse, and unlike other databases data deletion is\ndone asynchronously.\n\nAll import tables have MergeTree's deduplication mechanism _disabled_ by setting\n`replicated_deduplication_window` from default 100 to 0. When enabled, every insert\ninto a given table is compared against hashes of 100 previous inserts (as complete\nparts, not concrete rows) and ignored when match is found. The prupose of that\nmechanism is making inserts of exact same batches idempotent when retrying them\nshortly after - for instance due to timeout, when the client can't easily tell if\nprevious insert succeeded or not. Deduplication, however, only considers inserts,\nnot mutations. Deletions do not affect stored hashes, so further inserts of parts\nthat were deleted will still be treated as duplicates. That's why this feature\nis disabled for import tables.\n\nAlthough deletions are asynchronous, the parts to delete are \"remembered\", so there's\nno risk of overlapping deletion causing problems with import following right after it.\n\nIMPORTANT: Deletion requires revision if/when import tables get moved to sharded CH\ncluster setup. Mutation queries, which have to be run with `ON CLUSTER` in such setup,\ndispatch independent queries across shards and those queries can start at different\ntimes. This in turn means risk of deletions corrupting data of follow-up inserts\nin some edge cases. Ideally, imported entries should be unique for a given import\n- an extra `import_id` column can be introduced, holding identifier. Last processed\nimport identifier should be stored with other site data and should be used for scoping\nimported stats queries. No longer used imports can then be safely removed fully\nasynchronously.\n\n- [Clickhouse `ALTER TABLE ... DELETE` Statement](https://clickhouse.com/docs/en/sql-reference/statements/alter/delete)\n- [Synchronicity of `ALTER` Queries](https://clickhouse.com/docs/en/sql-reference/statements/alter/#synchronicity-of-alter-queries)","title":"Plausible.Purge","ref":"Plausible.Purge.html"},{"type":"function","doc":"Deletes imported stats from and clears the `stats_start_date` field.\n\nThe `stats_start_date` is expected to get repopulated the next time\n`Plausible.Sites.stats_start_date/1` is called.\n\nIf the input argument is a site, all imported stats are deleted. If it's a site import,\nonly imported stats for that import are deleted.","title":"Plausible.Purge.delete_imported_stats!/1","ref":"Plausible.Purge.html#delete_imported_stats!/1"},{"type":"function","doc":"","title":"Plausible.Purge.delete_imported_stats!/2","ref":"Plausible.Purge.html#delete_imported_stats!/2"},{"type":"function","doc":"Move stats pointers so that no historical stats are available.","title":"Plausible.Purge.delete_native_stats!/1","ref":"Plausible.Purge.html#delete_native_stats!/1"},{"type":"function","doc":"","title":"Plausible.Purge.reset!/1","ref":"Plausible.Purge.html#reset!/1"},{"type":"module","doc":"Thin wrapper around `:ets.update_counter/4` and a\nclean-up process to act as a rate limiter.","title":"Plausible.RateLimit","ref":"Plausible.RateLimit.html"},{"type":"function","doc":"Checks the rate-limit for a key.","title":"Plausible.RateLimit.check_rate/5","ref":"Plausible.RateLimit.html#check_rate/5"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.RateLimit.child_spec/1","ref":"Plausible.RateLimit.html#child_spec/1"},{"type":"function","doc":"Starts the process that creates and cleans the ETS table.\n\nAccepts the following options:\n - `GenServer.options()`\n - `:table` for the ETS table name, defaults to `Elixir.Plausible.RateLimit`\n - `:clean_period` for how often to perform garbage collection","title":"Plausible.RateLimit.start_link/1","ref":"Plausible.RateLimit.html#start_link/1"},{"type":"module","doc":"","title":"Plausible.Release","ref":"Plausible.Release.html"},{"type":"function","doc":"","title":"Plausible.Release.configure_ref_inspector/0","ref":"Plausible.Release.html#configure_ref_inspector/0"},{"type":"function","doc":"","title":"Plausible.Release.configure_ua_inspector/0","ref":"Plausible.Release.html#configure_ua_inspector/0"},{"type":"function","doc":"","title":"Plausible.Release.createdb/0","ref":"Plausible.Release.html#createdb/0"},{"type":"function","doc":"","title":"Plausible.Release.dump_plans/0","ref":"Plausible.Release.html#dump_plans/0"},{"type":"function","doc":"","title":"Plausible.Release.migrate/0","ref":"Plausible.Release.html#migrate/0"},{"type":"function","doc":"","title":"Plausible.Release.pending_migrations/0","ref":"Plausible.Release.html#pending_migrations/0"},{"type":"function","doc":"","title":"Plausible.Release.rollback/0","ref":"Plausible.Release.html#rollback/0"},{"type":"function","doc":"","title":"Plausible.Release.seed/0","ref":"Plausible.Release.html#seed/0"},{"type":"function","doc":"","title":"Plausible.Release.should_be_first_launch?/0","ref":"Plausible.Release.html#should_be_first_launch?/0"},{"type":"module","doc":"","title":"Plausible.Repo","ref":"Plausible.Repo.html"},{"type":"function","doc":"","title":"Plausible.Repo.aggregate/3","ref":"Plausible.Repo.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.Repo.aggregate/4","ref":"Plausible.Repo.html#aggregate/4"},{"type":"function","doc":"","title":"Plausible.Repo.all/2","ref":"Plausible.Repo.html#all/2"},{"type":"function","doc":"","title":"Plausible.Repo.checked_out?/0","ref":"Plausible.Repo.html#checked_out?/0"},{"type":"function","doc":"","title":"Plausible.Repo.checkout/2","ref":"Plausible.Repo.html#checkout/2"},{"type":"function","doc":"","title":"Plausible.Repo.child_spec/1","ref":"Plausible.Repo.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Repo.config/0","ref":"Plausible.Repo.html#config/0"},{"type":"function","doc":"","title":"Plausible.Repo.default_options/1","ref":"Plausible.Repo.html#default_options/1"},{"type":"function","doc":"","title":"Plausible.Repo.delete/2","ref":"Plausible.Repo.html#delete/2"},{"type":"function","doc":"","title":"Plausible.Repo.delete!/2","ref":"Plausible.Repo.html#delete!/2"},{"type":"function","doc":"","title":"Plausible.Repo.delete_all/2","ref":"Plausible.Repo.html#delete_all/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","title":"Plausible.Repo.disconnect_all/2","ref":"Plausible.Repo.html#disconnect_all/2"},{"type":"function","doc":"","title":"Plausible.Repo.exists?/2","ref":"Plausible.Repo.html#exists?/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes an EXPLAIN statement or similar\ndepending on the adapter to obtain statistics for the given query.\n\nSee `Ecto.Adapters.SQL.explain/4` for more information.","title":"Plausible.Repo.explain/3","ref":"Plausible.Repo.html#explain/3"},{"type":"function","doc":"","title":"Plausible.Repo.get/3","ref":"Plausible.Repo.html#get/3"},{"type":"function","doc":"","title":"Plausible.Repo.get!/3","ref":"Plausible.Repo.html#get!/3"},{"type":"function","doc":"","title":"Plausible.Repo.get_by/3","ref":"Plausible.Repo.html#get_by/3"},{"type":"function","doc":"","title":"Plausible.Repo.get_by!/3","ref":"Plausible.Repo.html#get_by!/3"},{"type":"function","doc":"","title":"Plausible.Repo.get_dynamic_repo/0","ref":"Plausible.Repo.html#get_dynamic_repo/0"},{"type":"function","doc":"","title":"Plausible.Repo.in_transaction?/0","ref":"Plausible.Repo.html#in_transaction?/0"},{"type":"function","doc":"","title":"Plausible.Repo.insert/2","ref":"Plausible.Repo.html#insert/2"},{"type":"function","doc":"","title":"Plausible.Repo.insert!/2","ref":"Plausible.Repo.html#insert!/2"},{"type":"function","doc":"","title":"Plausible.Repo.insert_all/3","ref":"Plausible.Repo.html#insert_all/3"},{"type":"function","doc":"","title":"Plausible.Repo.insert_or_update/2","ref":"Plausible.Repo.html#insert_or_update/2"},{"type":"function","doc":"","title":"Plausible.Repo.insert_or_update!/2","ref":"Plausible.Repo.html#insert_or_update!/2"},{"type":"function","doc":"","title":"Plausible.Repo.load/2","ref":"Plausible.Repo.html#load/2"},{"type":"function","doc":"","title":"Plausible.Repo.one/2","ref":"Plausible.Repo.html#one/2"},{"type":"function","doc":"","title":"Plausible.Repo.one!/2","ref":"Plausible.Repo.html#one!/2"},{"type":"function","doc":"","title":"Plausible.Repo.paginate/2","ref":"Plausible.Repo.html#paginate/2"},{"type":"function","doc":"","title":"Plausible.Repo.preload/3","ref":"Plausible.Repo.html#preload/3"},{"type":"function","doc":"","title":"Plausible.Repo.prepare_query/3","ref":"Plausible.Repo.html#prepare_query/3"},{"type":"function","doc":"","title":"Plausible.Repo.put_dynamic_repo/1","ref":"Plausible.Repo.html#put_dynamic_repo/1"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","title":"Plausible.Repo.query/3","ref":"Plausible.Repo.html#query/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","title":"Plausible.Repo.query!/3","ref":"Plausible.Repo.html#query!/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given multi-result query.\n\nSee `Ecto.Adapters.SQL.query_many/4` for more information.","title":"Plausible.Repo.query_many/3","ref":"Plausible.Repo.html#query_many/3"},{"type":"function","doc":"A convenience function for SQL-based repositories that executes the given multi-result query.\n\nSee `Ecto.Adapters.SQL.query_many!/4` for more information.","title":"Plausible.Repo.query_many!/3","ref":"Plausible.Repo.html#query_many!/3"},{"type":"function","doc":"","title":"Plausible.Repo.reload/2","ref":"Plausible.Repo.html#reload/2"},{"type":"function","doc":"","title":"Plausible.Repo.reload!/2","ref":"Plausible.Repo.html#reload!/2"},{"type":"function","doc":"","title":"Plausible.Repo.rollback/1","ref":"Plausible.Repo.html#rollback/1"},{"type":"function","doc":"","title":"Plausible.Repo.scrivener_defaults/0","ref":"Plausible.Repo.html#scrivener_defaults/0"},{"type":"function","doc":"","title":"Plausible.Repo.start_link/1","ref":"Plausible.Repo.html#start_link/1"},{"type":"function","doc":"","title":"Plausible.Repo.stop/1","ref":"Plausible.Repo.html#stop/1"},{"type":"function","doc":"","title":"Plausible.Repo.stream/2","ref":"Plausible.Repo.html#stream/2"},{"type":"function","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","title":"Plausible.Repo.to_sql/2","ref":"Plausible.Repo.html#to_sql/2"},{"type":"function","doc":"","title":"Plausible.Repo.transaction/2","ref":"Plausible.Repo.html#transaction/2"},{"type":"function","doc":"","title":"Plausible.Repo.update/2","ref":"Plausible.Repo.html#update/2"},{"type":"function","doc":"","title":"Plausible.Repo.update!/2","ref":"Plausible.Repo.html#update!/2"},{"type":"function","doc":"","title":"Plausible.Repo.update_all/3","ref":"Plausible.Repo.html#update_all/3"},{"type":"module","doc":"Custom request logger which:\n- Includes query parameters on the same line\n- Includes request duration on the same line","title":"Plausible.RequestLogger","ref":"Plausible.RequestLogger.html"},{"type":"function","doc":"","title":"Plausible.RequestLogger.log_request/4","ref":"Plausible.RequestLogger.html#log_request/4"},{"type":"module","doc":"Helper functions for S3 exports/imports.","title":"Plausible.S3","ref":"Plausible.S3.html"},{"type":"function","doc":"Returns a presigned URL to download the exported Zip archive from S3.\nThe URL expires in 300 seconds, which should be enough for a redirect.\n\nIn the current implementation the bucket always goes into the path component.","title":"Plausible.S3.download_url/2","ref":"Plausible.S3.html#download_url/2"},{"type":"function","doc":"Chunks and uploads Zip archive to the provided S3 destination.\n\nIn the current implementation the bucket always goes into the path component.","title":"Plausible.S3.export_upload_multipart/4","ref":"Plausible.S3.html#export_upload_multipart/4"},{"type":"function","doc":"Returns the pre-configured S3 bucket for CSV exports.\n\n config :plausible, Plausible.S3,\n exports_bucket: System.fetch_env!(\"S3_EXPORTS_BUCKET\")\n\nExample:\n\n iex> exports_bucket()\n \"test-exports\"","title":"Plausible.S3.exports_bucket/0","ref":"Plausible.S3.html#exports_bucket/0"},{"type":"function","doc":"Returns `access_key_id` and `secret_access_key` to be used by ClickHouse during imports from S3.\n\nExample:\n\n iex> import_clickhouse_credentials()\n %{access_key_id: \"minioadmin\", secret_access_key: \"minioadmin\"}","title":"Plausible.S3.import_clickhouse_credentials/0","ref":"Plausible.S3.html#import_clickhouse_credentials/0"},{"type":"function","doc":"Presigns an upload for an imported file.\n\nIn the current implementation the bucket always goes into the path component.\n\nExample:\n\n iex> upload = import_presign_upload(_site_id = 123, _filename = \"imported_browsers.csv\")\n iex> true = String.ends_with?(upload.s3_url, \"/test-imports/123/imported_browsers.csv\")\n iex> true = String.contains?(upload.presigned_url, \"/test-imports/123/imported_browsers.csv?X-Amz-Algorithm=AWS4-HMAC-SHA256&\")","title":"Plausible.S3.import_presign_upload/2","ref":"Plausible.S3.html#import_presign_upload/2"},{"type":"function","doc":"Returns the pre-configured S3 bucket for CSV imports.\n\n config :plausible, Plausible.S3,\n imports_bucket: System.fetch_env!(\"S3_IMPORTS_BUCKET\")\n\nExample:\n\n iex> imports_bucket()\n \"test-imports\"","title":"Plausible.S3.imports_bucket/0","ref":"Plausible.S3.html#imports_bucket/0"},{"type":"module","doc":"","title":"Plausible.Sentry.Client","ref":"Plausible.Sentry.Client.html"},{"type":"function","doc":"","title":"Plausible.Sentry.Client.post/3","ref":"Plausible.Sentry.Client.html#post/3"},{"type":"module","doc":"Sentry callbacks for filtering and grouping events","title":"Plausible.SentryFilter","ref":"Plausible.SentryFilter.html"},{"type":"function","doc":"","title":"Plausible.SentryFilter.before_send/1","ref":"Plausible.SentryFilter.html#before_send/1"},{"type":"module","doc":"","title":"Plausible.Session.CacheStore","ref":"Plausible.Session.CacheStore.html"},{"type":"function","doc":"","title":"Plausible.Session.CacheStore.lock_telemetry_event/0","ref":"Plausible.Session.CacheStore.html#lock_telemetry_event/0"},{"type":"function","doc":"","title":"Plausible.Session.CacheStore.on_event/4","ref":"Plausible.Session.CacheStore.html#on_event/4"},{"type":"module","doc":"","title":"Plausible.Session.Salts","ref":"Plausible.Session.Salts.html"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"Plausible.Session.Salts.child_spec/1","ref":"Plausible.Session.Salts.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Session.Salts.fetch/0","ref":"Plausible.Session.Salts.html#fetch/0"},{"type":"function","doc":"","title":"Plausible.Session.Salts.rotate/0","ref":"Plausible.Session.Salts.html#rotate/0"},{"type":"function","doc":"","title":"Plausible.Session.Salts.start_link/1","ref":"Plausible.Session.Salts.html#start_link/1"},{"type":"module","doc":"Schema for Country Block List","title":"Plausible.Shield.CountryRule","ref":"Plausible.Shield.CountryRule.html"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRule.changeset/2","ref":"Plausible.Shield.CountryRule.html#changeset/2"},{"type":"type","doc":"","title":"Plausible.Shield.CountryRule.t/0","ref":"Plausible.Shield.CountryRule.html#t:t/0"},{"type":"module","doc":"Allows retrieving Country Rules by domain and country code","title":"Plausible.Shield.CountryRuleCache","ref":"Plausible.Shield.CountryRuleCache.html"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.child_spec/1","ref":"Plausible.Shield.CountryRuleCache.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.get/2","ref":"Plausible.Shield.CountryRuleCache.html#get/2"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.get_or_store/3","ref":"Plausible.Shield.CountryRuleCache.html#get_or_store/3"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.hit_rate/1","ref":"Plausible.Shield.CountryRuleCache.html#hit_rate/1"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.merge_items/2","ref":"Plausible.Shield.CountryRuleCache.html#merge_items/2"},{"type":"function","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","title":"Plausible.Shield.CountryRuleCache.ready?/1","ref":"Plausible.Shield.CountryRuleCache.html#ready?/1"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.refresh_all/1","ref":"Plausible.Shield.CountryRuleCache.html#refresh_all/1"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.refresh_updated_recently/1","ref":"Plausible.Shield.CountryRuleCache.html#refresh_updated_recently/1"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.size/1","ref":"Plausible.Shield.CountryRuleCache.html#size/1"},{"type":"function","doc":"","title":"Plausible.Shield.CountryRuleCache.telemetry_event_refresh/2","ref":"Plausible.Shield.CountryRuleCache.html#telemetry_event_refresh/2"},{"type":"module","doc":"Schema for Hostnames allow list","title":"Plausible.Shield.HostnameRule","ref":"Plausible.Shield.HostnameRule.html"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRule.changeset/2","ref":"Plausible.Shield.HostnameRule.html#changeset/2"},{"type":"type","doc":"","title":"Plausible.Shield.HostnameRule.t/0","ref":"Plausible.Shield.HostnameRule.html#t:t/0"},{"type":"module","doc":"Allows retrieving Hostname Rules by domain","title":"Plausible.Shield.HostnameRuleCache","ref":"Plausible.Shield.HostnameRuleCache.html"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.child_spec/1","ref":"Plausible.Shield.HostnameRuleCache.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.get/2","ref":"Plausible.Shield.HostnameRuleCache.html#get/2"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.get_or_store/3","ref":"Plausible.Shield.HostnameRuleCache.html#get_or_store/3"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.hit_rate/1","ref":"Plausible.Shield.HostnameRuleCache.html#hit_rate/1"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.merge_items/2","ref":"Plausible.Shield.HostnameRuleCache.html#merge_items/2"},{"type":"function","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","title":"Plausible.Shield.HostnameRuleCache.ready?/1","ref":"Plausible.Shield.HostnameRuleCache.html#ready?/1"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.refresh_all/1","ref":"Plausible.Shield.HostnameRuleCache.html#refresh_all/1"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.refresh_updated_recently/1","ref":"Plausible.Shield.HostnameRuleCache.html#refresh_updated_recently/1"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.size/1","ref":"Plausible.Shield.HostnameRuleCache.html#size/1"},{"type":"function","doc":"","title":"Plausible.Shield.HostnameRuleCache.telemetry_event_refresh/2","ref":"Plausible.Shield.HostnameRuleCache.html#telemetry_event_refresh/2"},{"type":"module","doc":"Schema for IP block list","title":"Plausible.Shield.IPRule","ref":"Plausible.Shield.IPRule.html"},{"type":"function","doc":"","title":"Plausible.Shield.IPRule.changeset/2","ref":"Plausible.Shield.IPRule.html#changeset/2"},{"type":"type","doc":"","title":"Plausible.Shield.IPRule.t/0","ref":"Plausible.Shield.IPRule.html#t:t/0"},{"type":"module","doc":"Allows retrieving IP Rules by domain and IP","title":"Plausible.Shield.IPRuleCache","ref":"Plausible.Shield.IPRuleCache.html"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.child_spec/1","ref":"Plausible.Shield.IPRuleCache.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.get/2","ref":"Plausible.Shield.IPRuleCache.html#get/2"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.get_or_store/3","ref":"Plausible.Shield.IPRuleCache.html#get_or_store/3"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.hit_rate/1","ref":"Plausible.Shield.IPRuleCache.html#hit_rate/1"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.merge_items/2","ref":"Plausible.Shield.IPRuleCache.html#merge_items/2"},{"type":"function","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","title":"Plausible.Shield.IPRuleCache.ready?/1","ref":"Plausible.Shield.IPRuleCache.html#ready?/1"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.refresh_all/1","ref":"Plausible.Shield.IPRuleCache.html#refresh_all/1"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.refresh_updated_recently/1","ref":"Plausible.Shield.IPRuleCache.html#refresh_updated_recently/1"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.size/1","ref":"Plausible.Shield.IPRuleCache.html#size/1"},{"type":"function","doc":"","title":"Plausible.Shield.IPRuleCache.telemetry_event_refresh/2","ref":"Plausible.Shield.IPRuleCache.html#telemetry_event_refresh/2"},{"type":"module","doc":"Schema for Pages block list","title":"Plausible.Shield.PageRule","ref":"Plausible.Shield.PageRule.html"},{"type":"function","doc":"","title":"Plausible.Shield.PageRule.changeset/2","ref":"Plausible.Shield.PageRule.html#changeset/2"},{"type":"type","doc":"","title":"Plausible.Shield.PageRule.t/0","ref":"Plausible.Shield.PageRule.html#t:t/0"},{"type":"module","doc":"Allows retrieving Page Rules by domain","title":"Plausible.Shield.PageRuleCache","ref":"Plausible.Shield.PageRuleCache.html"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.child_spec/1","ref":"Plausible.Shield.PageRuleCache.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.get/2","ref":"Plausible.Shield.PageRuleCache.html#get/2"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.get_or_store/3","ref":"Plausible.Shield.PageRuleCache.html#get_or_store/3"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.hit_rate/1","ref":"Plausible.Shield.PageRuleCache.html#hit_rate/1"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.merge_items/2","ref":"Plausible.Shield.PageRuleCache.html#merge_items/2"},{"type":"function","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","title":"Plausible.Shield.PageRuleCache.ready?/1","ref":"Plausible.Shield.PageRuleCache.html#ready?/1"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.refresh_all/1","ref":"Plausible.Shield.PageRuleCache.html#refresh_all/1"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.refresh_updated_recently/1","ref":"Plausible.Shield.PageRuleCache.html#refresh_updated_recently/1"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.size/1","ref":"Plausible.Shield.PageRuleCache.html#size/1"},{"type":"function","doc":"","title":"Plausible.Shield.PageRuleCache.telemetry_event_refresh/2","ref":"Plausible.Shield.PageRuleCache.html#telemetry_event_refresh/2"},{"type":"module","doc":"Contextual interface for shields.","title":"Plausible.Shields","ref":"Plausible.Shields.html"},{"type":"function","doc":"","title":"Plausible.Shields.add_country_rule/3","ref":"Plausible.Shields.html#add_country_rule/3"},{"type":"function","doc":"","title":"Plausible.Shields.add_hostname_rule/3","ref":"Plausible.Shields.html#add_hostname_rule/3"},{"type":"function","doc":"","title":"Plausible.Shields.add_ip_rule/3","ref":"Plausible.Shields.html#add_ip_rule/3"},{"type":"function","doc":"","title":"Plausible.Shields.add_page_rule/3","ref":"Plausible.Shields.html#add_page_rule/3"},{"type":"function","doc":"","title":"Plausible.Shields.allowed_hostname_patterns/1","ref":"Plausible.Shields.html#allowed_hostname_patterns/1"},{"type":"function","doc":"","title":"Plausible.Shields.count_country_rules/1","ref":"Plausible.Shields.html#count_country_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.count_hostname_rules/1","ref":"Plausible.Shields.html#count_hostname_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.count_ip_rules/1","ref":"Plausible.Shields.html#count_ip_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.count_page_rules/1","ref":"Plausible.Shields.html#count_page_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.country_blocked?/2","ref":"Plausible.Shields.html#country_blocked?/2"},{"type":"function","doc":"","title":"Plausible.Shields.hostname_allowed?/2","ref":"Plausible.Shields.html#hostname_allowed?/2"},{"type":"function","doc":"","title":"Plausible.Shields.ip_blocked?/2","ref":"Plausible.Shields.html#ip_blocked?/2"},{"type":"function","doc":"","title":"Plausible.Shields.list_country_rules/1","ref":"Plausible.Shields.html#list_country_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.list_hostname_rules/1","ref":"Plausible.Shields.html#list_hostname_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.list_ip_rules/1","ref":"Plausible.Shields.html#list_ip_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.list_page_rules/1","ref":"Plausible.Shields.html#list_page_rules/1"},{"type":"function","doc":"","title":"Plausible.Shields.maximum_country_rules/0","ref":"Plausible.Shields.html#maximum_country_rules/0"},{"type":"function","doc":"","title":"Plausible.Shields.maximum_hostname_rules/0","ref":"Plausible.Shields.html#maximum_hostname_rules/0"},{"type":"function","doc":"","title":"Plausible.Shields.maximum_ip_rules/0","ref":"Plausible.Shields.html#maximum_ip_rules/0"},{"type":"function","doc":"","title":"Plausible.Shields.maximum_page_rules/0","ref":"Plausible.Shields.html#maximum_page_rules/0"},{"type":"function","doc":"","title":"Plausible.Shields.page_blocked?/2","ref":"Plausible.Shields.html#page_blocked?/2"},{"type":"function","doc":"","title":"Plausible.Shields.remove_country_rule/2","ref":"Plausible.Shields.html#remove_country_rule/2"},{"type":"function","doc":"","title":"Plausible.Shields.remove_hostname_rule/2","ref":"Plausible.Shields.html#remove_hostname_rule/2"},{"type":"function","doc":"","title":"Plausible.Shields.remove_ip_rule/2","ref":"Plausible.Shields.html#remove_ip_rule/2"},{"type":"function","doc":"","title":"Plausible.Shields.remove_page_rule/2","ref":"Plausible.Shields.html#remove_page_rule/2"},{"type":"module","doc":"Site schema","title":"Plausible.Site","ref":"Plausible.Site.html"},{"type":"function","doc":"","title":"Plausible.Site.changeset/2","ref":"Plausible.Site.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Site.crm_changeset/2","ref":"Plausible.Site.html#crm_changeset/2"},{"type":"function","doc":"","title":"Plausible.Site.make_private/1","ref":"Plausible.Site.html#make_private/1"},{"type":"function","doc":"","title":"Plausible.Site.make_public/1","ref":"Plausible.Site.html#make_public/1"},{"type":"function","doc":"","title":"Plausible.Site.new/1","ref":"Plausible.Site.html#new/1"},{"type":"function","doc":"","title":"Plausible.Site.set_native_stats_start_at/2","ref":"Plausible.Site.html#set_native_stats_start_at/2"},{"type":"function","doc":"","title":"Plausible.Site.set_stats_start_date/2","ref":"Plausible.Site.html#set_stats_start_date/2"},{"type":"function","doc":"","title":"Plausible.Site.tz_offset/2","ref":"Plausible.Site.html#tz_offset/2"},{"type":"function","doc":"","title":"Plausible.Site.update_changeset/3","ref":"Plausible.Site.html#update_changeset/3"},{"type":"type","doc":"","title":"Plausible.Site.t/0","ref":"Plausible.Site.html#t:t/0"},{"type":"module","doc":"The cache allows lookups by both `domain` and `domain_changed_from`\nfields - this is to allow traffic from sites whose domains changed within a certain\ngrace period (see: `Plausible.Site.Transfer`).\n\nTo differentiate cached Site structs from those retrieved directly from the\ndatabase, a virtual schema field `from_cache?: true` is set.\nThis indicates the `Plausible.Site` struct is incomplete in comparison to its\ndatabase counterpart -- to spare bandwidth and query execution time,\nonly selected database columns are retrieved and cached.\n\nThe `@cached_schema_fields` attribute defines the list of DB columns\nqueried on each cache refresh.\n\nAlso see tests for more comprehensive examples.","title":"Plausible.Site.Cache","ref":"Plausible.Site.Cache.html"},{"type":"function","doc":"","title":"Plausible.Site.Cache.child_spec/1","ref":"Plausible.Site.Cache.html#child_spec/1"},{"type":"function","doc":"","title":"Plausible.Site.Cache.get/2","ref":"Plausible.Site.Cache.html#get/2"},{"type":"function","doc":"","title":"Plausible.Site.Cache.get_or_store/3","ref":"Plausible.Site.Cache.html#get_or_store/3"},{"type":"function","doc":"","title":"Plausible.Site.Cache.get_site_id/2","ref":"Plausible.Site.Cache.html#get_site_id/2"},{"type":"function","doc":"","title":"Plausible.Site.Cache.hit_rate/1","ref":"Plausible.Site.Cache.html#hit_rate/1"},{"type":"function","doc":"","title":"Plausible.Site.Cache.merge_items/2","ref":"Plausible.Site.Cache.html#merge_items/2"},{"type":"function","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","title":"Plausible.Site.Cache.ready?/1","ref":"Plausible.Site.Cache.html#ready?/1"},{"type":"function","doc":"","title":"Plausible.Site.Cache.refresh_all/1","ref":"Plausible.Site.Cache.html#refresh_all/1"},{"type":"function","doc":"","title":"Plausible.Site.Cache.refresh_updated_recently/1","ref":"Plausible.Site.Cache.html#refresh_updated_recently/1"},{"type":"function","doc":"","title":"Plausible.Site.Cache.size/1","ref":"Plausible.Site.Cache.html#size/1"},{"type":"function","doc":"","title":"Plausible.Site.Cache.telemetry_event_refresh/2","ref":"Plausible.Site.Cache.html#telemetry_event_refresh/2"},{"type":"function","doc":"","title":"Plausible.Site.Cache.touch_site!/2","ref":"Plausible.Site.Cache.html#touch_site!/2"},{"type":"module","doc":"Basic interface for domain changes.\n\nOnce `Plausible.DataMigration.NumericIDs` schema migration is ready, \ndomain change operation will be enabled, accessible to the users.\n\nWe will set a transition period of 72 hours\nduring which, both old and new domains, will be accepted as traffic\nidentifiers to the same site. \n\nA periodic worker will call the `expire/0` function to end it where applicable.\nSee: `Plausible.Workers.ExpireDomainChangeTransitions`.\n\nThe underlying changeset for domain change (see: `Plausible.Site`) relies\non database trigger installed via `Plausible.Repo.Migrations.AllowDomainChange`\nPostgres migration. The trigger checks if either `domain` or `domain_changed_from`\nexist to ensure unicity.","title":"Plausible.Site.Domain","ref":"Plausible.Site.Domain.html"},{"type":"function","doc":"","title":"Plausible.Site.Domain.change/3","ref":"Plausible.Site.Domain.html#change/3"},{"type":"function","doc":"","title":"Plausible.Site.Domain.expire_change_transitions/1","ref":"Plausible.Site.Domain.html#expire_change_transitions/1"},{"type":"module","doc":"Thin wrapper around `Plausible.RateLimit` for gate keeping domain-specific events\nduring the ingestion phase. When the site is allowed, gate keeping\ncheck returns `:allow`, otherwise a `:deny` tagged tuple is returned\nwith one of the following policy markers:\n * `:not_found` (indicates site not found in cache)\n * `:block` (indicates disabled sites)\n * `:throttle` (indicates rate limiting)\n\nRate Limiting buckets are configured per site (externally via the CRM).\nSee: `Plausible.Site`\n\nTo look up each site's configuration, the RateLimiter fetches\na Site by domain using `Plausible.Cache` interface.\n\nThe module defines two policies outside the regular bucket inspection:\n * when the site is not found in cache: not_found\n * when the underlying rate limiting mechanism returns\n an internal error: :allow","title":"Plausible.Site.GateKeeper","ref":"Plausible.Site.GateKeeper.html"},{"type":"function","doc":"","title":"Plausible.Site.GateKeeper.check/2","ref":"Plausible.Site.GateKeeper.html#check/2"},{"type":"function","doc":"","title":"Plausible.Site.GateKeeper.key/1","ref":"Plausible.Site.GateKeeper.html#key/1"},{"type":"type","doc":"","title":"Plausible.Site.GateKeeper.policy/0","ref":"Plausible.Site.GateKeeper.html#t:policy/0"},{"type":"type","doc":"","title":"Plausible.Site.GateKeeper.t/0","ref":"Plausible.Site.GateKeeper.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Site.GoogleAuth","ref":"Plausible.Site.GoogleAuth.html"},{"type":"function","doc":"","title":"Plausible.Site.GoogleAuth.changeset/2","ref":"Plausible.Site.GoogleAuth.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Site.GoogleAuth.set_property/2","ref":"Plausible.Site.GoogleAuth.html#set_property/2"},{"type":"module","doc":"Embedded schema for analytics imports\n\nNOTE: needed by `SiteImports` data migration script","title":"Plausible.Site.ImportedData","ref":"Plausible.Site.ImportedData.html"},{"type":"type","doc":"","title":"Plausible.Site.ImportedData.t/0","ref":"Plausible.Site.ImportedData.html#t:t/0"},{"type":"module","doc":"Embedded schema for installation meta-data","title":"Plausible.Site.InstallationMeta","ref":"Plausible.Site.InstallationMeta.html"},{"type":"type","doc":"","title":"Plausible.Site.InstallationMeta.t/0","ref":"Plausible.Site.InstallationMeta.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Site.Membership","ref":"Plausible.Site.Membership.html"},{"type":"function","doc":"","title":"Plausible.Site.Membership.new/2","ref":"Plausible.Site.Membership.html#new/2"},{"type":"function","doc":"","title":"Plausible.Site.Membership.set_role/2","ref":"Plausible.Site.Membership.html#set_role/2"},{"type":"type","doc":"","title":"Plausible.Site.Membership.role/0","ref":"Plausible.Site.Membership.html#t:role/0"},{"type":"type","doc":"","title":"Plausible.Site.Membership.t/0","ref":"Plausible.Site.Membership.html#t:t/0"},{"type":"module","doc":"API for site memberships and invitations","title":"Plausible.Site.Memberships","ref":"Plausible.Site.Memberships.html"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.accept_invitation/2","ref":"Plausible.Site.Memberships.html#accept_invitation/2"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.all_pending_ownerships/1","ref":"Plausible.Site.Memberships.html#all_pending_ownerships/1"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.any?/1","ref":"Plausible.Site.Memberships.html#any?/1"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.any_or_pending?/1","ref":"Plausible.Site.Memberships.html#any_or_pending?/1"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.bulk_create_invitation/5","ref":"Plausible.Site.Memberships.html#bulk_create_invitation/5"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.bulk_transfer_ownership_direct/2","ref":"Plausible.Site.Memberships.html#bulk_transfer_ownership_direct/2"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.create_invitation/4","ref":"Plausible.Site.Memberships.html#create_invitation/4"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.pending?/1","ref":"Plausible.Site.Memberships.html#pending?/1"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.pending_ownerships?/1","ref":"Plausible.Site.Memberships.html#pending_ownerships?/1"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.reject_invitation/2","ref":"Plausible.Site.Memberships.html#reject_invitation/2"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.remove_invitation/2","ref":"Plausible.Site.Memberships.html#remove_invitation/2"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.transfer_ownership/2","ref":"Plausible.Site.Memberships.html#transfer_ownership/2"},{"type":"module","doc":"Service for accepting invitations, including ownership transfers.\n\nAccepting invitation accounts for the fact that it's possible\nthat accepting user has an existing membership for the site and\nacts permissively to not unnecessarily disrupt the flow while\nalso maintaining integrity of site memberships. This also applies\nto cases where users update their email address between issuing\nthe invitation and accepting it.","title":"Plausible.Site.Memberships.AcceptInvitation","ref":"Plausible.Site.Memberships.AcceptInvitation.html"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.AcceptInvitation.accept_invitation/2","ref":"Plausible.Site.Memberships.AcceptInvitation.html#accept_invitation/2"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.AcceptInvitation.transfer_ownership/2","ref":"Plausible.Site.Memberships.AcceptInvitation.html#transfer_ownership/2"},{"type":"module","doc":"Service for inviting new or existing users to a sites, including ownershhip\ntransfers.","title":"Plausible.Site.Memberships.CreateInvitation","ref":"Plausible.Site.Memberships.CreateInvitation.html"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.CreateInvitation.bulk_create_invitation/5","ref":"Plausible.Site.Memberships.CreateInvitation.html#bulk_create_invitation/5"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.CreateInvitation.bulk_transfer_ownership_direct/2","ref":"Plausible.Site.Memberships.CreateInvitation.html#bulk_transfer_ownership_direct/2"},{"type":"function","doc":"Invites a new team member to the given site. Returns a\n%Plausible.Auth.Invitation{} struct and sends the invitee an email to accept\nthis invitation.\n\nThe inviter must have enough permissions to invite the new team member,\notherwise this function returns `{:error, :forbidden}`.\n\nIf the new team member role is `:owner`, this function handles the invitation\nas an ownership transfer and requires the inviter to be the owner of the site.","title":"Plausible.Site.Memberships.CreateInvitation.create_invitation/4","ref":"Plausible.Site.Memberships.CreateInvitation.html#create_invitation/4"},{"type":"type","doc":"","title":"Plausible.Site.Memberships.CreateInvitation.invite_error/0","ref":"Plausible.Site.Memberships.CreateInvitation.html#t:invite_error/0"},{"type":"module","doc":"Service for rejecting invitations.","title":"Plausible.Site.Memberships.RejectInvitation","ref":"Plausible.Site.Memberships.RejectInvitation.html"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.RejectInvitation.reject_invitation/2","ref":"Plausible.Site.Memberships.RejectInvitation.html#reject_invitation/2"},{"type":"module","doc":"Service for removing invitations.","title":"Plausible.Site.Memberships.RemoveInvitation","ref":"Plausible.Site.Memberships.RemoveInvitation.html"},{"type":"function","doc":"","title":"Plausible.Site.Memberships.RemoveInvitation.remove_invitation/2","ref":"Plausible.Site.Memberships.RemoveInvitation.html#remove_invitation/2"},{"type":"module","doc":"","title":"Plausible.Site.MonthlyReport","ref":"Plausible.Site.MonthlyReport.html"},{"type":"function","doc":"","title":"Plausible.Site.MonthlyReport.add_recipient/2","ref":"Plausible.Site.MonthlyReport.html#add_recipient/2"},{"type":"function","doc":"","title":"Plausible.Site.MonthlyReport.changeset/2","ref":"Plausible.Site.MonthlyReport.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Site.MonthlyReport.remove_recipient/2","ref":"Plausible.Site.MonthlyReport.html#remove_recipient/2"},{"type":"module","doc":"A site deletion service stub.","title":"Plausible.Site.Removal","ref":"Plausible.Site.Removal.html"},{"type":"function","doc":"","title":"Plausible.Site.Removal.run/1","ref":"Plausible.Site.Removal.html#run/1"},{"type":"module","doc":"","title":"Plausible.Site.SharedLink","ref":"Plausible.Site.SharedLink.html"},{"type":"function","doc":"","title":"Plausible.Site.SharedLink.changeset/2","ref":"Plausible.Site.SharedLink.html#changeset/2"},{"type":"type","doc":"","title":"Plausible.Site.SharedLink.t/0","ref":"Plausible.Site.SharedLink.html#t:t/0"},{"type":"module","doc":"Configuration schema for site-specific traffic change notifications.","title":"Plausible.Site.TrafficChangeNotification","ref":"Plausible.Site.TrafficChangeNotification.html"},{"type":"function","doc":"","title":"Plausible.Site.TrafficChangeNotification.add_recipient/2","ref":"Plausible.Site.TrafficChangeNotification.html#add_recipient/2"},{"type":"function","doc":"","title":"Plausible.Site.TrafficChangeNotification.changeset/2","ref":"Plausible.Site.TrafficChangeNotification.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Site.TrafficChangeNotification.remove_recipient/2","ref":"Plausible.Site.TrafficChangeNotification.html#remove_recipient/2"},{"type":"function","doc":"","title":"Plausible.Site.TrafficChangeNotification.was_sent/1","ref":"Plausible.Site.TrafficChangeNotification.html#was_sent/1"},{"type":"module","doc":"User-specific site preferences schema","title":"Plausible.Site.UserPreference","ref":"Plausible.Site.UserPreference.html"},{"type":"function","doc":"","title":"Plausible.Site.UserPreference.changeset/3","ref":"Plausible.Site.UserPreference.html#changeset/3"},{"type":"macro","doc":"","title":"Plausible.Site.UserPreference.options/0","ref":"Plausible.Site.UserPreference.html#options/0"},{"type":"type","doc":"","title":"Plausible.Site.UserPreference.t/0","ref":"Plausible.Site.UserPreference.html#t:t/0"},{"type":"module","doc":"","title":"Plausible.Site.WeeklyReport","ref":"Plausible.Site.WeeklyReport.html"},{"type":"function","doc":"","title":"Plausible.Site.WeeklyReport.add_recipient/2","ref":"Plausible.Site.WeeklyReport.html#add_recipient/2"},{"type":"function","doc":"","title":"Plausible.Site.WeeklyReport.changeset/2","ref":"Plausible.Site.WeeklyReport.html#changeset/2"},{"type":"function","doc":"","title":"Plausible.Site.WeeklyReport.remove_recipient/2","ref":"Plausible.Site.WeeklyReport.html#remove_recipient/2"},{"type":"module","doc":"","title":"Plausible.SiteAdmin","ref":"Plausible.SiteAdmin.html"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.before_update/2","ref":"Plausible.SiteAdmin.html#before_update/2"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.create_changeset/2","ref":"Plausible.SiteAdmin.html#create_changeset/2"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.custom_index_query/3","ref":"Plausible.SiteAdmin.html#custom_index_query/3"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.form_fields/1","ref":"Plausible.SiteAdmin.html#form_fields/1"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.get_struct_fields/1","ref":"Plausible.SiteAdmin.html#get_struct_fields/1"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.index/1","ref":"Plausible.SiteAdmin.html#index/1"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.list_actions/1","ref":"Plausible.SiteAdmin.html#list_actions/1"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.ordering/1","ref":"Plausible.SiteAdmin.html#ordering/1"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.search_fields/1","ref":"Plausible.SiteAdmin.html#search_fields/1"},{"type":"function","doc":"","title":"Plausible.SiteAdmin.update_changeset/2","ref":"Plausible.SiteAdmin.html#update_changeset/2"},{"type":"module","doc":"Sites context functions.","title":"Plausible.Sites","ref":"Plausible.Sites.html"},{"type":"function","doc":"","title":"Plausible.Sites.clear_stats_start_date!/1","ref":"Plausible.Sites.html#clear_stats_start_date!/1"},{"type":"function","doc":"","title":"Plausible.Sites.create/2","ref":"Plausible.Sites.html#create/2"},{"type":"function","doc":"","title":"Plausible.Sites.create_shared_link/3","ref":"Plausible.Sites.html#create_shared_link/3"},{"type":"function","doc":"","title":"Plausible.Sites.for_user_query/1","ref":"Plausible.Sites.html#for_user_query/1"},{"type":"function","doc":"","title":"Plausible.Sites.get_by_domain/1","ref":"Plausible.Sites.html#get_by_domain/1"},{"type":"function","doc":"","title":"Plausible.Sites.get_by_domain!/1","ref":"Plausible.Sites.html#get_by_domain!/1"},{"type":"function","doc":"","title":"Plausible.Sites.get_for_user/3","ref":"Plausible.Sites.html#get_for_user/3"},{"type":"function","doc":"","title":"Plausible.Sites.get_for_user!/3","ref":"Plausible.Sites.html#get_for_user!/3"},{"type":"function","doc":"","title":"Plausible.Sites.has_admin_access?/2","ref":"Plausible.Sites.html#has_admin_access?/2"},{"type":"function","doc":"","title":"Plausible.Sites.has_goals?/1","ref":"Plausible.Sites.html#has_goals?/1"},{"type":"function","doc":"","title":"Plausible.Sites.has_stats?/1","ref":"Plausible.Sites.html#has_stats?/1"},{"type":"function","doc":"","title":"Plausible.Sites.is_member?/2","ref":"Plausible.Sites.html#is_member?/2"},{"type":"function","doc":"","title":"Plausible.Sites.list/3","ref":"Plausible.Sites.html#list/3"},{"type":"function","doc":"","title":"Plausible.Sites.list_with_invitations/3","ref":"Plausible.Sites.html#list_with_invitations/3"},{"type":"function","doc":"","title":"Plausible.Sites.locked?/1","ref":"Plausible.Sites.html#locked?/1"},{"type":"function","doc":"","title":"Plausible.Sites.native_stats_start_date/1","ref":"Plausible.Sites.html#native_stats_start_date/1"},{"type":"function","doc":"","title":"Plausible.Sites.owned_site_ids/1","ref":"Plausible.Sites.html#owned_site_ids/1"},{"type":"function","doc":"","title":"Plausible.Sites.owned_sites_count/1","ref":"Plausible.Sites.html#owned_sites_count/1"},{"type":"function","doc":"","title":"Plausible.Sites.owned_sites_domains/1","ref":"Plausible.Sites.html#owned_sites_domains/1"},{"type":"function","doc":"","title":"Plausible.Sites.owned_sites_locked?/1","ref":"Plausible.Sites.html#owned_sites_locked?/1"},{"type":"function","doc":"","title":"Plausible.Sites.role/2","ref":"Plausible.Sites.html#role/2"},{"type":"function","doc":"","title":"Plausible.Sites.set_option/4","ref":"Plausible.Sites.html#set_option/4"},{"type":"function","doc":"","title":"Plausible.Sites.shared_link_url/2","ref":"Plausible.Sites.html#shared_link_url/2"},{"type":"function","doc":"Returns the date of the first event of the given site, or `nil` if the site\ndoes not have stats yet.\n\nIf this is the first time the function is called for the site, it queries\nimported stats and Clickhouse, choosing the earliest start date and saves\nit in the sites table.","title":"Plausible.Sites.stats_start_date/1","ref":"Plausible.Sites.html#stats_start_date/1"},{"type":"function","doc":"","title":"Plausible.Sites.toggle_pin/2","ref":"Plausible.Sites.html#toggle_pin/2"},{"type":"function","doc":"","title":"Plausible.Sites.update_installation_meta!/2","ref":"Plausible.Sites.html#update_installation_meta!/2"},{"type":"type","doc":"","title":"Plausible.Sites.list_opt/0","ref":"Plausible.Sites.html#t:list_opt/0"},{"type":"module","doc":"","title":"Plausible.Stats","ref":"Plausible.Stats.html"},{"type":"function","doc":"","title":"Plausible.Stats.aggregate/3","ref":"Plausible.Stats.html#aggregate/3"},{"type":"function","doc":"","title":"Plausible.Stats.breakdown/4","ref":"Plausible.Stats.html#breakdown/4"},{"type":"function","doc":"","title":"Plausible.Stats.current_visitors/2","ref":"Plausible.Stats.html#current_visitors/2"},{"type":"function","doc":"","title":"Plausible.Stats.filter_suggestions/4","ref":"Plausible.Stats.html#filter_suggestions/4"},{"type":"function","doc":"","title":"Plausible.Stats.funnel/3","ref":"Plausible.Stats.html#funnel/3"},{"type":"function","doc":"","title":"Plausible.Stats.query/2","ref":"Plausible.Stats.html#query/2"},{"type":"function","doc":"","title":"Plausible.Stats.timeseries/3","ref":"Plausible.Stats.html#timeseries/3"},{"type":"module","doc":"Builds aggregate results for v1 of our stats API and dashboards.\n\nAvoid adding new logic here - update QueryBuilder etc instead.","title":"Plausible.Stats.Aggregate","ref":"Plausible.Stats.Aggregate.html"},{"type":"function","doc":"","title":"Plausible.Stats.Aggregate.aggregate/3","ref":"Plausible.Stats.Aggregate.html#aggregate/3"},{"type":"module","doc":"","title":"Plausible.Stats.Base","ref":"Plausible.Stats.Base.html"},{"type":"function","doc":"","title":"Plausible.Stats.Base.base_event_query/2","ref":"Plausible.Stats.Base.html#base_event_query/2"},{"type":"function","doc":"","title":"Plausible.Stats.Base.query_sessions/2","ref":"Plausible.Stats.Base.html#query_sessions/2"},{"type":"module","doc":"Builds breakdown results for v1 of our stats API and dashboards.\n\nAvoid adding new logic here - update QueryBuilder etc instead.","title":"Plausible.Stats.Breakdown","ref":"Plausible.Stats.Breakdown.html"},{"type":"function","doc":"","title":"Plausible.Stats.Breakdown.breakdown/5","ref":"Plausible.Stats.Breakdown.html#breakdown/5"},{"type":"function","doc":"","title":"Plausible.Stats.Breakdown.transform_dimensions/1","ref":"Plausible.Stats.Breakdown.html#transform_dimensions/1"},{"type":"module","doc":"","title":"Plausible.Stats.Clickhouse","ref":"Plausible.Stats.Clickhouse.html"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.current_visitors/1","ref":"Plausible.Stats.Clickhouse.html#current_visitors/1"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.current_visitors_12h/1","ref":"Plausible.Stats.Clickhouse.html#current_visitors_12h/1"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.empty_24h_visitors_hourly_intervals/2","ref":"Plausible.Stats.Clickhouse.html#empty_24h_visitors_hourly_intervals/2"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.has_pageviews?/1","ref":"Plausible.Stats.Clickhouse.html#has_pageviews?/1"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.imported_pageview_count/1","ref":"Plausible.Stats.Clickhouse.html#imported_pageview_count/1"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.imported_pageview_counts/1","ref":"Plausible.Stats.Clickhouse.html#imported_pageview_counts/1"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.last_24h_visitors_hourly_intervals/2","ref":"Plausible.Stats.Clickhouse.html#last_24h_visitors_hourly_intervals/2"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.pageview_start_date_local/1","ref":"Plausible.Stats.Clickhouse.html#pageview_start_date_local/1"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.top_sources_for_spike/4","ref":"Plausible.Stats.Clickhouse.html#top_sources_for_spike/4"},{"type":"function","doc":"","title":"Plausible.Stats.Clickhouse.usage_breakdown/2","ref":"Plausible.Stats.Clickhouse.html#usage_breakdown/2"},{"type":"module","doc":"","title":"Plausible.Stats.Compare","ref":"Plausible.Stats.Compare.html"},{"type":"function","doc":"","title":"Plausible.Stats.Compare.calculate_change/3","ref":"Plausible.Stats.Compare.html#calculate_change/3"},{"type":"function","doc":"","title":"Plausible.Stats.Compare.percent_change/2","ref":"Plausible.Stats.Compare.html#percent_change/2"},{"type":"module","doc":"This module provides functions for comparing query periods.\n\nIt allows you to compare a given period with a previous period or with the\nsame period from the previous year. For example, you can compare this month's\nmain graph with last month or with the same month from last year.","title":"Plausible.Stats.Comparisons","ref":"Plausible.Stats.Comparisons.html"},{"type":"function","doc":"Generates a comparison query based on the source query and comparison mode.\n\nCurrently only historical periods are supported for comparisons (not `realtime`\nand `30m` periods).\n\nThe mode parameter specifies the type of comparison and can be one of the\nfollowing:\n\n * `\"previous_period\"` - shifts back the query by the same number of days the\n source query has.\n\n * `\"year_over_year\"` - shifts back the query by 1 year.\n\n * `\"custom\"` - compares the query using a custom date range. See options for\n more details.\n\nThe comparison query returned by the function has its end date restricted to\nthe current day. This can be overridden by the `now` option, described below.","title":"Plausible.Stats.Comparisons.compare/4","ref":"Plausible.Stats.Comparisons.html#compare/4"},{"type":"function","doc":"* `:now` - a `NaiveDateTime` struct with the current date and time. This is\n optional and used for testing purposes.\n\n * `:from` - a ISO-8601 date string used when mode is `\"custom\"`.\n\n * `:to` - a ISO-8601 date string used when mode is `\"custom\"`. Must be\n after `from`.\n\n * `:match_day_of_week?` - determines whether the comparison query should be\n adjusted to match the day of the week of the source query. When this option\n is set to true, the comparison query is shifted to start on the same day of\n the week as the source query, rather than on the exact same date. For\n example, if the source query starts on Sunday, January 1st, 2023 and the\n `year_over_year` comparison query is configured to `match_day_of_week?`,\n it will be shifted to start on Sunday, January 2nd, 2022 instead of\n January 1st. Defaults to false.","title":"Options - Plausible.Stats.Comparisons.compare/4","ref":"Plausible.Stats.Comparisons.html#compare/4-options"},{"type":"type","doc":"","title":"Plausible.Stats.Comparisons.mode/0","ref":"Plausible.Stats.Comparisons.html#t:mode/0"},{"type":"module","doc":"","title":"Plausible.Stats.CurrentVisitors","ref":"Plausible.Stats.CurrentVisitors.html"},{"type":"function","doc":"","title":"Plausible.Stats.CurrentVisitors.current_visitors/2","ref":"Plausible.Stats.CurrentVisitors.html#current_visitors/2"},{"type":"module","doc":"Module for querying user defined 'custom properties'.","title":"Plausible.Stats.CustomProps","ref":"Plausible.Stats.CustomProps.html"},{"type":"function","doc":"","title":"Plausible.Stats.CustomProps.fetch_prop_names/2","ref":"Plausible.Stats.CustomProps.html#fetch_prop_names/2"},{"type":"function","doc":"","title":"Plausible.Stats.CustomProps.maybe_allowed_props_only/2","ref":"Plausible.Stats.CustomProps.html#maybe_allowed_props_only/2"},{"type":"module","doc":"Defines a struct similar `Date.Range`, but with `DateTime` instead of `Date`.\n\nThe structs should be created with the `new!/2` function.","title":"Plausible.Stats.DateTimeRange","ref":"Plausible.Stats.DateTimeRange.html"},{"type":"function","doc":"","title":"Plausible.Stats.DateTimeRange.new!/2","ref":"Plausible.Stats.DateTimeRange.html#new!/2"},{"type":"function","doc":"Creates a `DateTimeRange` struct from the given `%Date{}` structs.\n\nThe first datetime will become the first date at 00:00:00, and the last datetime\nwill become the last date at 23:59:59. Both dates will be turned into `%DateTime{}`\nstructs in the given timezone.","title":"Plausible.Stats.DateTimeRange.new!/3","ref":"Plausible.Stats.DateTimeRange.html#new!/3"},{"type":"function","doc":"","title":"Plausible.Stats.DateTimeRange.to_date_range/1","ref":"Plausible.Stats.DateTimeRange.html#to_date_range/1"},{"type":"type","doc":"","title":"Plausible.Stats.DateTimeRange.t/0","ref":"Plausible.Stats.DateTimeRange.html#t:t/0"},{"type":"module","doc":"This module exposes a `get/2` function that returns a map\nof stats needed for email reports. These stats include:\n\n* Total pageviews\n* Unique visitors\n* Bounce rate\n* A list of Top 5 sources (excluding \"Direct / None\")\n* A list of Top 5 pages\n\nwhere total pageviews, unique visitors, and bounce rate\nalso include the change compared to previous period.","title":"Plausible.Stats.EmailReport","ref":"Plausible.Stats.EmailReport.html"},{"type":"function","doc":"","title":"Plausible.Stats.EmailReport.get/2","ref":"Plausible.Stats.EmailReport.html#get/2"},{"type":"module","doc":"","title":"Plausible.Stats.FilterSuggestions","ref":"Plausible.Stats.FilterSuggestions.html"},{"type":"function","doc":"","title":"Plausible.Stats.FilterSuggestions.filter_suggestions/4","ref":"Plausible.Stats.FilterSuggestions.html#filter_suggestions/4"},{"type":"module","doc":"A module for parsing filters used in stat queries.","title":"Plausible.Stats.Filters","ref":"Plausible.Stats.Filters.html"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.event_props/0","ref":"Plausible.Stats.Filters.html#event_props/0"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.event_table_visit_props/0","ref":"Plausible.Stats.Filters.html#event_table_visit_props/0"},{"type":"function","doc":"Parses different filter formats.\n\nDepending on the format and type of the `filters` argument, returns:\n\n * a decoded list, when `filters` is encoded JSON\n * a parsed filter list, when `filters` is a filter expression string\n * the same list, when `filters` is a map\n\nReturns an empty list when argument type is unexpected (e.g. `nil`).\n\n#","title":"Plausible.Stats.Filters.parse/1","ref":"Plausible.Stats.Filters.html#parse/1"},{"type":"function","doc":"iex> Filters.parse(\"{\\\"page\\\":\\\"/blog/**\\\"}\")\n [[:matches, \"event:page\", [\"/blog/**\"]]]\n\n iex> Filters.parse(\"visit:browser!=Chrome\")\n [[:is_not, \"visit:browser\", [\"Chrome\"]]]\n\n iex> Filters.parse(nil)\n []","title":"Examples: - Plausible.Stats.Filters.parse/1","ref":"Plausible.Stats.Filters.html#parse/1-examples"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.visit_props/0","ref":"Plausible.Stats.Filters.html#visit_props/0"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.without_prefix/1","ref":"Plausible.Stats.Filters.html#without_prefix/1"},{"type":"module","doc":"Contains utility functions shared between `LegacyDashboardFilterParser`\nand `StatsAPIFilterParser`.","title":"Plausible.Stats.Filters.Utils","ref":"Plausible.Stats.Filters.Utils.html"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.list_expression?/1","ref":"Plausible.Stats.Filters.Utils.html#list_expression?/1"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.page_regex/1","ref":"Plausible.Stats.Filters.Utils.html#page_regex/1"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.parse_member_list/1","ref":"Plausible.Stats.Filters.Utils.html#parse_member_list/1"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.remove_escape_chars/1","ref":"Plausible.Stats.Filters.Utils.html#remove_escape_chars/1"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.split_goals/1","ref":"Plausible.Stats.Filters.Utils.html#split_goals/1"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.split_goals_query_expressions/1","ref":"Plausible.Stats.Filters.Utils.html#split_goals_query_expressions/1"},{"type":"function","doc":"","title":"Plausible.Stats.Filters.Utils.wildcard_expression?/1","ref":"Plausible.Stats.Filters.Utils.html#wildcard_expression?/1"},{"type":"module","doc":"Module responsible for funnel evaluation, i.e. building and executing\nClickHouse funnel query based on `Plausible.Funnel` definition.","title":"Plausible.Stats.Funnel","ref":"Plausible.Stats.Funnel.html"},{"type":"function","doc":"","title":"Plausible.Stats.Funnel.funnel/3","ref":"Plausible.Stats.Funnel.html#funnel/3"},{"type":"module","doc":"Revenue specific functions for the stats scope","title":"Plausible.Stats.Goal.Revenue","ref":"Plausible.Stats.Goal.Revenue.html"},{"type":"function","doc":"","title":"Plausible.Stats.Goal.Revenue.cast_revenue_metrics_to_money/2","ref":"Plausible.Stats.Goal.Revenue.html#cast_revenue_metrics_to_money/2"},{"type":"function","doc":"Returns the common currency for the goal filters in a query. If there are no\ngoal filters, multiple currencies or the site owner does not have access to\nrevenue goals, `nil` is returned and revenue metrics are dropped.\n\nAggregating revenue data works only for same currency goals. If the query is\nfiltered by goals with different currencies, for example, one USD and other\nEUR, revenue metrics are dropped.","title":"Plausible.Stats.Goal.Revenue.get_revenue_tracking_currency/3","ref":"Plausible.Stats.Goal.Revenue.html#get_revenue_tracking_currency/3"},{"type":"function","doc":"","title":"Plausible.Stats.Goal.Revenue.revenue_metrics/0","ref":"Plausible.Stats.Goal.Revenue.html#revenue_metrics/0"},{"type":"module","doc":"","title":"Plausible.Stats.Imported","ref":"Plausible.Stats.Imported.html"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.goals_with_path/0","ref":"Plausible.Stats.Imported.html#goals_with_path/0"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.goals_with_url/0","ref":"Plausible.Stats.Imported.html#goals_with_url/0"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.merge_imported/4","ref":"Plausible.Stats.Imported.html#merge_imported/4"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.merge_imported_city_suggestions/3","ref":"Plausible.Stats.Imported.html#merge_imported_city_suggestions/3"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.merge_imported_country_suggestions/3","ref":"Plausible.Stats.Imported.html#merge_imported_country_suggestions/3"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.merge_imported_filter_suggestions/5","ref":"Plausible.Stats.Imported.html#merge_imported_filter_suggestions/5"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.merge_imported_region_suggestions/3","ref":"Plausible.Stats.Imported.html#merge_imported_region_suggestions/3"},{"type":"function","doc":"Returns a boolean indicating whether the combination of filters and\nbreakdown property is possible to query from the imported tables.\n\nUsually, when no filters are used, the imported schema supports the\nquery. There is one exception though - breakdown by a custom property.\nWe are currently importing only two custom properties - `url` and `path.\nBoth these properties can only be used with their special goal filter\n(see `@goals_with_url` and `@goals_with_path`).","title":"Plausible.Stats.Imported.schema_supports_query?/1","ref":"Plausible.Stats.Imported.html#schema_supports_query?/1"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.total_imported_visitors/2","ref":"Plausible.Stats.Imported.html#total_imported_visitors/2"},{"type":"module","doc":"A module for building the base of an imported stats query","title":"Plausible.Stats.Imported.Base","ref":"Plausible.Stats.Imported.Base.html"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.Base.decide_tables/1","ref":"Plausible.Stats.Imported.Base.html#decide_tables/1"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.Base.property_to_table_mappings/0","ref":"Plausible.Stats.Imported.Base.html#property_to_table_mappings/0"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.Base.query_imported/2","ref":"Plausible.Stats.Imported.Base.html#query_imported/2"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.Base.query_imported/3","ref":"Plausible.Stats.Imported.Base.html#query_imported/3"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.Base.special_goals_for/1","ref":"Plausible.Stats.Imported.Base.html#special_goals_for/1"},{"type":"module","doc":"This module is responsible for generating SQL/Ecto expressions\nfor dimensions, filters and metrics used in import table queries","title":"Plausible.Stats.Imported.SQL.Builder","ref":"Plausible.Stats.Imported.SQL.Builder.html"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.SQL.Builder.group_imported_by/2","ref":"Plausible.Stats.Imported.SQL.Builder.html#group_imported_by/2"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.SQL.Builder.naive_dimension_join/3","ref":"Plausible.Stats.Imported.SQL.Builder.html#naive_dimension_join/3"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.SQL.Builder.select_imported_metrics/2","ref":"Plausible.Stats.Imported.SQL.Builder.html#select_imported_metrics/2"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.SQL.Builder.select_joined_dimensions/2","ref":"Plausible.Stats.Imported.SQL.Builder.html#select_joined_dimensions/2"},{"type":"function","doc":"","title":"Plausible.Stats.Imported.SQL.Builder.select_joined_metrics/2","ref":"Plausible.Stats.Imported.SQL.Builder.html#select_joined_metrics/2"},{"type":"module","doc":"Collection of functions to work with intervals.\n\nThe interval of a query defines the granularity of the data. You can think of\nit as a `GROUP BY` clause. Possible values are `minute`, `hour`, `day`,\n`week`, and `month`.","title":"Plausible.Stats.Interval","ref":"Plausible.Stats.Interval.html"},{"type":"function","doc":"Returns the suggested interval for the given `DateTimeRange` struct.","title":"Plausible.Stats.Interval.default_for_date_range/1","ref":"Plausible.Stats.Interval.html#default_for_date_range/1"},{"type":"function","doc":"Returns the suggested interval for the given time period.","title":"Plausible.Stats.Interval.default_for_period/1","ref":"Plausible.Stats.Interval.html#default_for_period/1"},{"type":"function","doc":"","title":"Plausible.Stats.Interval.list/0","ref":"Plausible.Stats.Interval.html#list/0"},{"type":"function","doc":"","title":"Plausible.Stats.Interval.valid?/1","ref":"Plausible.Stats.Interval.html#valid?/1"},{"type":"function","doc":"","title":"Plausible.Stats.Interval.valid_by_period/1","ref":"Plausible.Stats.Interval.html#valid_by_period/1"},{"type":"function","doc":"Returns whether the given interval is valid for a time period.\n\nIntervals longer than periods are not supported, e.g. current month stats with\na month interval, or today stats with a week interval.\n\nThere are two dynamic states:\n* `custom` period is only applicable with `month` or `week` intervals,\n if the `opts[:from]` and `opts[:to]` range difference exceeds 12 months\n* `all` period's interval options depend on particular site's `stats_start_date`\n - daily interval is excluded if the all-time range exceeds 12 months","title":"Plausible.Stats.Interval.valid_for_period?/3","ref":"Plausible.Stats.Interval.html#valid_for_period?/3"},{"type":"type","doc":"","title":"Plausible.Stats.Interval.opt/0","ref":"Plausible.Stats.Interval.html#t:opt/0"},{"type":"type","doc":"","title":"Plausible.Stats.Interval.opts/0","ref":"Plausible.Stats.Interval.html#t:opts/0"},{"type":"type","doc":"","title":"Plausible.Stats.Interval.t/0","ref":"Plausible.Stats.Interval.html#t:t/0"},{"type":"module","doc":"Module for validating query parameters against JSON schema.\n\nNote that `internal` queries expose some metrics, filter types and other features not\navailable on the public API.","title":"Plausible.Stats.JSONSchema","ref":"Plausible.Stats.JSONSchema.html"},{"type":"function","doc":"","title":"Plausible.Stats.JSONSchema.raw_public_schema/0","ref":"Plausible.Stats.JSONSchema.html#raw_public_schema/0"},{"type":"function","doc":"","title":"Plausible.Stats.JSONSchema.validate/2","ref":"Plausible.Stats.JSONSchema.html#validate/2"},{"type":"module","doc":"Deprecated module. See QueryParser for list of valid dimensions","title":"Plausible.Stats.Legacy.Dimensions","ref":"Plausible.Stats.Legacy.Dimensions.html"},{"type":"function","doc":"","title":"Plausible.Stats.Legacy.Dimensions.valid?/1","ref":"Plausible.Stats.Legacy.Dimensions.html#valid?/1"},{"type":"module","doc":"Module used to parse URL search params to a valid Query, used to power the API for the dashboard.\n@deprecated","title":"Plausible.Stats.Legacy.QueryBuilder","ref":"Plausible.Stats.Legacy.QueryBuilder.html"},{"type":"function","doc":"","title":"Plausible.Stats.Legacy.QueryBuilder.from/3","ref":"Plausible.Stats.Legacy.QueryBuilder.html#from/3"},{"type":"function","doc":"#","title":"Plausible.Stats.Legacy.QueryBuilder.parse_order_by/1","ref":"Plausible.Stats.Legacy.QueryBuilder.html#parse_order_by/1"},{"type":"function","doc":"iex> QueryBuilder.parse_order_by(nil)\n []\n\n iex> QueryBuilder.parse_order_by(\"\")\n []\n\n iex> QueryBuilder.parse_order_by(\"0\")\n []\n\n iex> QueryBuilder.parse_order_by(\"[}\")\n []\n\n iex> QueryBuilder.parse_order_by(~s({\"any\":\"object\"}))\n []\n\n iex> QueryBuilder.parse_order_by(~s([[\"visitors\",\"invalid\"]]))\n []\n\n iex> QueryBuilder.parse_order_by(~s([[\"visitors\",\"desc\"]]))\n [{:visitors, :desc}]\n\n iex> QueryBuilder.parse_order_by(~s([[\"visitors\",\"asc\"],[\"visit:source\",\"desc\"]]))\n [{:visitors, :asc}, {\"visit:source\", :desc}]","title":"Examples: - Plausible.Stats.Legacy.QueryBuilder.parse_order_by/1","ref":"Plausible.Stats.Legacy.QueryBuilder.html#parse_order_by/1-examples"},{"type":"module","doc":"A module listing all available metrics in Plausible.\n\nUseful for an explicit string to atom conversion.","title":"Plausible.Stats.Metrics","ref":"Plausible.Stats.Metrics.html"},{"type":"function","doc":"","title":"Plausible.Stats.Metrics.from_string/1","ref":"Plausible.Stats.Metrics.html#from_string/1"},{"type":"function","doc":"","title":"Plausible.Stats.Metrics.from_string!/1","ref":"Plausible.Stats.Metrics.html#from_string!/1"},{"type":"function","doc":"","title":"Plausible.Stats.Metrics.metric?/1","ref":"Plausible.Stats.Metrics.html#metric?/1"},{"type":"module","doc":"","title":"Plausible.Stats.Query","ref":"Plausible.Stats.Query.html"},{"type":"function","doc":"","title":"Plausible.Stats.Query.add_filter/2","ref":"Plausible.Stats.Query.html#add_filter/2"},{"type":"function","doc":"","title":"Plausible.Stats.Query.build/4","ref":"Plausible.Stats.Query.html#build/4"},{"type":"function","doc":"","title":"Plausible.Stats.Query.ensure_include_imported/2","ref":"Plausible.Stats.Query.html#ensure_include_imported/2"},{"type":"function","doc":"Builds query from old-style params. New code should prefer Query.build","title":"Plausible.Stats.Query.from/3","ref":"Plausible.Stats.Query.html#from/3"},{"type":"function","doc":"","title":"Plausible.Stats.Query.get_filter/2","ref":"Plausible.Stats.Query.html#get_filter/2"},{"type":"function","doc":"","title":"Plausible.Stats.Query.get_filter_by_prefix/2","ref":"Plausible.Stats.Query.html#get_filter_by_prefix/2"},{"type":"function","doc":"","title":"Plausible.Stats.Query.has_event_filters?/1","ref":"Plausible.Stats.Query.html#has_event_filters?/1"},{"type":"function","doc":"","title":"Plausible.Stats.Query.put_experimental_reduced_joins/3","ref":"Plausible.Stats.Query.html#put_experimental_reduced_joins/3"},{"type":"function","doc":"","title":"Plausible.Stats.Query.put_imported_opts/3","ref":"Plausible.Stats.Query.html#put_imported_opts/3"},{"type":"function","doc":"","title":"Plausible.Stats.Query.remove_filters/2","ref":"Plausible.Stats.Query.html#remove_filters/2"},{"type":"function","doc":"","title":"Plausible.Stats.Query.set/2","ref":"Plausible.Stats.Query.html#set/2"},{"type":"function","doc":"","title":"Plausible.Stats.Query.trace/2","ref":"Plausible.Stats.Query.html#trace/2"},{"type":"type","doc":"","title":"Plausible.Stats.Query.t/0","ref":"Plausible.Stats.Query.html#t:t/0"},{"type":"module","doc":"Methods to manipulate Query for business logic reasons before building an ecto query.","title":"Plausible.Stats.QueryOptimizer","ref":"Plausible.Stats.QueryOptimizer.html"},{"type":"function","doc":"This module manipulates an existing query, updating it according to business logic.\n\n For example, it:\n 1. Figures out what the right granularity to group by time is\n 2. Adds a missing order_by clause to a query\n 3. Updating \"time\" dimension in order_by to the right granularity","title":"Plausible.Stats.QueryOptimizer.optimize/1","ref":"Plausible.Stats.QueryOptimizer.html#optimize/1"},{"type":"function","doc":"Splits a query into event and sessions subcomponents as not all metrics can be\nqueried from a single table.\n\nevent:page dimension is treated in a special way, doing a breakdown of visit:entry_page\nfor sessions.","title":"Plausible.Stats.QueryOptimizer.split/1","ref":"Plausible.Stats.QueryOptimizer.html#split/1"},{"type":"module","doc":"This struct contains the (JSON-encodable) response for a query and\nis responsible for building it from database query results.\n\nFor the convenience of API docs and consumers, the JSON result\nproduced by Jason.encode(query_result) is ordered.","title":"Plausible.Stats.QueryResult","ref":"Plausible.Stats.QueryResult.html"},{"type":"function","doc":"","title":"Plausible.Stats.QueryResult.from/3","ref":"Plausible.Stats.QueryResult.html#from/3"},{"type":"module","doc":"This module is responsible for generating SQL/Ecto expressions\nfor dimensions and metrics used in query SELECT statement.\n\nEach dimension and metric is tagged with with selected_as for easier\nusage down the line.","title":"Plausible.Stats.SQL.Expression","ref":"Plausible.Stats.SQL.Expression.html"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Expression.event_goal_join/2","ref":"Plausible.Stats.SQL.Expression.html#event_goal_join/2"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.Expression.event_metric/1","ref":"Plausible.Stats.SQL.Expression.html#event_metric/1"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.Expression.select_dimension/5","ref":"Plausible.Stats.SQL.Expression.html#select_dimension/5"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.Expression.session_metric/2","ref":"Plausible.Stats.SQL.Expression.html#session_metric/2"},{"type":"module","doc":"Various macros and common SQL fragments used in Stats code.","title":"Plausible.Stats.SQL.Fragments","ref":"Plausible.Stats.SQL.Fragments.html"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Fragments.bounce_rate/0","ref":"Plausible.Stats.SQL.Fragments.html#bounce_rate/0"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Fragments.coalesce_string/2","ref":"Plausible.Stats.SQL.Fragments.html#coalesce_string/2"},{"type":"macro","doc":"Returns value of a key (usually property) under `meta.value` array or similar.\n\nThis macro is used for operating on custom properties.\nCallsites should also check whether key exists first in SQL via `has_key` macro.","title":"Plausible.Stats.SQL.Fragments.get_by_key/3","ref":"Plausible.Stats.SQL.Fragments.html#get_by_key/3"},{"type":"macro","doc":"`get_by_key(e, :meta, \"some_property_name\")` expands to SQL `meta.value[indexOf(meta.key, \"some_property\")]`","title":"Examples - Plausible.Stats.SQL.Fragments.get_by_key/3","ref":"Plausible.Stats.SQL.Fragments.html#get_by_key/3-examples"},{"type":"macro","doc":"Returns whether a key (usually property) exists under `meta.key` array or similar.\n\nThis macro is used for operating on custom properties.","title":"Plausible.Stats.SQL.Fragments.has_key/3","ref":"Plausible.Stats.SQL.Fragments.html#has_key/3"},{"type":"macro","doc":"`has_key(e, :meta, \"some_property_name\")` expands to SQL `has(meta.key, \"some_property_name\")`","title":"Examples - Plausible.Stats.SQL.Fragments.has_key/3","ref":"Plausible.Stats.SQL.Fragments.html#has_key/3-examples"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.Fragments.meta_key_column/1","ref":"Plausible.Stats.SQL.Fragments.html#meta_key_column/1"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.Fragments.meta_value_column/1","ref":"Plausible.Stats.SQL.Fragments.html#meta_value_column/1"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Fragments.sample_percent/0","ref":"Plausible.Stats.SQL.Fragments.html#sample_percent/0"},{"type":"macro","doc":"Macro that helps join two Ecto queries by selecting fields from either one","title":"Plausible.Stats.SQL.Fragments.select_join_fields/4","ref":"Plausible.Stats.SQL.Fragments.html#select_join_fields/4"},{"type":"macro","doc":"Convenience Ecto macro for wrapping select_merge where each value gets in turn passed to selected_as.\n\n#","title":"Plausible.Stats.SQL.Fragments.select_merge_as/3","ref":"Plausible.Stats.SQL.Fragments.html#select_merge_as/3"},{"type":"macro","doc":"iex> select_merge_as(q, [t], %{ foo: t.column }) |> expand_macro_once\n \"select_merge(q, [], ^wrap_alias([t], %{foo: t.column}))\"","title":"Examples - Plausible.Stats.SQL.Fragments.select_merge_as/3","ref":"Plausible.Stats.SQL.Fragments.html#select_merge_as/3-examples"},{"type":"macro","doc":"Converts time or date and time to the specified timezone.\n\nReference: https://clickhouse.com/docs/en/sql-reference/functions/date-time-functions/#totimezone","title":"Plausible.Stats.SQL.Fragments.to_timezone/2","ref":"Plausible.Stats.SQL.Fragments.html#to_timezone/2"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Fragments.total/0","ref":"Plausible.Stats.SQL.Fragments.html#total/0"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Fragments.uniq/1","ref":"Plausible.Stats.SQL.Fragments.html#uniq/1"},{"type":"macro","doc":"","title":"Plausible.Stats.SQL.Fragments.visit_duration/0","ref":"Plausible.Stats.SQL.Fragments.html#visit_duration/0"},{"type":"macro","doc":"Returns the weekstart for `date`. If the weekstart is before the `not_before`\nboundary, `not_before` is returned.","title":"Plausible.Stats.SQL.Fragments.weekstart_not_before/2","ref":"Plausible.Stats.SQL.Fragments.html#weekstart_not_before/2"},{"type":"macro","doc":"In this pseudo-code example, the fragment returns the weekstart. The\n`not_before` boundary is set to the past Saturday, which is before the\nweekstart, therefore the cap does not apply.\n\n ```\n > this_wednesday = ~D[2022-11-09]\n > past_saturday = ~D[2022-11-05]\n > weekstart_not_before(this_wednesday, past_saturday)\n ~D[2022-11-07]\n ```\n\nIn this other example, the fragment returns Tuesday and not the weekstart.\nThe `not_before` boundary is set to Tuesday, which is past the weekstart,\ntherefore the cap applies.\n\n ```\n > this_wednesday = ~D[2022-11-09]\n > this_tuesday = ~D[2022-11-08]\n > weekstart_not_before(this_wednesday, this_tuesday)\n ~D[2022-11-08]\n ```","title":"Examples - Plausible.Stats.SQL.Fragments.weekstart_not_before/2","ref":"Plausible.Stats.SQL.Fragments.html#weekstart_not_before/2-examples"},{"type":"macro","doc":"Convenience Ecto macro for wrapping a map passed to select_merge_as such that each\nexpression gets wrapped in dynamic and set as selected_as.\n\n#","title":"Plausible.Stats.SQL.Fragments.wrap_alias/2","ref":"Plausible.Stats.SQL.Fragments.html#wrap_alias/2"},{"type":"macro","doc":"iex> wrap_alias([t], %{ foo: t.column }) |> expand_macro_once\n \"%{foo: dynamic([t], selected_as(t.column, :foo))}\"","title":"Examples - Plausible.Stats.SQL.Fragments.wrap_alias/2","ref":"Plausible.Stats.SQL.Fragments.html#wrap_alias/2-examples"},{"type":"module","doc":"This module defines how special metrics like `conversion_rate` and\n`percentage` are calculated.","title":"Plausible.Stats.SQL.SpecialMetrics","ref":"Plausible.Stats.SQL.SpecialMetrics.html"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.SpecialMetrics.add/3","ref":"Plausible.Stats.SQL.SpecialMetrics.html#add/3"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.SpecialMetrics.maybe_add_global_conversion_rate/3","ref":"Plausible.Stats.SQL.SpecialMetrics.html#maybe_add_global_conversion_rate/3"},{"type":"function","doc":"","title":"Plausible.Stats.SQL.SpecialMetrics.maybe_add_group_conversion_rate/3","ref":"Plausible.Stats.SQL.SpecialMetrics.html#maybe_add_group_conversion_rate/3"},{"type":"module","doc":"A module for building am ecto where clause of a query out of a query.","title":"Plausible.Stats.SQL.WhereBuilder","ref":"Plausible.Stats.SQL.WhereBuilder.html"},{"type":"function","doc":"Builds WHERE clause for a given Query against sessions or events table","title":"Plausible.Stats.SQL.WhereBuilder.build/3","ref":"Plausible.Stats.SQL.WhereBuilder.html#build/3"},{"type":"function","doc":"Builds WHERE clause condition based off of a filter and a custom column name\nUsed for special business logic cases\n\nAccepts nil as the `filter` parameter, in which case the condition is a no-op (WHERE TRUE).","title":"Plausible.Stats.SQL.WhereBuilder.build_condition/2","ref":"Plausible.Stats.SQL.WhereBuilder.html#build_condition/2"},{"type":"module","doc":"Sampling related functions","title":"Plausible.Stats.Sampling","ref":"Plausible.Stats.Sampling.html"},{"type":"function","doc":"","title":"Plausible.Stats.Sampling.add_query_hint/1","ref":"Plausible.Stats.Sampling.html#add_query_hint/1"},{"type":"function","doc":"","title":"Plausible.Stats.Sampling.add_query_hint/2","ref":"Plausible.Stats.Sampling.html#add_query_hint/2"},{"type":"function","doc":"","title":"Plausible.Stats.Sampling.put_threshold/2","ref":"Plausible.Stats.Sampling.html#put_threshold/2"},{"type":"module","doc":"This module contains logic for deciding which tables need to be queried given a query\nand metrics, with the purpose of reducing the number of queries and JOINs needed to perform.","title":"Plausible.Stats.TableDecider","ref":"Plausible.Stats.TableDecider.html"},{"type":"function","doc":"","title":"Plausible.Stats.TableDecider.events_join_sessions?/1","ref":"Plausible.Stats.TableDecider.html#events_join_sessions?/1"},{"type":"function","doc":"","title":"Plausible.Stats.TableDecider.partition_metrics/2","ref":"Plausible.Stats.TableDecider.html#partition_metrics/2"},{"type":"module","doc":"Collection of functions to work with time in queries.","title":"Plausible.Stats.Time","ref":"Plausible.Stats.Time.html"},{"type":"function","doc":"","title":"Plausible.Stats.Time.date_or_weekstart/2","ref":"Plausible.Stats.Time.html#date_or_weekstart/2"},{"type":"function","doc":"","title":"Plausible.Stats.Time.format_datetime/1","ref":"Plausible.Stats.Time.html#format_datetime/1"},{"type":"function","doc":"Returns list of time bucket labels for the given query.","title":"Plausible.Stats.Time.time_dimension/1","ref":"Plausible.Stats.Time.html#time_dimension/1"},{"type":"function","doc":"","title":"Plausible.Stats.Time.time_labels/1","ref":"Plausible.Stats.Time.html#time_labels/1"},{"type":"function","doc":"","title":"Plausible.Stats.Time.utc_boundaries/2","ref":"Plausible.Stats.Time.html#utc_boundaries/2"},{"type":"module","doc":"Builds timeseries results for v1 of our stats API and dashboards.\n\nAvoid adding new logic here - update QueryBuilder etc instead.","title":"Plausible.Stats.Timeseries","ref":"Plausible.Stats.Timeseries.html"},{"type":"function","doc":"","title":"Plausible.Stats.Timeseries.timeseries/3","ref":"Plausible.Stats.Timeseries.html#timeseries/3"},{"type":"module","doc":"Utilities for modifying stat results","title":"Plausible.Stats.Util","ref":"Plausible.Stats.Util.html"},{"type":"function","doc":"Sometimes we need to manually add metrics in order to calculate the value for\nother metrics. E.g:\n\n* `__internal_visits` is fetched when querying bounce rate, visit duration,\n or views_per_visit, as it is needed to calculate these from imported data.\n\n* `visitors` metric might be added manually via `maybe_add_visitors_metric/1`,\n in order to be able to calculate conversion rate.\n\nThis function can be used for stripping those metrics from a breakdown (list),\nor an aggregate (map) result. We do not want to return metrics that we're not\nrequested.","title":"Plausible.Stats.Util.keep_requested_metrics/2","ref":"Plausible.Stats.Util.html#keep_requested_metrics/2"},{"type":"function","doc":"This function adds the `visitors` metric into the list of\ngiven metrics if it's not already there and if it is needed\nfor any of the other metrics to be calculated.","title":"Plausible.Stats.Util.maybe_add_visitors_metric/1","ref":"Plausible.Stats.Util.html#maybe_add_visitors_metric/1"},{"type":"function","doc":"","title":"Plausible.Stats.Util.shortname/2","ref":"Plausible.Stats.Util.html#shortname/2"},{"type":"module","doc":"Floki wrappers to help make assertions about HTML/DOM structures","title":"Plausible.Test.Support.HTML","ref":"Plausible.Test.Support.HTML.html"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.class_of_element/2","ref":"Plausible.Test.Support.HTML.html#class_of_element/2"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.element_exists?/2","ref":"Plausible.Test.Support.HTML.html#element_exists?/2"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.find/2","ref":"Plausible.Test.Support.HTML.html#find/2"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.form_exists?/2","ref":"Plausible.Test.Support.HTML.html#form_exists?/2"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.name_of/1","ref":"Plausible.Test.Support.HTML.html#name_of/1"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.submit_button/2","ref":"Plausible.Test.Support.HTML.html#submit_button/2"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.text/1","ref":"Plausible.Test.Support.HTML.html#text/1"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.text_of_attr/2","ref":"Plausible.Test.Support.HTML.html#text_of_attr/2"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.text_of_attr/3","ref":"Plausible.Test.Support.HTML.html#text_of_attr/3"},{"type":"function","doc":"","title":"Plausible.Test.Support.HTML.text_of_element/2","ref":"Plausible.Test.Support.HTML.html#text_of_element/2"},{"type":"module","doc":"Currently only supports post request, it's a drop-in replacement\nfor our exvcr usage that wasn't ever needed (e.g. we had no way to\nre-record the cassettes anyway).","title":"Plausible.Test.Support.HTTPMocker","ref":"Plausible.Test.Support.HTTPMocker.html"},{"type":"module","doc":"","title":"Plausible.TestUtils","ref":"Plausible.TestUtils.html"},{"type":"function","doc":"","title":"Plausible.TestUtils.await_clickhouse_count/2","ref":"Plausible.TestUtils.html#await_clickhouse_count/2"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_api_key/1","ref":"Plausible.TestUtils.html#create_api_key/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_legacy_site_import/1","ref":"Plausible.TestUtils.html#create_legacy_site_import/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_new_site/1","ref":"Plausible.TestUtils.html#create_new_site/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_pageviews/1","ref":"Plausible.TestUtils.html#create_pageviews/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_site/1","ref":"Plausible.TestUtils.html#create_site/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_site_import/1","ref":"Plausible.TestUtils.html#create_site_import/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.create_user/1","ref":"Plausible.TestUtils.html#create_user/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.ensure_minio/0","ref":"Plausible.TestUtils.html#ensure_minio/0"},{"type":"function","doc":"","title":"Plausible.TestUtils.eventually/3","ref":"Plausible.TestUtils.html#eventually/3"},{"type":"function","doc":"","title":"Plausible.TestUtils.generate_usage_for/3","ref":"Plausible.TestUtils.html#generate_usage_for/3"},{"type":"function","doc":"","title":"Plausible.TestUtils.init_session/1","ref":"Plausible.TestUtils.html#init_session/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.log_in/1","ref":"Plausible.TestUtils.html#log_in/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.maybe_fake_minio/1","ref":"Plausible.TestUtils.html#maybe_fake_minio/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.minio_running?/0","ref":"Plausible.TestUtils.html#minio_running?/0"},{"type":"macro","doc":"","title":"Plausible.TestUtils.patch_env/2","ref":"Plausible.TestUtils.html#patch_env/2"},{"type":"function","doc":"","title":"Plausible.TestUtils.populate_stats/1","ref":"Plausible.TestUtils.html#populate_stats/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.populate_stats/2","ref":"Plausible.TestUtils.html#populate_stats/2"},{"type":"function","doc":"","title":"Plausible.TestUtils.populate_stats/3","ref":"Plausible.TestUtils.html#populate_stats/3"},{"type":"function","doc":"","title":"Plausible.TestUtils.random_ip/0","ref":"Plausible.TestUtils.html#random_ip/0"},{"type":"function","doc":"","title":"Plausible.TestUtils.relative_time/1","ref":"Plausible.TestUtils.html#relative_time/1"},{"type":"macro","doc":"","title":"Plausible.TestUtils.setup_patch_env/2","ref":"Plausible.TestUtils.html#setup_patch_env/2"},{"type":"function","doc":"","title":"Plausible.TestUtils.to_naive_truncate/1","ref":"Plausible.TestUtils.html#to_naive_truncate/1"},{"type":"function","doc":"","title":"Plausible.TestUtils.use_api_key/1","ref":"Plausible.TestUtils.html#use_api_key/1"},{"type":"module","doc":"","title":"Plausible.Themes","ref":"Plausible.Themes.html"},{"type":"function","doc":"","title":"Plausible.Themes.options/0","ref":"Plausible.Themes.html#options/0"},{"type":"module","doc":"","title":"Plausible.Timezones","ref":"Plausible.Timezones.html"},{"type":"function","doc":"","title":"Plausible.Timezones.options/1","ref":"Plausible.Timezones.html#options/1"},{"type":"function","doc":"","title":"Plausible.Timezones.to_date_in_timezone/2","ref":"Plausible.Timezones.html#to_date_in_timezone/2"},{"type":"function","doc":"","title":"Plausible.Timezones.to_datetime_in_timezone/2","ref":"Plausible.Timezones.html#to_datetime_in_timezone/2"},{"type":"module","doc":"User context","title":"Plausible.Users","ref":"Plausible.Users.html"},{"type":"function","doc":"","title":"Plausible.Users.accept_traffic_until/1","ref":"Plausible.Users.html#accept_traffic_until/1"},{"type":"function","doc":"","title":"Plausible.Users.allow_next_upgrade_override/1","ref":"Plausible.Users.html#allow_next_upgrade_override/1"},{"type":"function","doc":"","title":"Plausible.Users.bump_last_seen/2","ref":"Plausible.Users.html#bump_last_seen/2"},{"type":"function","doc":"","title":"Plausible.Users.has_email_code?/1","ref":"Plausible.Users.html#has_email_code?/1"},{"type":"function","doc":"","title":"Plausible.Users.last_subscription_join_query/0","ref":"Plausible.Users.html#last_subscription_join_query/0"},{"type":"function","doc":"","title":"Plausible.Users.maybe_reset_next_upgrade_override/1","ref":"Plausible.Users.html#maybe_reset_next_upgrade_override/1"},{"type":"function","doc":"","title":"Plausible.Users.on_trial?/1","ref":"Plausible.Users.html#on_trial?/1"},{"type":"function","doc":"","title":"Plausible.Users.trial_days_left/1","ref":"Plausible.Users.html#trial_days_left/1"},{"type":"function","doc":"","title":"Plausible.Users.update_accept_traffic_until/1","ref":"Plausible.Users.html#update_accept_traffic_until/1"},{"type":"function","doc":"","title":"Plausible.Users.with_subscription/1","ref":"Plausible.Users.html#with_subscription/1"},{"type":"module","doc":"Module defining the user-agent used for site verification.","title":"Plausible.Verification","ref":"Plausible.Verification.html"},{"type":"function","doc":"","title":"Plausible.Verification.user_agent/0","ref":"Plausible.Verification.html#user_agent/0"},{"type":"behaviour","doc":"Behaviour to be implemented by specific site verification checks.\n`report_progress_as()` doesn't necessarily reflect the actual check description,\nit serves as a user-facing message grouping mechanism, to prevent frequent message flashing when checks rotate often.\nEach check operates on `state()` and is expected to return it, optionally modified, by all means.\n`perform_safe/1` is used to guarantee no exceptions are thrown by faulty implementations, not to interrupt LiveView.","title":"Plausible.Verification.Check","ref":"Plausible.Verification.Check.html"},{"type":"callback","doc":"","title":"Plausible.Verification.Check.perform/1","ref":"Plausible.Verification.Check.html#c:perform/1"},{"type":"callback","doc":"","title":"Plausible.Verification.Check.report_progress_as/0","ref":"Plausible.Verification.Check.html#c:report_progress_as/0"},{"type":"type","doc":"","title":"Plausible.Verification.Check.state/0","ref":"Plausible.Verification.Check.html#t:state/0"},{"type":"module","doc":"Checks that are performed during site verification.\nEach module defined in `@checks` implements the `Plausible.Verification.Check` behaviour.\nChecks are normally run asynchronously, except when synchronous execution is optionally required\nfor tests. Slowdowns can be optionally added, the user doesn't benefit from running the checks too quickly.\n\nIn async execution, each check notifies the caller by sending a message to it.","title":"Plausible.Verification.Checks","ref":"Plausible.Verification.Checks.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.interpret_diagnostics/1","ref":"Plausible.Verification.Checks.html#interpret_diagnostics/1"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.run/3","ref":"Plausible.Verification.Checks.html#run/3"},{"type":"module","doc":"Scans the Content Security Policy header to ensure that the Plausible domain is allowed.\nSee `Plausible.Verification.Checks` for the execution sequence.","title":"Plausible.Verification.Checks.CSP","ref":"Plausible.Verification.Checks.CSP.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.CSP.perform_safe/1","ref":"Plausible.Verification.Checks.CSP.html#perform_safe/1"},{"type":"module","doc":"Fetches the body of the site and extracts the HTML document, if available, for\nfurther processing.\nSee `Plausible.Verification.Checks` for the execution sequence.","title":"Plausible.Verification.Checks.FetchBody","ref":"Plausible.Verification.Checks.FetchBody.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.FetchBody.perform_safe/1","ref":"Plausible.Verification.Checks.FetchBody.html#perform_safe/1"},{"type":"module","doc":"Calls the browserless.io service (local instance can be spawned with `make browserless`)\nand runs verification/verify_plausible_installed.js via the [function API](https://docs.browserless.io/HTTP-APIs/function).\n\nThe successful execution assumes the following JSON payload:\n - `data.plausibleInstalled` - boolean indicating whether the `plausible()` window function was found\n - `data.callbackStatus` - integer. 202 indicates that the server acknowledged the test event.\n\nThe test event ingestion is discarded based on user-agent, see: `Plausible.Verification.user_agent/0`","title":"Plausible.Verification.Checks.Installation","ref":"Plausible.Verification.Checks.Installation.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.Installation.perform_safe/1","ref":"Plausible.Verification.Checks.Installation.html#perform_safe/1"},{"type":"module","doc":"Naive way of detecting GTM and WordPress powered sites.","title":"Plausible.Verification.Checks.ScanBody","ref":"Plausible.Verification.Checks.ScanBody.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.ScanBody.perform_safe/1","ref":"Plausible.Verification.Checks.ScanBody.html#perform_safe/1"},{"type":"module","doc":"The check looks for Plausible snippets and tries to address the common\nintegration issues, such as bad placement, data-domain typos, unknown \nattributes frequently added by performance optimization plugins, etc.","title":"Plausible.Verification.Checks.Snippet","ref":"Plausible.Verification.Checks.Snippet.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.Snippet.perform_safe/1","ref":"Plausible.Verification.Checks.Snippet.html#perform_safe/1"},{"type":"module","doc":"A naive way of trying to figure out whether the latest site contents \nis wrapped with some CDN/caching layer.\nIn case no snippets were found, we'll try to bust the cache by appending a random query parameter\nand re-run `Plausible.Verification.Checks.FetchBody` and `Plausible.Verification.Checks.Snippet` checks.\nIf the result is different this time, we'll assume cache likely.","title":"Plausible.Verification.Checks.SnippetCacheBust","ref":"Plausible.Verification.Checks.SnippetCacheBust.html"},{"type":"function","doc":"","title":"Plausible.Verification.Checks.SnippetCacheBust.perform_safe/1","ref":"Plausible.Verification.Checks.SnippetCacheBust.html#perform_safe/1"},{"type":"module","doc":"Module responsible for translating diagnostics to user-friendly errors and recommendations.","title":"Plausible.Verification.Diagnostics","ref":"Plausible.Verification.Diagnostics.html"},{"type":"function","doc":"","title":"Plausible.Verification.Diagnostics.interpret/2","ref":"Plausible.Verification.Diagnostics.html#interpret/2"},{"type":"type","doc":"","title":"Plausible.Verification.Diagnostics.t/0","ref":"Plausible.Verification.Diagnostics.html#t:t/0"},{"type":"module","doc":"Diagnostics interpretation result.","title":"Plausible.Verification.Diagnostics.Result","ref":"Plausible.Verification.Diagnostics.Result.html"},{"type":"type","doc":"","title":"Plausible.Verification.Diagnostics.Result.t/0","ref":"Plausible.Verification.Diagnostics.Result.html#t:t/0"},{"type":"module","doc":"A go-to definition of all verification errors","title":"Plausible.Verification.Errors","ref":"Plausible.Verification.Errors.html"},{"type":"function","doc":"","title":"Plausible.Verification.Errors.all/0","ref":"Plausible.Verification.Errors.html#all/0"},{"type":"module","doc":"The struct and interface describing the state of the site verification process.\nAssigns are meant to be used to communicate between checks, while diagnostics\nare later on interpreted (translated into user-friendly messages and recommendations)\nvia `Plausible.Verification.Diagnostics` module.","title":"Plausible.Verification.State","ref":"Plausible.Verification.State.html"},{"type":"function","doc":"","title":"Plausible.Verification.State.assign/2","ref":"Plausible.Verification.State.html#assign/2"},{"type":"function","doc":"","title":"Plausible.Verification.State.put_diagnostics/2","ref":"Plausible.Verification.State.html#put_diagnostics/2"},{"type":"type","doc":"","title":"Plausible.Verification.State.t/0","ref":"Plausible.Verification.State.html#t:t/0"},{"type":"module","doc":"Busting some caches by appending ?plausible_verification=12345 to it.","title":"Plausible.Verification.URL","ref":"Plausible.Verification.URL.html"},{"type":"function","doc":"","title":"Plausible.Verification.URL.bust_url/1","ref":"Plausible.Verification.URL.html#bust_url/1"},{"type":"module","doc":"A worker meant to be run once a day that sends out e-mail notifications to site\nowners assuming:\n - their sites still receive traffic (i.e. have stats for yesterday)\n - `site.accept_traffic_until` is approaching either tomorrow or exactly in 7 days\n\nUsers having no sites or sites that receive no traffic, won't be notified.\nWe make a tiny effort here to make sure we send the same notification at most once a day.","title":"Plausible.Workers.AcceptTrafficUntil","ref":"Plausible.Workers.AcceptTrafficUntil.html"},{"type":"function","doc":"","title":"Plausible.Workers.AcceptTrafficUntil.dry_run/1","ref":"Plausible.Workers.AcceptTrafficUntil.html#dry_run/1"},{"type":"module","doc":"","title":"Plausible.Workers.CheckUsage","ref":"Plausible.Workers.CheckUsage.html"},{"type":"function","doc":"","title":"Plausible.Workers.CheckUsage.check_enterprise_subscriber/2","ref":"Plausible.Workers.CheckUsage.html#check_enterprise_subscriber/2"},{"type":"macro","doc":"","title":"Plausible.Workers.CheckUsage.day_of_month/1","ref":"Plausible.Workers.CheckUsage.html#day_of_month/1"},{"type":"macro","doc":"","title":"Plausible.Workers.CheckUsage.last_day_of_month/1","ref":"Plausible.Workers.CheckUsage.html#last_day_of_month/1"},{"type":"macro","doc":"","title":"Plausible.Workers.CheckUsage.least/2","ref":"Plausible.Workers.CheckUsage.html#least/2"},{"type":"function","doc":"","title":"Plausible.Workers.CheckUsage.maybe_remove_grace_period/2","ref":"Plausible.Workers.CheckUsage.html#maybe_remove_grace_period/2"},{"type":"macro","doc":"","title":"Plausible.Workers.CheckUsage.yesterday/0","ref":"Plausible.Workers.CheckUsage.html#yesterday/0"},{"type":"module","doc":"","title":"Plausible.Workers.CleanInvitations","ref":"Plausible.Workers.CleanInvitations.html"},{"type":"module","doc":"Job removing expired user sessions. A grace period is applied.","title":"Plausible.Workers.CleanUserSessions","ref":"Plausible.Workers.CleanUserSessions.html"},{"type":"function","doc":"","title":"Plausible.Workers.CleanUserSessions.grace_period_duration/0","ref":"Plausible.Workers.CleanUserSessions.html#grace_period_duration/0"},{"type":"module","doc":"Cleans deleted site data from ClickHouse asynchronously.\n\nWe batch up data deletions from ClickHouse as deleting a single site is\njust as expensive as deleting many.","title":"Plausible.Workers.ClickhouseCleanSites","ref":"Plausible.Workers.ClickhouseCleanSites.html"},{"type":"function","doc":"","title":"Plausible.Workers.ClickhouseCleanSites.get_deleted_sites_with_clickhouse_data/0","ref":"Plausible.Workers.ClickhouseCleanSites.html#get_deleted_sites_with_clickhouse_data/0"},{"type":"function","doc":"","title":"Plausible.Workers.ClickhouseCleanSites.perform/1","ref":"Plausible.Workers.ClickhouseCleanSites.html#perform/1"},{"type":"module","doc":"Periodic worker that expires domain change transition period.\nOld domains are frozen for a given time, so users can still access them\nbefore redeploying their scripts and integrations.","title":"Plausible.Workers.ExpireDomainChangeTransitions","ref":"Plausible.Workers.ExpireDomainChangeTransitions.html"},{"type":"module","doc":"Worker for running CSV export jobs. Supports S3 and local storage.\nTo avoid blocking the queue, a timeout of 15 minutes is enforced.","title":"Plausible.Workers.ExportAnalytics","ref":"Plausible.Workers.ExportAnalytics.html"},{"type":"function","doc":"This base query filters export jobs for a site","title":"Plausible.Workers.ExportAnalytics.base_query/1","ref":"Plausible.Workers.ExportAnalytics.html#base_query/1"},{"type":"module","doc":"Worker for running analytics import jobs.","title":"Plausible.Workers.ImportAnalytics","ref":"Plausible.Workers.ImportAnalytics.html"},{"type":"function","doc":"","title":"Plausible.Workers.ImportAnalytics.import_complete/1","ref":"Plausible.Workers.ImportAnalytics.html#import_complete/1"},{"type":"function","doc":"","title":"Plausible.Workers.ImportAnalytics.import_fail/2","ref":"Plausible.Workers.ImportAnalytics.html#import_fail/2"},{"type":"function","doc":"","title":"Plausible.Workers.ImportAnalytics.import_fail_transient/1","ref":"Plausible.Workers.ImportAnalytics.html#import_fail_transient/1"},{"type":"module","doc":"Worker for cleaning local files left after analytics import jobs.","title":"Plausible.Workers.LocalImportAnalyticsCleaner","ref":"Plausible.Workers.LocalImportAnalyticsCleaner.html"},{"type":"module","doc":"","title":"Plausible.Workers.LockSites","ref":"Plausible.Workers.LockSites.html"},{"type":"module","doc":"","title":"Plausible.Workers.NotifyAnnualRenewal","ref":"Plausible.Workers.NotifyAnnualRenewal.html"},{"type":"function","doc":"Sends a notification at most 7 days and at least 1 day before the renewal of an annual subscription","title":"Plausible.Workers.NotifyAnnualRenewal.perform/1","ref":"Plausible.Workers.NotifyAnnualRenewal.html#perform/1"},{"type":"module","doc":"This worker delivers emails for successful and failed exports","title":"Plausible.Workers.NotifyExportedAnalytics","ref":"Plausible.Workers.NotifyExportedAnalytics.html"},{"type":"module","doc":"","title":"Plausible.Workers.RotateSalts","ref":"Plausible.Workers.RotateSalts.html"},{"type":"module","doc":"","title":"Plausible.Workers.ScheduleEmailReports","ref":"Plausible.Workers.ScheduleEmailReports.html"},{"type":"function","doc":"","title":"Plausible.Workers.ScheduleEmailReports.first_of_month_9am/1","ref":"Plausible.Workers.ScheduleEmailReports.html#first_of_month_9am/1"},{"type":"function","doc":"","title":"Plausible.Workers.ScheduleEmailReports.monday_9am/1","ref":"Plausible.Workers.ScheduleEmailReports.html#monday_9am/1"},{"type":"function","doc":"Email reports should be sent on Monday at 9am according to the timezone\nof a site. This job runs every day at midnight to ensure that all sites\nhave a scheduled job for email reports.","title":"Plausible.Workers.ScheduleEmailReports.perform/1","ref":"Plausible.Workers.ScheduleEmailReports.html#perform/1"},{"type":"module","doc":"","title":"Plausible.Workers.SendCheckStatsEmails","ref":"Plausible.Workers.SendCheckStatsEmails.html"},{"type":"module","doc":"","title":"Plausible.Workers.SendEmailReport","ref":"Plausible.Workers.SendEmailReport.html"},{"type":"module","doc":"","title":"Plausible.Workers.SendSiteSetupEmails","ref":"Plausible.Workers.SendSiteSetupEmails.html"},{"type":"module","doc":"","title":"Plausible.Workers.SendTrialNotifications","ref":"Plausible.Workers.SendTrialNotifications.html"},{"type":"module","doc":"Oban service sending out traffic drop/spike notifications","title":"Plausible.Workers.TrafficChangeNotifier","ref":"Plausible.Workers.TrafficChangeNotifier.html"},{"type":"module","doc":"","title":"PlausibleWeb","ref":"PlausibleWeb.html"},{"type":"macro","doc":"When used, dispatch to the appropriate controller/view/etc.","title":"PlausibleWeb.__using__/1","ref":"PlausibleWeb.html#__using__/1"},{"type":"function","doc":"","title":"PlausibleWeb.channel/0","ref":"PlausibleWeb.html#channel/0"},{"type":"function","doc":"","title":"PlausibleWeb.controller/0","ref":"PlausibleWeb.html#controller/0"},{"type":"function","doc":"","title":"PlausibleWeb.live_view/1","ref":"PlausibleWeb.html#live_view/1"},{"type":"function","doc":"","title":"PlausibleWeb.open_api_schema/0","ref":"PlausibleWeb.html#open_api_schema/0"},{"type":"function","doc":"","title":"PlausibleWeb.plugins_api_controller/0","ref":"PlausibleWeb.html#plugins_api_controller/0"},{"type":"function","doc":"","title":"PlausibleWeb.plugins_api_view/0","ref":"PlausibleWeb.html#plugins_api_view/0"},{"type":"function","doc":"","title":"PlausibleWeb.router/0","ref":"PlausibleWeb.html#router/0"},{"type":"function","doc":"","title":"PlausibleWeb.view/0","ref":"PlausibleWeb.html#view/0"},{"type":"module","doc":"","title":"PlausibleWeb.AdminController","ref":"PlausibleWeb.AdminController.html"},{"type":"function","doc":"","title":"PlausibleWeb.AdminController.current_plan/2","ref":"PlausibleWeb.AdminController.html#current_plan/2"},{"type":"function","doc":"","title":"PlausibleWeb.AdminController.usage/2","ref":"PlausibleWeb.AdminController.html#usage/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.ExternalController","ref":"PlausibleWeb.Api.ExternalController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalController.error/2","ref":"PlausibleWeb.Api.ExternalController.html#error/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalController.event/2","ref":"PlausibleWeb.Api.ExternalController.html#event/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalController.health/2","ref":"PlausibleWeb.Api.ExternalController.html#health/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalController.info/2","ref":"PlausibleWeb.Api.ExternalController.html#info/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.ExternalSitesController","ref":"PlausibleWeb.Api.ExternalSitesController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.create_site/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#create_site/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.delete_goal/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#delete_goal/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.delete_site/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#delete_site/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.find_or_create_goal/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#find_or_create_goal/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.find_or_create_shared_link/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#find_or_create_shared_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.get_site/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#get_site/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.goals_index/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#goals_index/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.index/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#index/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalSitesController.update_site/2","ref":"PlausibleWeb.Api.ExternalSitesController.html#update_site/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.ExternalStatsController","ref":"PlausibleWeb.Api.ExternalStatsController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalStatsController.aggregate/2","ref":"PlausibleWeb.Api.ExternalStatsController.html#aggregate/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalStatsController.breakdown/2","ref":"PlausibleWeb.Api.ExternalStatsController.html#breakdown/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalStatsController.realtime_visitors/2","ref":"PlausibleWeb.Api.ExternalStatsController.html#realtime_visitors/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.ExternalStatsController.timeseries/2","ref":"PlausibleWeb.Api.ExternalStatsController.html#timeseries/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.Helpers","ref":"PlausibleWeb.Api.Helpers.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.Helpers.bad_request/2","ref":"PlausibleWeb.Api.Helpers.html#bad_request/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.Helpers.not_found/2","ref":"PlausibleWeb.Api.Helpers.html#not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.Helpers.payment_required/2","ref":"PlausibleWeb.Api.Helpers.html#payment_required/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.Helpers.too_many_requests/2","ref":"PlausibleWeb.Api.Helpers.html#too_many_requests/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.Helpers.unauthorized/2","ref":"PlausibleWeb.Api.Helpers.html#unauthorized/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.InternalController","ref":"PlausibleWeb.Api.InternalController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.InternalController.disable_feature/2","ref":"PlausibleWeb.Api.InternalController.html#disable_feature/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.InternalController.sites/2","ref":"PlausibleWeb.Api.InternalController.html#sites/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.PaddleController","ref":"PlausibleWeb.Api.PaddleController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.PaddleController.currency/2","ref":"PlausibleWeb.Api.PaddleController.html#currency/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.PaddleController.verify_signature/2","ref":"PlausibleWeb.Api.PaddleController.html#verify_signature/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.PaddleController.webhook/2","ref":"PlausibleWeb.Api.PaddleController.html#webhook/2"},{"type":"module","doc":"","title":"PlausibleWeb.Api.StatsController","ref":"PlausibleWeb.Api.StatsController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.all_custom_prop_values/2","ref":"PlausibleWeb.Api.StatsController.html#all_custom_prop_values/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.browser_versions/2","ref":"PlausibleWeb.Api.StatsController.html#browser_versions/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.browsers/2","ref":"PlausibleWeb.Api.StatsController.html#browsers/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.build_intervals/4","ref":"PlausibleWeb.Api.StatsController.html#build_intervals/4"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.cities/2","ref":"PlausibleWeb.Api.StatsController.html#cities/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.conversions/2","ref":"PlausibleWeb.Api.StatsController.html#conversions/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.countries/2","ref":"PlausibleWeb.Api.StatsController.html#countries/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.current_visitors/2","ref":"PlausibleWeb.Api.StatsController.html#current_visitors/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.custom_prop_values/2","ref":"PlausibleWeb.Api.StatsController.html#custom_prop_values/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.entry_pages/2","ref":"PlausibleWeb.Api.StatsController.html#entry_pages/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.exit_pages/2","ref":"PlausibleWeb.Api.StatsController.html#exit_pages/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.filter_suggestions/2","ref":"PlausibleWeb.Api.StatsController.html#filter_suggestions/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.format_money/1","ref":"PlausibleWeb.Api.StatsController.html#format_money/1"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.format_revenue_metric/1","ref":"PlausibleWeb.Api.StatsController.html#format_revenue_metric/1"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.funnel/2","ref":"PlausibleWeb.Api.StatsController.html#funnel/2"},{"type":"function","doc":"Returns a time-series based on given parameters.","title":"PlausibleWeb.Api.StatsController.main_graph/2","ref":"PlausibleWeb.Api.StatsController.html#main_graph/2"},{"type":"function","doc":"This API accepts the following parameters:\n\n * `period` - x-axis of the graph, e.g. `12mo`, `day`, `custom`.\n\n * `metric` - y-axis of the graph, e.g. `visits`, `visitors`, `pageviews`.\n See the Stats API [\"Metrics\"](https://plausible.io/docs/stats-api#metrics)\n section for more details. Defaults to `visitors`.\n\n * `interval` - granularity of the time-series data. You can think of it as\n a `GROUP BY` clause. Possible values are `minute`, `hour`, `date`, `week`,\n and `month`. The default depends on the `period` parameter. Check\n `Plausible.Query.from/2` for each default.\n\n * `filters` - optional filters to drill down data. See the Stats API\n [\"Filtering\"](https://plausible.io/docs/stats-api#filtering) section for\n more details.\n\n * `with_imported` - boolean indicating whether to include Google Analytics\n imported data or not. Defaults to `false`.\n\nFull example:\n```elixir\n%{\n \"from\" => \"2021-09-06\",\n \"interval\" => \"month\",\n \"metric\" => \"visitors\",\n \"period\" => \"custom\",\n \"to\" => \"2021-12-13\"\n}\n```","title":"Parameters - PlausibleWeb.Api.StatsController.main_graph/2","ref":"PlausibleWeb.Api.StatsController.html#main_graph/2-parameters"},{"type":"function","doc":"Returns a map with the following keys:\n\n * `plot` - list of values for the requested metric representing the y-axis\n of the graph.\n\n * `labels` - list of date times representing the x-axis of the graph.\n\n * `present_index` - index of the element representing the current date in\n `labels` and `plot` lists.\n\n * `interval` - the interval used for querying.\n\n * `includes_imported` - boolean indicating whether imported data\n was queried or not.\n\n * `imports_exist` - boolean indicating whether there are any completed\n imports for a given site or not.\n\n * `full_intervals` - map of dates indicating whether the interval has been\n cut off by the requested date range or not. For example, if looking at a\n month week-by-week, some weeks may be cut off by the month boundaries.\n It's useful to adjust the graph display slightly in case the interval is\n not 'full' so that the user understands why the numbers might be lower for\n those partial periods.\n\nFull example:\n```elixir\n%{\n \"full_intervals\" => %{\n \"2021-09-01\" => false,\n \"2021-10-01\" => true,\n \"2021-11-01\" => true,\n \"2021-12-01\" => false\n },\n \"imports_exist\" => false,\n \"interval\" => \"month\",\n \"labels\" => [\"2021-09-01\", \"2021-10-01\", \"2021-11-01\", \"2021-12-01\"],\n \"plot\" => [0, 0, 0, 0],\n \"present_index\" => nil,\n \"includes_imported\" => false\n}\n```","title":"Response - PlausibleWeb.Api.StatsController.main_graph/2","ref":"PlausibleWeb.Api.StatsController.html#main_graph/2-response"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.operating_system_versions/2","ref":"PlausibleWeb.Api.StatsController.html#operating_system_versions/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.operating_systems/2","ref":"PlausibleWeb.Api.StatsController.html#operating_systems/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.pages/2","ref":"PlausibleWeb.Api.StatsController.html#pages/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.put_combined_name_with_version/2","ref":"PlausibleWeb.Api.StatsController.html#put_combined_name_with_version/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.referrer_drilldown/2","ref":"PlausibleWeb.Api.StatsController.html#referrer_drilldown/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.referrers/2","ref":"PlausibleWeb.Api.StatsController.html#referrers/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.regions/2","ref":"PlausibleWeb.Api.StatsController.html#regions/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.screen_sizes/2","ref":"PlausibleWeb.Api.StatsController.html#screen_sizes/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.sources/2","ref":"PlausibleWeb.Api.StatsController.html#sources/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.top_stats/2","ref":"PlausibleWeb.Api.StatsController.html#top_stats/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.utm_campaigns/2","ref":"PlausibleWeb.Api.StatsController.html#utm_campaigns/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.utm_contents/2","ref":"PlausibleWeb.Api.StatsController.html#utm_contents/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.utm_mediums/2","ref":"PlausibleWeb.Api.StatsController.html#utm_mediums/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.utm_sources/2","ref":"PlausibleWeb.Api.StatsController.html#utm_sources/2"},{"type":"function","doc":"","title":"PlausibleWeb.Api.StatsController.utm_terms/2","ref":"PlausibleWeb.Api.StatsController.html#utm_terms/2"},{"type":"module","doc":"","title":"PlausibleWeb.AuthController","ref":"PlausibleWeb.AuthController.html"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.activate/2","ref":"PlausibleWeb.AuthController.html#activate/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.activate_form/2","ref":"PlausibleWeb.AuthController.html#activate_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.cancel_update_email/2","ref":"PlausibleWeb.AuthController.html#cancel_update_email/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.create_api_key/2","ref":"PlausibleWeb.AuthController.html#create_api_key/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.delete_api_key/2","ref":"PlausibleWeb.AuthController.html#delete_api_key/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.delete_me/2","ref":"PlausibleWeb.AuthController.html#delete_me/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.disable_2fa/2","ref":"PlausibleWeb.AuthController.html#disable_2fa/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.generate_2fa_recovery_codes/2","ref":"PlausibleWeb.AuthController.html#generate_2fa_recovery_codes/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.google_auth_callback/2","ref":"PlausibleWeb.AuthController.html#google_auth_callback/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.initiate_2fa_setup/2","ref":"PlausibleWeb.AuthController.html#initiate_2fa_setup/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.login/2","ref":"PlausibleWeb.AuthController.html#login/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.login_form/2","ref":"PlausibleWeb.AuthController.html#login_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.logout/2","ref":"PlausibleWeb.AuthController.html#logout/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.new_api_key/2","ref":"PlausibleWeb.AuthController.html#new_api_key/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.password_reset/2","ref":"PlausibleWeb.AuthController.html#password_reset/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.password_reset_form/2","ref":"PlausibleWeb.AuthController.html#password_reset_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.password_reset_request/2","ref":"PlausibleWeb.AuthController.html#password_reset_request/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.password_reset_request_form/2","ref":"PlausibleWeb.AuthController.html#password_reset_request_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.request_activation_code/2","ref":"PlausibleWeb.AuthController.html#request_activation_code/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.save_settings/2","ref":"PlausibleWeb.AuthController.html#save_settings/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.update_email/2","ref":"PlausibleWeb.AuthController.html#update_email/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.user_settings/2","ref":"PlausibleWeb.AuthController.html#user_settings/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.verify_2fa/2","ref":"PlausibleWeb.AuthController.html#verify_2fa/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.verify_2fa_form/2","ref":"PlausibleWeb.AuthController.html#verify_2fa_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.verify_2fa_recovery_code/2","ref":"PlausibleWeb.AuthController.html#verify_2fa_recovery_code/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.verify_2fa_recovery_code_form/2","ref":"PlausibleWeb.AuthController.html#verify_2fa_recovery_code_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.verify_2fa_setup/2","ref":"PlausibleWeb.AuthController.html#verify_2fa_setup/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthController.verify_2fa_setup_form/2","ref":"PlausibleWeb.AuthController.html#verify_2fa_setup_form/2"},{"type":"module","doc":"Plug for populating conn assigns with user data\non the basis of authenticated session token.\n\nMust be kept in sync with `PlausibleWeb.Live.AuthContext`.","title":"PlausibleWeb.AuthPlug","ref":"PlausibleWeb.AuthPlug.html"},{"type":"function","doc":"","title":"PlausibleWeb.AuthPlug.call/2","ref":"PlausibleWeb.AuthPlug.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthPlug.init/1","ref":"PlausibleWeb.AuthPlug.html#init/1"},{"type":"module","doc":"","title":"PlausibleWeb.AuthView","ref":"PlausibleWeb.AuthView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.AuthView.__resource__/0","ref":"PlausibleWeb.AuthView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.activate.html/1","ref":"PlausibleWeb.AuthView.html#activate.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.delimit_integer/1","ref":"PlausibleWeb.AuthView.html#delimit_integer/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.format_invoices/1","ref":"PlausibleWeb.AuthView.html#format_invoices/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.generate_2fa_recovery_codes.html/1","ref":"PlausibleWeb.AuthView.html#generate_2fa_recovery_codes.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.initiate_2fa_setup.html/1","ref":"PlausibleWeb.AuthView.html#initiate_2fa_setup.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.login_form.html/1","ref":"PlausibleWeb.AuthView.html#login_form.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.new_api_key.html/1","ref":"PlausibleWeb.AuthView.html#new_api_key.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.password_reset_form.html/1","ref":"PlausibleWeb.AuthView.html#password_reset_form.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.password_reset_request_form.html/1","ref":"PlausibleWeb.AuthView.html#password_reset_request_form.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.password_reset_request_success.html/1","ref":"PlausibleWeb.AuthView.html#password_reset_request_success.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.present_subscription_status/1","ref":"PlausibleWeb.AuthView.html#present_subscription_status/1"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.AuthView.render/2","ref":"PlausibleWeb.AuthView.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.subscription_colors/1","ref":"PlausibleWeb.AuthView.html#subscription_colors/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.subscription_interval/1","ref":"PlausibleWeb.AuthView.html#subscription_interval/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.subscription_quota/2","ref":"PlausibleWeb.AuthView.html#subscription_quota/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.AuthView.template_not_found/2","ref":"PlausibleWeb.AuthView.html#template_not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.user_settings.html/1","ref":"PlausibleWeb.AuthView.html#user_settings.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.verify_2fa.html/1","ref":"PlausibleWeb.AuthView.html#verify_2fa.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.verify_2fa_recovery_code.html/1","ref":"PlausibleWeb.AuthView.html#verify_2fa_recovery_code.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.AuthView.verify_2fa_setup.html/1","ref":"PlausibleWeb.AuthView.html#verify_2fa_setup.html/1"},{"type":"module","doc":"","title":"PlausibleWeb.AuthorizeSiteAccess","ref":"PlausibleWeb.AuthorizeSiteAccess.html"},{"type":"function","doc":"","title":"PlausibleWeb.AuthorizeSiteAccess.call/2","ref":"PlausibleWeb.AuthorizeSiteAccess.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.AuthorizeSiteAccess.init/1","ref":"PlausibleWeb.AuthorizeSiteAccess.html#init/1"},{"type":"module","doc":"This module proxies requests to BASE_URL/avatar/:hash to www.gravatar.com/avatar/:hash.\n\nThe purpose is to make use of Gravatar's convenient avatar service without exposing information\nthat could be used for tracking the Plausible user. Compared to requesting the Gravatar directly\nfrom the browser, this proxy module protects the Plausible user from disclosing to Gravatar:\n1. The client IP address\n2. User-Agent\n3. Referer header which can be used to track which site the user is visiting (i.e. plausible.io or self-hosted URL)\n\nThe downside is the added latency from the request having to go through the Plausible server, rather than contacting the\nlocal CDN server operated by Gravatar's service.","title":"PlausibleWeb.AvatarController","ref":"PlausibleWeb.AvatarController.html"},{"type":"function","doc":"","title":"PlausibleWeb.AvatarController.avatar/2","ref":"PlausibleWeb.AvatarController.html#avatar/2"},{"type":"module","doc":"","title":"PlausibleWeb.BillingController","ref":"PlausibleWeb.BillingController.html"},{"type":"function","doc":"","title":"PlausibleWeb.BillingController.change_plan/2","ref":"PlausibleWeb.BillingController.html#change_plan/2"},{"type":"function","doc":"","title":"PlausibleWeb.BillingController.change_plan_preview/2","ref":"PlausibleWeb.BillingController.html#change_plan_preview/2"},{"type":"function","doc":"","title":"PlausibleWeb.BillingController.choose_plan/2","ref":"PlausibleWeb.BillingController.html#choose_plan/2"},{"type":"function","doc":"","title":"PlausibleWeb.BillingController.ping_subscription/2","ref":"PlausibleWeb.BillingController.html#ping_subscription/2"},{"type":"function","doc":"","title":"PlausibleWeb.BillingController.upgrade_success/2","ref":"PlausibleWeb.BillingController.html#upgrade_success/2"},{"type":"function","doc":"","title":"PlausibleWeb.BillingController.upgrade_to_enterprise_plan/2","ref":"PlausibleWeb.BillingController.html#upgrade_to_enterprise_plan/2"},{"type":"module","doc":"","title":"PlausibleWeb.BillingView","ref":"PlausibleWeb.BillingView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.BillingView.__resource__/0","ref":"PlausibleWeb.BillingView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.change_enterprise_plan_contact_us.html/1","ref":"PlausibleWeb.BillingView.html#change_enterprise_plan_contact_us.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.change_plan_preview.html/1","ref":"PlausibleWeb.BillingView.html#change_plan_preview.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.choose_plan.html/1","ref":"PlausibleWeb.BillingView.html#choose_plan.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.present_currency/1","ref":"PlausibleWeb.BillingView.html#present_currency/1"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.present_date/1","ref":"PlausibleWeb.BillingView.html#present_date/1"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.BillingView.render/2","ref":"PlausibleWeb.BillingView.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.BillingView.template_not_found/2","ref":"PlausibleWeb.BillingView.html#template_not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.upgrade_success.html/1","ref":"PlausibleWeb.BillingView.html#upgrade_success.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.BillingView.upgrade_to_enterprise_plan.html/1","ref":"PlausibleWeb.BillingView.html#upgrade_to_enterprise_plan.html/1"},{"type":"module","doc":"","title":"PlausibleWeb.Captcha","ref":"PlausibleWeb.Captcha.html"},{"type":"function","doc":"","title":"PlausibleWeb.Captcha.enabled?/0","ref":"PlausibleWeb.Captcha.html#enabled?/0"},{"type":"function","doc":"","title":"PlausibleWeb.Captcha.sitekey/0","ref":"PlausibleWeb.Captcha.html#sitekey/0"},{"type":"function","doc":"","title":"PlausibleWeb.Captcha.verify/1","ref":"PlausibleWeb.Captcha.html#verify/1"},{"type":"module","doc":"This module exposes functions for rendering and returning plan\nbenefits for Growth, Business, and Enterprise plans.","title":"PlausibleWeb.Components.Billing.PlanBenefits","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html"},{"type":"function","doc":"Returns Business benefits for the given Business plan.\n\nA second argument is also required - list of Growth benefits. This\nis because we don't want to list the same benefits in both Growth\nand Business. Everything in Growth is also included in Business.","title":"PlausibleWeb.Components.Billing.PlanBenefits.for_business/2","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#for_business/2"},{"type":"function","doc":"This function only takes a list of business benefits. Since all\nlimits and features of enterprise plans are configurable, we can\nsay on the upgrade page that enterprise plans include everything\nin Business.","title":"PlausibleWeb.Components.Billing.PlanBenefits.for_enterprise/1","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#for_enterprise/1"},{"type":"function","doc":"This function takes a growth plan and returns a list representing\nthe different benefits a user gets when subscribing to this plan.","title":"PlausibleWeb.Components.Billing.PlanBenefits.for_growth/1","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#for_growth/1"},{"type":"function","doc":"This function takes a list of benefits returned by either one of:\n\n* `for_growth/1`\n* `for_business/2`\n* `for_enterprise/1`.\n\nand renders them as HTML.\n\nThe benefits in the given list can be either strings or functions\nreturning a Phoenix component. This allows, for example, to render\nlinks within the plan benefit text.","title":"PlausibleWeb.Components.Billing.PlanBenefits.render/1","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#render/1"},{"type":"function","doc":"* `benefits` (`:list`) (required)\n* `class` (`:string`) - Defaults to `nil`.","title":"Attributes - PlausibleWeb.Components.Billing.PlanBenefits.render/1","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#render/1-attributes"},{"type":"module","doc":"A banner that appears on the first dashboard launch","title":"PlausibleWeb.Components.FirstDashboardLaunchBanner","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html"},{"type":"function","doc":"","title":"PlausibleWeb.Components.FirstDashboardLaunchBanner.render/1","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#render/1"},{"type":"function","doc":"* `site` (`Plausible.Site`) (required)","title":"Attributes - PlausibleWeb.Components.FirstDashboardLaunchBanner.render/1","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#render/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Components.FirstDashboardLaunchBanner.set/1","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#set/1"},{"type":"function","doc":"* `site` (`Plausible.Site`) (required)","title":"Attributes - PlausibleWeb.Components.FirstDashboardLaunchBanner.set/1","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#set/1-attributes"},{"type":"module","doc":"Component for provisioning/registration flows displaying\nprogress status. See `PlausibleWeb.Flows` for the list of\nflow definitions.","title":"PlausibleWeb.Components.FlowProgress","ref":"PlausibleWeb.Components.FlowProgress.html"},{"type":"function","doc":"","title":"PlausibleWeb.Components.FlowProgress.render/1","ref":"PlausibleWeb.Components.FlowProgress.html#render/1"},{"type":"function","doc":"* `flow` (`:string`) (required) - Must be one of `\"register\"`, `\"invitation\"`, `\"domain_change\"`, `\"review\"`, or `\"provisioning\"`.\n* `current_step` (`:string`) (required) - Must be one of `\"Register\"`, `\"Activate account\"`, `\"Add site info\"`, `\"Install Plausible\"`, `\"Verify installation\"`, or `\"Set up new domain\"`.","title":"Attributes - PlausibleWeb.Components.FlowProgress.render/1","ref":"PlausibleWeb.Components.FlowProgress.html#render/1-attributes"},{"type":"module","doc":"Generic reusable components","title":"PlausibleWeb.Components.Generic","ref":"PlausibleWeb.Components.Generic.html"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.button/1","ref":"PlausibleWeb.Components.Generic.html#button/1"},{"type":"function","doc":"* `type` (`:string`) - Defaults to `\"button\"`.\n* `theme` (`:string`) - Defaults to `\"primary\"`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* `disabled` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.button/1","ref":"PlausibleWeb.Components.Generic.html#button/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Components.Generic.button/1","ref":"PlausibleWeb.Components.Generic.html#button/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.button_link/1","ref":"PlausibleWeb.Components.Generic.html#button_link/1"},{"type":"function","doc":"* `href` (`:string`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.\n* `theme` (`:string`) - Defaults to `\"primary\"`.\n* `disabled` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.button_link/1","ref":"PlausibleWeb.Components.Generic.html#button_link/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Components.Generic.button_link/1","ref":"PlausibleWeb.Components.Generic.html#button_link/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.docs_info/1","ref":"PlausibleWeb.Components.Generic.html#docs_info/1"},{"type":"function","doc":"* `slug` (`:string`) (required)","title":"Attributes - PlausibleWeb.Components.Generic.docs_info/1","ref":"PlausibleWeb.Components.Generic.html#docs_info/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.dropdown/1","ref":"PlausibleWeb.Components.Generic.html#dropdown/1"},{"type":"function","doc":"* `button` (required) - Accepts attributes:\n\n * `class` (`:string`)\n* `panel` (required) - Accepts attributes:\n\n * `class` (`:string`)","title":"Slots - PlausibleWeb.Components.Generic.dropdown/1","ref":"PlausibleWeb.Components.Generic.html#dropdown/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.dropdown_link/1","ref":"PlausibleWeb.Components.Generic.html#dropdown_link/1"},{"type":"function","doc":"* `href` (`:string`) (required)\n* `new_tab` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.dropdown_link/1","ref":"PlausibleWeb.Components.Generic.html#dropdown_link/1-attributes"},{"type":"function","doc":"* `inner_block` (required)","title":"Slots - PlausibleWeb.Components.Generic.dropdown_link/1","ref":"PlausibleWeb.Components.Generic.html#dropdown_link/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.dynamic_icon/1","ref":"PlausibleWeb.Components.Generic.html#dynamic_icon/1"},{"type":"function","doc":"* `name` (`:atom`) (required)\n* `outline` (`:boolean`) - Defaults to `true`.\n* `solid` (`:boolean`) - Defaults to `false`.\n* `mini` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted. Supports all globals plus: `[\"fill\", \"stroke\", \"stroke-width\"]`.","title":"Attributes - PlausibleWeb.Components.Generic.dynamic_icon/1","ref":"PlausibleWeb.Components.Generic.html#dynamic_icon/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.focus_box/1","ref":"PlausibleWeb.Components.Generic.html#focus_box/1"},{"type":"function","doc":"* `title`\n* `subtitle`\n* `inner_block` (required)\n* `footer`","title":"Slots - PlausibleWeb.Components.Generic.focus_box/1","ref":"PlausibleWeb.Components.Generic.html#focus_box/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.focus_list/1","ref":"PlausibleWeb.Components.Generic.html#focus_list/1"},{"type":"function","doc":"* `item` (required)","title":"Slots - PlausibleWeb.Components.Generic.focus_list/1","ref":"PlausibleWeb.Components.Generic.html#focus_list/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.notice/1","ref":"PlausibleWeb.Components.Generic.html#notice/1"},{"type":"function","doc":"* `title` (`:any`) - Defaults to `nil`.\n* `size` (`:atom`) - Defaults to `:sm`.\n* `theme` (`:atom`) - Defaults to `:yellow`.\n* `dismissable_id` (`:any`) - Defaults to `nil`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.notice/1","ref":"PlausibleWeb.Components.Generic.html#notice/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Components.Generic.notice/1","ref":"PlausibleWeb.Components.Generic.html#notice/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.spinner/1","ref":"PlausibleWeb.Components.Generic.html#spinner/1"},{"type":"function","doc":"* `class` (`:any`) - Defaults to `\"\"`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.spinner/1","ref":"PlausibleWeb.Components.Generic.html#spinner/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.styled_link/1","ref":"PlausibleWeb.Components.Generic.html#styled_link/1"},{"type":"function","doc":"* `id` (`:any`) - Defaults to `nil`.\n* `href` (`:string`) - Defaults to `\"#\"`.\n* `new_tab` (`:boolean`) - Defaults to `false`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* `method` (`:string`) - Defaults to `\"get\"`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.styled_link/1","ref":"PlausibleWeb.Components.Generic.html#styled_link/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Components.Generic.styled_link/1","ref":"PlausibleWeb.Components.Generic.html#styled_link/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.tooltip/1","ref":"PlausibleWeb.Components.Generic.html#tooltip/1"},{"type":"function","doc":"* `wrapper_class` (`:any`) - Defaults to `\"\"`.\n* `class` (`:any`) - Defaults to `\"\"`.\n* `icon?` (`:boolean`) - Defaults to `true`.\n* `sticky?` (`:boolean`) - Defaults to `true`.\n* `position` (`:string`) - Defaults to `\"bottom-10 margin-x-auto left-10 right-10\"`.","title":"Attributes - PlausibleWeb.Components.Generic.tooltip/1","ref":"PlausibleWeb.Components.Generic.html#tooltip/1-attributes"},{"type":"function","doc":"* `inner_block` (required)\n* `tooltip_content` (required)","title":"Slots - PlausibleWeb.Components.Generic.tooltip/1","ref":"PlausibleWeb.Components.Generic.html#tooltip/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Generic.unstyled_link/1","ref":"PlausibleWeb.Components.Generic.html#unstyled_link/1"},{"type":"function","doc":"* `href` (`:string`) (required)\n* `new_tab` (`:boolean`) - Defaults to `false`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* `id` (`:any`) - Defaults to `nil`.\n* `method` (`:string`) - Defaults to `\"get\"`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Components.Generic.unstyled_link/1","ref":"PlausibleWeb.Components.Generic.html#unstyled_link/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Components.Generic.unstyled_link/1","ref":"PlausibleWeb.Components.Generic.html#unstyled_link/1-slots"},{"type":"module","doc":"Google-related components","title":"PlausibleWeb.Components.Google","ref":"PlausibleWeb.Components.Google.html"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Google.button/1","ref":"PlausibleWeb.Components.Google.html#button/1"},{"type":"function","doc":"* `to` (`:string`) (required)\n* `id` (`:string`) (required)","title":"Attributes - PlausibleWeb.Components.Google.button/1","ref":"PlausibleWeb.Components.Google.html#button/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Google.logo/1","ref":"PlausibleWeb.Components.Google.html#logo/1"},{"type":"module","doc":"An umbrella module for the Integrations settings section","title":"PlausibleWeb.Components.Settings","ref":"PlausibleWeb.Components.Settings.html"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Settings.settings_search_console/1","ref":"PlausibleWeb.Components.Settings.html#settings_search_console/1"},{"type":"module","doc":"Phoenix Component for rendering a user-facing feature toggle\ncapable of flipping booleans in `Plausible.Site` via the `toggle_feature` controller action.","title":"PlausibleWeb.Components.Site.Feature","ref":"PlausibleWeb.Components.Site.Feature.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Components.Site.Feature.__resource__/0","ref":"PlausibleWeb.Components.Site.Feature.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Components.Site.Feature.render/2","ref":"PlausibleWeb.Components.Site.Feature.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Site.Feature.target/4","ref":"PlausibleWeb.Components.Site.Feature.html#target/4"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Components.Site.Feature.template_not_found/2","ref":"PlausibleWeb.Components.Site.Feature.html#template_not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.Components.Site.Feature.toggle/1","ref":"PlausibleWeb.Components.Site.Feature.html#toggle/1"},{"type":"function","doc":"* `site` (`Plausible.Site`) (required)\n* `feature_mod` (`:atom`) (required) - Must be one of `Plausible.Billing.Feature.Goals`, `Plausible.Billing.Feature.StatsAPI`, `Plausible.Billing.Feature.Props`, `Plausible.Billing.Feature.Funnels`, or `Plausible.Billing.Feature.RevenueGoals`.\n* `conn` (`Plug.Conn`) (required)","title":"Attributes - PlausibleWeb.Components.Site.Feature.toggle/1","ref":"PlausibleWeb.Components.Site.Feature.html#toggle/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Components.Site.Feature.toggle/1","ref":"PlausibleWeb.Components.Site.Feature.html#toggle/1-slots"},{"type":"module","doc":"Reusable components specific to 2FA","title":"PlausibleWeb.Components.TwoFactor","ref":"PlausibleWeb.Components.TwoFactor.html"},{"type":"function","doc":"","title":"PlausibleWeb.Components.TwoFactor.modal/1","ref":"PlausibleWeb.Components.TwoFactor.html#modal/1"},{"type":"function","doc":"* `id` (`:string`) (required)\n* `state_param` (`:string`) (required)\n* `form_data` (`:any`) (required)\n* `form_target` (`:string`) (required)\n* `onsubmit` (`:string`) - Defaults to `nil`.\n* `title` (`:string`) (required)","title":"Attributes - PlausibleWeb.Components.TwoFactor.modal/1","ref":"PlausibleWeb.Components.TwoFactor.html#modal/1-attributes"},{"type":"function","doc":"* `icon` (required)\n* `inner_block` (required)\n* `buttons` (required)","title":"Slots - PlausibleWeb.Components.TwoFactor.modal/1","ref":"PlausibleWeb.Components.TwoFactor.html#modal/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Components.TwoFactor.qr_code/1","ref":"PlausibleWeb.Components.TwoFactor.html#qr_code/1"},{"type":"function","doc":"* `text` (`:string`) (required)\n* `scale` (`:integer`) - Defaults to `4`.","title":"Attributes - PlausibleWeb.Components.TwoFactor.qr_code/1","ref":"PlausibleWeb.Components.TwoFactor.html#qr_code/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Components.TwoFactor.verify_2fa_input/1","ref":"PlausibleWeb.Components.TwoFactor.html#verify_2fa_input/1"},{"type":"function","doc":"* `id` (`:string`) - Defaults to `\"verify-button\"`.\n* `form` (`:any`) (required)\n* `field` (`:any`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.","title":"Attributes - PlausibleWeb.Components.TwoFactor.verify_2fa_input/1","ref":"PlausibleWeb.Components.TwoFactor.html#verify_2fa_input/1-attributes"},{"type":"module","doc":"This module defines the test case to be used by\ntests that require setting up a connection.\n\nSuch tests rely on `Phoenix.ConnTest` and also\nimport other functionality to make it easier\nto build common data structures and query the data layer.\n\nFinally, if the test case interacts with the database,\nit cannot be async. For this reason, every test runs\ninside a transaction which is reset at the beginning\nof the test unless the test case is marked as async.","title":"PlausibleWeb.ConnCase","ref":"PlausibleWeb.ConnCase.html"},{"type":"module","doc":"","title":"PlausibleWeb.ControllerHelpers","ref":"PlausibleWeb.ControllerHelpers.html"},{"type":"function","doc":"","title":"PlausibleWeb.ControllerHelpers.debug_metadata/1","ref":"PlausibleWeb.ControllerHelpers.html#debug_metadata/1"},{"type":"function","doc":"","title":"PlausibleWeb.ControllerHelpers.render_error/2","ref":"PlausibleWeb.ControllerHelpers.html#render_error/2"},{"type":"function","doc":"","title":"PlausibleWeb.ControllerHelpers.render_error/3","ref":"PlausibleWeb.ControllerHelpers.html#render_error/3"},{"type":"module","doc":"Revenue specific functions for the API scope","title":"PlausibleWeb.Controllers.API.Revenue","ref":"PlausibleWeb.Controllers.API.Revenue.html"},{"type":"function","doc":"","title":"PlausibleWeb.Controllers.API.Revenue.format_money/1","ref":"PlausibleWeb.Controllers.API.Revenue.html#format_money/1"},{"type":"function","doc":"","title":"PlausibleWeb.Controllers.API.Revenue.format_revenue_metric/1","ref":"PlausibleWeb.Controllers.API.Revenue.html#format_revenue_metric/1"},{"type":"module","doc":"","title":"PlausibleWeb.DebugController","ref":"PlausibleWeb.DebugController.html"},{"type":"function","doc":"","title":"PlausibleWeb.DebugController.clickhouse/2","ref":"PlausibleWeb.DebugController.html#clickhouse/2"},{"type":"module","doc":"","title":"PlausibleWeb.DebugView","ref":"PlausibleWeb.DebugView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.DebugView.__resource__/0","ref":"PlausibleWeb.DebugView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.DebugView.clickhouse.html/1","ref":"PlausibleWeb.DebugView.html#clickhouse.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.DebugView.controller_name/1","ref":"PlausibleWeb.DebugView.html#controller_name/1"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.DebugView.render/2","ref":"PlausibleWeb.DebugView.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.DebugView.template_not_found/2","ref":"PlausibleWeb.DebugView.html#template_not_found/2"},{"type":"module","doc":"Plausible tracking itself functions","title":"PlausibleWeb.Dogfood","ref":"PlausibleWeb.Dogfood.html"},{"type":"function","doc":"Temporary override to do more testing of the new ingest.plausible.io endpoint for accepting events. In staging and locally\nwill fall back to staging.plausible.io/api/event and localhost:8000/api/event respectively.","title":"PlausibleWeb.Dogfood.api_destination/0","ref":"PlausibleWeb.Dogfood.html#api_destination/0"},{"type":"function","doc":"","title":"PlausibleWeb.Dogfood.domain/1","ref":"PlausibleWeb.Dogfood.html#domain/1"},{"type":"function","doc":"","title":"PlausibleWeb.Dogfood.script_url/0","ref":"PlausibleWeb.Dogfood.html#script_url/0"},{"type":"module","doc":"","title":"PlausibleWeb.Email","ref":"PlausibleWeb.Email.html"},{"type":"function","doc":"","title":"PlausibleWeb.Email.activation_email/2","ref":"PlausibleWeb.Email.html#activation_email/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.approaching_accept_traffic_until/1","ref":"PlausibleWeb.Email.html#approaching_accept_traffic_until/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.approaching_accept_traffic_until_tomorrow/1","ref":"PlausibleWeb.Email.html#approaching_accept_traffic_until_tomorrow/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.base_email/0","ref":"PlausibleWeb.Email.html#base_email/0"},{"type":"function","doc":"","title":"PlausibleWeb.Email.base_email/1","ref":"PlausibleWeb.Email.html#base_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.cancellation_email/1","ref":"PlausibleWeb.Email.html#cancellation_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.check_stats_email/1","ref":"PlausibleWeb.Email.html#check_stats_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.create_site_email/1","ref":"PlausibleWeb.Email.html#create_site_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.dashboard_locked/3","ref":"PlausibleWeb.Email.html#dashboard_locked/3"},{"type":"function","doc":"","title":"PlausibleWeb.Email.drop_notification/5","ref":"PlausibleWeb.Email.html#drop_notification/5"},{"type":"function","doc":"","title":"PlausibleWeb.Email.enterprise_over_limit_internal_email/4","ref":"PlausibleWeb.Email.html#enterprise_over_limit_internal_email/4"},{"type":"function","doc":"","title":"PlausibleWeb.Email.error_report/3","ref":"PlausibleWeb.Email.html#error_report/3"},{"type":"function","doc":"","title":"PlausibleWeb.Email.existing_user_invitation/1","ref":"PlausibleWeb.Email.html#existing_user_invitation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.export_failure/2","ref":"PlausibleWeb.Email.html#export_failure/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.export_success/3","ref":"PlausibleWeb.Email.html#export_success/3"},{"type":"function","doc":"","title":"PlausibleWeb.Email.import_failure/2","ref":"PlausibleWeb.Email.html#import_failure/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.import_success/2","ref":"PlausibleWeb.Email.html#import_success/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.invitation_accepted/1","ref":"PlausibleWeb.Email.html#invitation_accepted/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.invitation_rejected/1","ref":"PlausibleWeb.Email.html#invitation_rejected/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.mailer_email_from/0","ref":"PlausibleWeb.Email.html#mailer_email_from/0"},{"type":"function","doc":"","title":"PlausibleWeb.Email.new_user_invitation/1","ref":"PlausibleWeb.Email.html#new_user_invitation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.over_limit_email/3","ref":"PlausibleWeb.Email.html#over_limit_email/3"},{"type":"function","doc":"","title":"PlausibleWeb.Email.ownership_transfer_accepted/1","ref":"PlausibleWeb.Email.html#ownership_transfer_accepted/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.ownership_transfer_rejected/1","ref":"PlausibleWeb.Email.html#ownership_transfer_rejected/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.ownership_transfer_request/2","ref":"PlausibleWeb.Email.html#ownership_transfer_request/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.password_reset_email/2","ref":"PlausibleWeb.Email.html#password_reset_email/2"},{"type":"function","doc":"Unlike the default 'base' emails, priority emails cannot be unsubscribed from. This is achieved\n by sending them through a dedicated 'priority' message stream in Postmark.","title":"PlausibleWeb.Email.priority_email/0","ref":"PlausibleWeb.Email.html#priority_email/0"},{"type":"function","doc":"","title":"PlausibleWeb.Email.priority_email/1","ref":"PlausibleWeb.Email.html#priority_email/1"},{"type":"function","doc":"Render an Phoenix template and set the body on the email.\n\nPass an atom as the template name (:welcome_email) to render HTML *and* plain\ntext emails. Use a string if you only want to render one type, e.g.\n\"welcome_email.text\" or \"welcome_email.html\". Scroll to the top for more examples.","title":"PlausibleWeb.Email.render/3","ref":"PlausibleWeb.Email.html#render/3"},{"type":"function","doc":"","title":"PlausibleWeb.Email.site_member_removed/1","ref":"PlausibleWeb.Email.html#site_member_removed/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.site_setup_help/2","ref":"PlausibleWeb.Email.html#site_setup_help/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.site_setup_success/2","ref":"PlausibleWeb.Email.html#site_setup_success/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.spike_notification/5","ref":"PlausibleWeb.Email.html#spike_notification/5"},{"type":"function","doc":"","title":"PlausibleWeb.Email.stats_report/2","ref":"PlausibleWeb.Email.html#stats_report/2"},{"type":"function","doc":"","title":"PlausibleWeb.Email.trial_one_week_reminder/1","ref":"PlausibleWeb.Email.html#trial_one_week_reminder/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.trial_over_email/1","ref":"PlausibleWeb.Email.html#trial_over_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.trial_upgrade_email/3","ref":"PlausibleWeb.Email.html#trial_upgrade_email/3"},{"type":"function","doc":"","title":"PlausibleWeb.Email.two_factor_disabled_email/1","ref":"PlausibleWeb.Email.html#two_factor_disabled_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.two_factor_enabled_email/1","ref":"PlausibleWeb.Email.html#two_factor_enabled_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.welcome_email/1","ref":"PlausibleWeb.Email.html#welcome_email/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.yearly_expiration_notification/1","ref":"PlausibleWeb.Email.html#yearly_expiration_notification/1"},{"type":"function","doc":"","title":"PlausibleWeb.Email.yearly_renewal_notification/1","ref":"PlausibleWeb.Email.html#yearly_renewal_notification/1"},{"type":"module","doc":"","title":"PlausibleWeb.EmailView","ref":"PlausibleWeb.EmailView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.EmailView.__resource__/0","ref":"PlausibleWeb.EmailView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.activation_email.html/1","ref":"PlausibleWeb.EmailView.html#activation_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.approaching_accept_traffic_until.html/1","ref":"PlausibleWeb.EmailView.html#approaching_accept_traffic_until.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.cancellation_email.html/1","ref":"PlausibleWeb.EmailView.html#cancellation_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.check_stats_email.html/1","ref":"PlausibleWeb.EmailView.html#check_stats_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.create_site_email.html/1","ref":"PlausibleWeb.EmailView.html#create_site_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.csv_import.html/1","ref":"PlausibleWeb.EmailView.html#csv_import.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.dashboard_locked.html/1","ref":"PlausibleWeb.EmailView.html#dashboard_locked.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.date_format/1","ref":"PlausibleWeb.EmailView.html#date_format/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.drop_notification.html/1","ref":"PlausibleWeb.EmailView.html#drop_notification.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.enterprise_over_limit_internal.html/1","ref":"PlausibleWeb.EmailView.html#enterprise_over_limit_internal.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.error_report_email.html/1","ref":"PlausibleWeb.EmailView.html#error_report_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.existing_user_invitation.html/1","ref":"PlausibleWeb.EmailView.html#existing_user_invitation.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.export_failure.html/1","ref":"PlausibleWeb.EmailView.html#export_failure.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.export_success.html/1","ref":"PlausibleWeb.EmailView.html#export_success.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.google_analytics_import.html/1","ref":"PlausibleWeb.EmailView.html#google_analytics_import.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.greet_recipient/1","ref":"PlausibleWeb.EmailView.html#greet_recipient/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.invitation_accepted.html/1","ref":"PlausibleWeb.EmailView.html#invitation_accepted.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.invitation_rejected.html/1","ref":"PlausibleWeb.EmailView.html#invitation_rejected.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.new_user_invitation.html/1","ref":"PlausibleWeb.EmailView.html#new_user_invitation.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.over_limit.html/1","ref":"PlausibleWeb.EmailView.html#over_limit.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.ownership_transfer_accepted.html/1","ref":"PlausibleWeb.EmailView.html#ownership_transfer_accepted.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.ownership_transfer_rejected.html/1","ref":"PlausibleWeb.EmailView.html#ownership_transfer_rejected.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.ownership_transfer_request.html/1","ref":"PlausibleWeb.EmailView.html#ownership_transfer_request.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.password_reset_email.html/1","ref":"PlausibleWeb.EmailView.html#password_reset_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.plausible_url/0","ref":"PlausibleWeb.EmailView.html#plausible_url/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.EmailView.render/2","ref":"PlausibleWeb.EmailView.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.sentry_link/2","ref":"PlausibleWeb.EmailView.html#sentry_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.site_member_removed.html/1","ref":"PlausibleWeb.EmailView.html#site_member_removed.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.site_setup_help_email.html/1","ref":"PlausibleWeb.EmailView.html#site_setup_help_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.site_setup_success_email.html/1","ref":"PlausibleWeb.EmailView.html#site_setup_success_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.spike_notification.html/1","ref":"PlausibleWeb.EmailView.html#spike_notification.html/1"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.EmailView.template_not_found/2","ref":"PlausibleWeb.EmailView.html#template_not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.trial_one_week_reminder.html/1","ref":"PlausibleWeb.EmailView.html#trial_one_week_reminder.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.trial_over_email.html/1","ref":"PlausibleWeb.EmailView.html#trial_over_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.trial_upgrade_email.html/1","ref":"PlausibleWeb.EmailView.html#trial_upgrade_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.two_factor_disabled_email.html/1","ref":"PlausibleWeb.EmailView.html#two_factor_disabled_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.two_factor_enabled_email.html/1","ref":"PlausibleWeb.EmailView.html#two_factor_enabled_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.welcome_email.html/1","ref":"PlausibleWeb.EmailView.html#welcome_email.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.yearly_expiration_notification.html/1","ref":"PlausibleWeb.EmailView.html#yearly_expiration_notification.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.EmailView.yearly_renewal_notification.html/1","ref":"PlausibleWeb.EmailView.html#yearly_renewal_notification.html/1"},{"type":"module","doc":"","title":"PlausibleWeb.Endpoint","ref":"PlausibleWeb.Endpoint.html"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.broadcast/3","ref":"PlausibleWeb.Endpoint.html#broadcast/3"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.broadcast!/3","ref":"PlausibleWeb.Endpoint.html#broadcast!/3"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.broadcast_from/4","ref":"PlausibleWeb.Endpoint.html#broadcast_from/4"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.broadcast_from!/4","ref":"PlausibleWeb.Endpoint.html#broadcast_from!/4"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.call/2","ref":"PlausibleWeb.Endpoint.html#call/2"},{"type":"function","doc":"Returns the child specification to start the endpoint\nunder a supervision tree.","title":"PlausibleWeb.Endpoint.child_spec/1","ref":"PlausibleWeb.Endpoint.html#child_spec/1"},{"type":"function","doc":"Returns the endpoint configuration for `key`\n\nReturns `default` if the key does not exist.","title":"PlausibleWeb.Endpoint.config/2","ref":"PlausibleWeb.Endpoint.html#config/2"},{"type":"function","doc":"Reloads the configuration given the application environment changes.","title":"PlausibleWeb.Endpoint.config_change/2","ref":"PlausibleWeb.Endpoint.html#config_change/2"},{"type":"function","doc":"Returns the host for the given endpoint.","title":"PlausibleWeb.Endpoint.host/0","ref":"PlausibleWeb.Endpoint.html#host/0"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.init/1","ref":"PlausibleWeb.Endpoint.html#init/1"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.local_broadcast/3","ref":"PlausibleWeb.Endpoint.html#local_broadcast/3"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.local_broadcast_from/4","ref":"PlausibleWeb.Endpoint.html#local_broadcast_from/4"},{"type":"function","doc":"Generates the path information when routing to this endpoint.","title":"PlausibleWeb.Endpoint.path/1","ref":"PlausibleWeb.Endpoint.html#path/1"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.runtime_session/2","ref":"PlausibleWeb.Endpoint.html#runtime_session/2"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.runtime_session_opts/0","ref":"PlausibleWeb.Endpoint.html#runtime_session_opts/0"},{"type":"function","doc":"Generates the script name.","title":"PlausibleWeb.Endpoint.script_name/0","ref":"PlausibleWeb.Endpoint.html#script_name/0"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.secure_cookie?/0","ref":"PlausibleWeb.Endpoint.html#secure_cookie?/0"},{"type":"function","doc":"Returns the address and port that the server is running on","title":"PlausibleWeb.Endpoint.server_info/1","ref":"PlausibleWeb.Endpoint.html#server_info/1"},{"type":"function","doc":"Starts the endpoint supervision tree.\n\nAll other options are merged into the endpoint configuration.","title":"PlausibleWeb.Endpoint.start_link/1","ref":"PlausibleWeb.Endpoint.html#start_link/1"},{"type":"function","doc":"Generates a base64-encoded cryptographic hash (sha512) to a static file\nin `priv/static`. Meant to be used for Subresource Integrity with CDNs.","title":"PlausibleWeb.Endpoint.static_integrity/1","ref":"PlausibleWeb.Endpoint.html#static_integrity/1"},{"type":"function","doc":"Returns a two item tuple with the first item being the `static_path`\nand the second item being the `static_integrity`.","title":"PlausibleWeb.Endpoint.static_lookup/1","ref":"PlausibleWeb.Endpoint.html#static_lookup/1"},{"type":"function","doc":"Generates a route to a static file in `priv/static`.","title":"PlausibleWeb.Endpoint.static_path/1","ref":"PlausibleWeb.Endpoint.html#static_path/1"},{"type":"function","doc":"Generates the static URL without any path information.\n\nIt uses the configuration under `:static_url` to generate\nsuch. It falls back to `:url` if `:static_url` is not set.","title":"PlausibleWeb.Endpoint.static_url/0","ref":"PlausibleWeb.Endpoint.html#static_url/0"},{"type":"function","doc":"Generates the endpoint base URL but as a `URI` struct.\n\nIt uses the configuration under `:url` to generate such.\nUseful for manipulating the URL data and passing it to\nURL helpers.","title":"PlausibleWeb.Endpoint.struct_url/0","ref":"PlausibleWeb.Endpoint.html#struct_url/0"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.subscribe/2","ref":"PlausibleWeb.Endpoint.html#subscribe/2"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.unsubscribe/1","ref":"PlausibleWeb.Endpoint.html#unsubscribe/1"},{"type":"function","doc":"Generates the endpoint base URL without any path information.\n\nIt uses the configuration under `:url` to generate such.","title":"PlausibleWeb.Endpoint.url/0","ref":"PlausibleWeb.Endpoint.html#url/0"},{"type":"function","doc":"","title":"PlausibleWeb.Endpoint.websocket_url/0","ref":"PlausibleWeb.Endpoint.html#websocket_url/0"},{"type":"module","doc":"","title":"PlausibleWeb.ErrorHelpers","ref":"PlausibleWeb.ErrorHelpers.html"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorHelpers.error_tag/2","ref":"PlausibleWeb.ErrorHelpers.html#error_tag/2"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorHelpers.translate_error/1","ref":"PlausibleWeb.ErrorHelpers.html#translate_error/1"},{"type":"module","doc":"","title":"PlausibleWeb.ErrorReportController","ref":"PlausibleWeb.ErrorReportController.html"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorReportController.submit_error_report/2","ref":"PlausibleWeb.ErrorReportController.html#submit_error_report/2"},{"type":"module","doc":"","title":"PlausibleWeb.ErrorView","ref":"PlausibleWeb.ErrorView.html"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorView.404_error.html/1","ref":"PlausibleWeb.ErrorView.html#404_error.html/1"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.ErrorView.__resource__/0","ref":"PlausibleWeb.ErrorView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorView.generic_error.html/1","ref":"PlausibleWeb.ErrorView.html#generic_error.html/1"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.ErrorView.render/2","ref":"PlausibleWeb.ErrorView.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorView.server_error.html/1","ref":"PlausibleWeb.ErrorView.html#server_error.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.ErrorView.server_error_report_thanks.html/1","ref":"PlausibleWeb.ErrorView.html#server_error_report_thanks.html/1"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.ErrorView.template_not_found/2","ref":"PlausibleWeb.ErrorView.html#template_not_found/2"},{"type":"module","doc":"A Plug that fetches favicon images from DuckDuckGo and returns them\nto the Plausible frontend.\n\nThe proxying is there so we can reduce the number of third-party domains that\nthe browser clients need to connect to. Our goal is to have 0 third-party domain\nconnections on the website for privacy reasons.\n\nThis module also maps between categorized sources and their respective URLs for favicons.\nWhat does that mean exactly? During ingestion we use `PlausibleWeb.RefInspector.parse/1` to\ncategorize our referrer sources like so:\n\ngoogle.com -> Google\ngoogle.co.uk -> Google\ngoogle.com.au -> Google\n\nSo when we show Google as a source in the dashboard, the request to this plug will come as:\nhttps://plausible/io/favicon/sources/Google\n\nNow, when we want to show a favicon for Google, we need to convert Google -> google.com or\nsome other hostname owned by Google:\nhttps://icons.duckduckgo.com/ip3/google.com.ico\n\nThe mapping from source category -> source hostname is stored in \"priv/referer_favicon_domains.json\" and\nmanaged by `Mix.Tasks.GenerateReferrerFavicons.run/1`","title":"PlausibleWeb.Favicon","ref":"PlausibleWeb.Favicon.html"},{"type":"function","doc":"Proxies HTTP request to DuckDuckGo favicon service. Swallows hop-by-hop HTTP\nheaders that should not be forwarded as defined in [RFC 2616](https://www.rfc-editor.org/rfc/rfc2616#section-13.5.1)","title":"PlausibleWeb.Favicon.call/2","ref":"PlausibleWeb.Favicon.html#call/2"},{"type":"function","doc":"Cases where we show a placeholder icon instead:\n\n1. In case of network error to DuckDuckGo\n2. In case of non-2xx status code from DuckDuckGo\n3. In case of broken image response body from DuckDuckGo\n\nI'm not sure why DDG sometimes returns a broken PNG image in their response\nbut we filter that out. When the icon request fails, we show a placeholder\nfavicon instead. The placeholder is an emoji from\n[https://favicon.io/emoji-favicons/](https://favicon.io/emoji-favicons/)\n\nDuckDuckGo favicon service has some issues with [SVG favicons](https://css-tricks.com/svg-favicons-and-all-the-fun-things-we-can-do-with-them/).\nFor some reason, they return them with `content-type=image/x-icon` whereas SVG\nicons should be returned with `content-type=image/svg+xml`. This Plug detects\nwhen the response body starts with `\n<.input name=\"my-input\" errors={[\"oh no!\"]} />","title":"Examples - PlausibleWeb.Live.Components.Form.input/1","ref":"PlausibleWeb.Live.Components.Form.html#input/1-examples"},{"type":"function","doc":"* `id` (`:any`) - Defaults to `nil`.\n* `name` (`:any`)\n* `label` (`:string`) - Defaults to `nil`.\n* `value` (`:any`)\n* `type` (`:string`) - Defaults to `\"text\"`.\n* `field` (`Phoenix.HTML.FormField`) - a form field struct retrieved from the form, for example: @form[:email].\n* `errors` (`:list`) - Defaults to `[]`.\n* `checked` (`:boolean`) - the checked flag for checkbox inputs.\n* `prompt` (`:string`) - the prompt for select inputs. Defaults to `nil`.\n* `options` (`:list`) - the options to pass to Phoenix.HTML.Form.options_for_select/2.\n* `multiple` (`:boolean`) - the multiple flag for select inputs. Defaults to `false`.\n* Global attributes are accepted. Supports all globals plus: `[\"accept\", \"autocomplete\", \"capture\", \"cols\", \"disabled\", \"form\", \"list\", \"max\", \"maxlength\", \"min\", \"minlength\", \"multiple\", \"pattern\", \"placeholder\", \"readonly\", \"required\", \"rows\", \"size\", \"step\"]`.","title":"Attributes - PlausibleWeb.Live.Components.Form.input/1","ref":"PlausibleWeb.Live.Components.Form.html#input/1-attributes"},{"type":"function","doc":"* `inner_block`","title":"Slots - PlausibleWeb.Live.Components.Form.input/1","ref":"PlausibleWeb.Live.Components.Form.html#input/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Form.input_with_clipboard/1","ref":"PlausibleWeb.Live.Components.Form.html#input_with_clipboard/1"},{"type":"function","doc":"* `id` (`:string`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.\n* `name` (`:string`) (required)\n* `label` (`:string`) (required)\n* `value` (`:string`) - Defaults to `\"\"`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Live.Components.Form.input_with_clipboard/1","ref":"PlausibleWeb.Live.Components.Form.html#input_with_clipboard/1-attributes"},{"type":"function","doc":"Renders a label.","title":"PlausibleWeb.Live.Components.Form.label/1","ref":"PlausibleWeb.Live.Components.Form.html#label/1"},{"type":"function","doc":"* `for` (`:string`) - Defaults to `nil`.","title":"Attributes - PlausibleWeb.Live.Components.Form.label/1","ref":"PlausibleWeb.Live.Components.Form.html#label/1-attributes"},{"type":"function","doc":"* `inner_block` (required)","title":"Slots - PlausibleWeb.Live.Components.Form.label/1","ref":"PlausibleWeb.Live.Components.Form.html#label/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Form.password_input_with_strength/1","ref":"PlausibleWeb.Live.Components.Form.html#password_input_with_strength/1"},{"type":"function","doc":"* `id` (`:any`) - Defaults to `nil`.\n* `label` (`:string`) - Defaults to `nil`.\n* `field` (`Phoenix.HTML.FormField`) (required) - a form field struct retrieved from the form, for example: @form[:password].\n* `strength` (`:any`)\n* Global attributes are accepted. Supports all globals plus: `[\"autocomplete\", \"disabled\", \"form\", \"maxlength\", \"minlength\", \"readonly\", \"required\", \"size\"]`.","title":"Attributes - PlausibleWeb.Live.Components.Form.password_input_with_strength/1","ref":"PlausibleWeb.Live.Components.Form.html#password_input_with_strength/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Form.password_length_hint/1","ref":"PlausibleWeb.Live.Components.Form.html#password_length_hint/1"},{"type":"function","doc":"* `minimum` (`:integer`) (required)\n* `class` (`:any`)\n* `ok_class` (`:any`)\n* `error_class` (`:any`)\n* `field` (`Phoenix.HTML.FormField`) (required) - a form field struct retrieved from the form, for example: @form[:password].","title":"Attributes - PlausibleWeb.Live.Components.Form.password_length_hint/1","ref":"PlausibleWeb.Live.Components.Form.html#password_length_hint/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Form.strength_meter/1","ref":"PlausibleWeb.Live.Components.Form.html#strength_meter/1"},{"type":"function","doc":"* `score` (`:integer`) - Defaults to `0`.\n* `warning` (`:string`) - Defaults to `\"\"`.\n* `suggestions` (`:list`) - Defaults to `[]`.","title":"Attributes - PlausibleWeb.Live.Components.Form.strength_meter/1","ref":"PlausibleWeb.Live.Components.Form.html#strength_meter/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Form.translate_error/1","ref":"PlausibleWeb.Live.Components.Form.html#translate_error/1"},{"type":"module","doc":"LiveView implementation of modal component.\n\nThis component is a general purpose modal implementation for LiveView\nwith emphasis on keeping nested components largely agnostic of the fact\nthat they are placed in a modal and maintaining good user experience\non connections with high latency.","title":"PlausibleWeb.Live.Components.Modal","ref":"PlausibleWeb.Live.Components.Modal.html"},{"type":"module","doc":"An example use case for a modal is embedding a form inside\nexisting live view which allows adding new entries of some kind:\n\n```\n<.live_component module={Modal} id=\"some-form-modal\" :let={modal_unique_id}>\n <.live_component\n module={SomeForm}\n id={\"some-form-#{modal_unique_id}\"}\n on_save_form={\n fn entry, socket ->\n send(self(), {:entry_added, entry})\n Modal.close(socket, \"some-form-modal\")\n end\n }\n />\n\n```\n\nThen somewhere in the same live view the modal is rendered in:\n\n```\n<.button x-data x-on:click={Modal.JS.open(\"goals-form-modal\")}>\n + Add Entry\n\n```","title":"Usage - PlausibleWeb.Live.Components.Modal","ref":"PlausibleWeb.Live.Components.Modal.html#module-usage"},{"type":"module","doc":"The component embedded inside the modal is always rendered when\nthe live view is mounted but is kept hidden until `Modal.JS.open`\nis called on it. On subsequent openings within the same session\nthe contents of the modal are completely remounted. This assures\nthat any stateful components inside the modal are reset to their\ninitial state. The modal component provides `modal_unique_id`\nas an argument to its inner block. Appending this ID to every\nlive components' ID nested inside the modal is important for\nconsistent state reset on every reopening. This also applies\nto live components nested inside live components embedded directly\nin the modal's inner block - then the unique ID should be also\npassed down as an attribute and appended accordingly. Appending can\nbe skipped if embedded component handles state reset explicitly\n(via, for instance, `phx-click-away` callback).\n\n`Modal` exposes a number of functions for managing window state:\n\n * `Modal.JS.preopen/1` - to preopen the modal on the frontend.\n Useful when the actual opening is done server-side with\n `Modal.open/2` - helps avoid lack of feedback to the end user\n when server-side state change before opening the modal is\n still in progress.\n * `Modal.JS.open/1` - to open the modal from the frontend. It's\n important to make sure the element triggering that call is\n wrapped in an Alpine UI component - or is an Alpine component\n itself - adding `x-data` attribute without any value is enough\n to ensure that.\n * `Modal.open/2` - to open the modal from the backend; usually\n called from `handle_event/2` of component wrapping the modal\n and providing the state. Should be used together with\n `Modal.JS.preopen/1` for optimal user experience.\n * `Modal.close/2` - to close the modal from the backend; usually\n done inside wrapped component's `handle_event/2`. The example\n quoted above shows one way to implement this, under that assumption\n that the component exposes a callback, like this:\n\n ```\n defmodule SomeForm do\n use Phoenix.LiveComponent\n\n def update(assigns, socket) do\n # ...\n\n {:ok, assign(socket, :on_save_form, assigns.on_save_form)}\n end\n\n #...\n\n def handle_event(\"save-form\", %{\"form\" => form}, socket) do\n case save_entry(form) do\n {:ok, entry} ->\n {:noreply, socket.assigns.on_save_form(entry, socket)}\n\n # error case handling ...\n end\n end\n end\n ```\n\n Using callback approach has an added benefit of making the\n component more flexible.","title":"Explanation - PlausibleWeb.Live.Components.Modal","ref":"PlausibleWeb.Live.Components.Modal.html#module-explanation"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Modal.close/2","ref":"PlausibleWeb.Live.Components.Modal.html#close/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Modal.open/2","ref":"PlausibleWeb.Live.Components.Modal.html#open/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Modal.render/1","ref":"PlausibleWeb.Live.Components.Modal.html#render/1"},{"type":"function","doc":"* `id` (`:any`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.\n* `preload?` (`:boolean`) - Defaults to `true`.","title":"Attributes - PlausibleWeb.Live.Components.Modal.render/1","ref":"PlausibleWeb.Live.Components.Modal.html#render/1-attributes"},{"type":"function","doc":"* `inner_block` (required)","title":"Slots - PlausibleWeb.Live.Components.Modal.render/1","ref":"PlausibleWeb.Live.Components.Modal.html#render/1-slots"},{"type":"module","doc":"Pagination components for LiveViews.","title":"PlausibleWeb.Live.Components.Pagination","ref":"PlausibleWeb.Live.Components.Pagination.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Pagination.pagination/1","ref":"PlausibleWeb.Live.Components.Pagination.html#pagination/1"},{"type":"module","doc":"This component is responsible for rendering the verification progress\nand diagnostics.","title":"PlausibleWeb.Live.Components.Verification","ref":"PlausibleWeb.Live.Components.Verification.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Verification.render/1","ref":"PlausibleWeb.Live.Components.Verification.html#render/1"},{"type":"function","doc":"* `domain` (`:string`) (required)\n* `message` (`:string`) - Defaults to `\"We're visiting your site to ensure that everything is working\"`.\n* `finished?` (`:boolean`) - Defaults to `false`.\n* `success?` (`:boolean`) - Defaults to `false`.\n* `interpretation` (`Plausible.Verification.Diagnostics.Result`) - Defaults to `nil`.\n* `attempts` (`:integer`) - Defaults to `0`.\n* `flow` (`:string`) - Defaults to `\"\"`.\n* `installation_type` (`:string`) - Defaults to `nil`.\n* `awaiting_first_pageview?` (`:boolean`) - Defaults to `false`.","title":"Attributes - PlausibleWeb.Live.Components.Verification.render/1","ref":"PlausibleWeb.Live.Components.Verification.html#render/1-attributes"},{"type":"module","doc":"Component rendering mini-graph of site's visitors over the last 24 hours.\n\nThe `gradient_defs` component should be rendered once before using `chart`\none or more times.\n\nAccepts input generated via `Plausible.Stats.Clickhouse.last_24h_visitors_hourly_intervals/2`.","title":"PlausibleWeb.Live.Components.Visitors","ref":"PlausibleWeb.Live.Components.Visitors.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Visitors.chart/1","ref":"PlausibleWeb.Live.Components.Visitors.html#chart/1"},{"type":"function","doc":"* `intervals` (`:list`) (required)\n* `height` (`:integer`) - Defaults to `50`.\n* `tick` (`:integer`) - Defaults to `20`.","title":"Attributes - PlausibleWeb.Live.Components.Visitors.chart/1","ref":"PlausibleWeb.Live.Components.Visitors.html#chart/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Components.Visitors.gradient_defs/1","ref":"PlausibleWeb.Live.Components.Visitors.html#gradient_defs/1"},{"type":"module","doc":"Flash component for LiveViews - works also when embedded within dead views","title":"PlausibleWeb.Live.Flash","ref":"PlausibleWeb.Live.Flash.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Flash.clear_flash_button/1","ref":"PlausibleWeb.Live.Flash.html#clear_flash_button/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Flash.flash/1","ref":"PlausibleWeb.Live.Flash.html#flash/1"},{"type":"function","doc":"* `key` (`:string`) - Defaults to `nil`.\n* `on_close` (`:any`) - Defaults to `\"lv:clear-flash\"`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Live.Flash.flash/1","ref":"PlausibleWeb.Live.Flash.html#flash/1-attributes"},{"type":"function","doc":"* `icon` (required)\n* `title`\n* `message` (required)","title":"Slots - PlausibleWeb.Live.Flash.flash/1","ref":"PlausibleWeb.Live.Flash.html#flash/1-slots"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Flash.flash_messages/1","ref":"PlausibleWeb.Live.Flash.html#flash_messages/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Flash.icon_error/1","ref":"PlausibleWeb.Live.Flash.html#icon_error/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Flash.icon_success/1","ref":"PlausibleWeb.Live.Flash.html#icon_success/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Flash.put_live_flash/3","ref":"PlausibleWeb.Live.Flash.html#put_live_flash/3"},{"type":"module","doc":"LiveView allowing listing, creating and deleting funnels.","title":"PlausibleWeb.Live.FunnelSettings","ref":"PlausibleWeb.Live.FunnelSettings.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.handle_event/3","ref":"PlausibleWeb.Live.FunnelSettings.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.handle_info/2","ref":"PlausibleWeb.Live.FunnelSettings.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.mount/3","ref":"PlausibleWeb.Live.FunnelSettings.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.render/1","ref":"PlausibleWeb.Live.FunnelSettings.html#render/1"},{"type":"module","doc":"Phoenix LiveComponent that renders a form used for setting up funnels.\nMakes use of dynamically placed `PlausibleWeb.Live.FunnelSettings.ComboBox` components\nto allow building searchable funnel definitions out of list of goals available.","title":"PlausibleWeb.Live.FunnelSettings.Form","ref":"PlausibleWeb.Live.FunnelSettings.Form.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.add_step_button/1","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#add_step_button/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.evaluation/1","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#evaluation/1"},{"type":"function","doc":"* `at` (`:integer`) (required)\n* `result` (`:map`) (required)","title":"Attributes - PlausibleWeb.Live.FunnelSettings.Form.evaluation/1","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#evaluation/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.handle_event/3","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.handle_info/2","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.mount/3","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.remove_step_button/1","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#remove_step_button/1"},{"type":"function","doc":"* `step_idx` (`:integer`) (required)","title":"Attributes - PlausibleWeb.Live.FunnelSettings.Form.remove_step_button/1","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#remove_step_button/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.Form.render/1","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#render/1"},{"type":"module","doc":"Phoenix LiveComponent module that renders a list of funnels with their names\nand the number of steps they have.\n\nEach funnel is displayed with a delete button, which triggers a confirmation\nmessage before deleting the funnel from the UI. If there are no funnels\nconfigured for the site, a message is displayed indicating so.","title":"PlausibleWeb.Live.FunnelSettings.List","ref":"PlausibleWeb.Live.FunnelSettings.List.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.FunnelSettings.List.render/1","ref":"PlausibleWeb.Live.FunnelSettings.List.html#render/1"},{"type":"module","doc":"LiveView allowing listing, creating and deleting goals.","title":"PlausibleWeb.Live.GoalSettings","ref":"PlausibleWeb.Live.GoalSettings.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.handle_event/3","ref":"PlausibleWeb.Live.GoalSettings.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.handle_info/2","ref":"PlausibleWeb.Live.GoalSettings.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.mount/3","ref":"PlausibleWeb.Live.GoalSettings.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.render/1","ref":"PlausibleWeb.Live.GoalSettings.html#render/1"},{"type":"module","doc":"Live view for the goal creation form","title":"PlausibleWeb.Live.GoalSettings.Form","ref":"PlausibleWeb.Live.GoalSettings.Form.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.create_form/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#create_form/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.custom_event_fields/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#custom_event_fields/1"},{"type":"function","doc":"* `f` (`Phoenix.HTML.Form`)\n* `site` (`Plausible.Site`)\n* `current_user` (`Plausible.Auth.User`)\n* `suffix` (`:string`)\n* `existing_goals` (`:list`)\n* `goal_options` (`:list`)\n* `goal` (`Plausible.Goal`) - Defaults to `nil`.\n* `has_access_to_revenue_goals?` (`:boolean`)\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Live.GoalSettings.Form.custom_event_fields/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#custom_event_fields/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.edit_form/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#edit_form/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.handle_event/3","ref":"PlausibleWeb.Live.GoalSettings.Form.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.pageview_fields/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#pageview_fields/1"},{"type":"function","doc":"* `f` (`Phoenix.HTML.Form`)\n* `site` (`Plausible.Site`)\n* `suffix` (`:string`)\n* `goal` (`Plausible.Goal`) - Defaults to `nil`.\n* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Live.GoalSettings.Form.pageview_fields/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#pageview_fields/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.pageviews_tab/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#pageviews_tab/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.render/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.revenue_goal_settings/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#revenue_goal_settings/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.suggest_event_names/3","ref":"PlausibleWeb.Live.GoalSettings.Form.html#suggest_event_names/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.suggest_page_paths/2","ref":"PlausibleWeb.Live.GoalSettings.Form.html#suggest_page_paths/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.tabs/1","ref":"PlausibleWeb.Live.GoalSettings.Form.html#tabs/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.Form.update/2","ref":"PlausibleWeb.Live.GoalSettings.Form.html#update/2"},{"type":"module","doc":"Phoenix LiveComponent module that renders a list of goals","title":"PlausibleWeb.Live.GoalSettings.List","ref":"PlausibleWeb.Live.GoalSettings.List.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.List.custom_event_description/1","ref":"PlausibleWeb.Live.GoalSettings.List.html#custom_event_description/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.List.goal_description/1","ref":"PlausibleWeb.Live.GoalSettings.List.html#goal_description/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.List.pageview_description/1","ref":"PlausibleWeb.Live.GoalSettings.List.html#pageview_description/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.GoalSettings.List.render/1","ref":"PlausibleWeb.Live.GoalSettings.List.html#render/1"},{"type":"function","doc":"* `goals` (`:list`) (required)\n* `domain` (`:string`) (required)\n* `filter_text` (`:string`)\n* `site` (`Plausible.Site`) (required)","title":"Attributes - PlausibleWeb.Live.GoalSettings.List.render/1","ref":"PlausibleWeb.Live.GoalSettings.List.html#render/1-attributes"},{"type":"module","doc":"LiveView allowing listing and deleting imports.","title":"PlausibleWeb.Live.ImportsExportsSettings","ref":"PlausibleWeb.Live.ImportsExportsSettings.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ImportsExportsSettings.handle_info/2","ref":"PlausibleWeb.Live.ImportsExportsSettings.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ImportsExportsSettings.mount/3","ref":"PlausibleWeb.Live.ImportsExportsSettings.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ImportsExportsSettings.render/1","ref":"PlausibleWeb.Live.ImportsExportsSettings.html#render/1"},{"type":"module","doc":"User assistance module around Plausible installation instructions/onboarding","title":"PlausibleWeb.Live.Installation","ref":"PlausibleWeb.Live.Installation.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.handle_event/3","ref":"PlausibleWeb.Live.Installation.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.handle_info/2","ref":"PlausibleWeb.Live.Installation.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.handle_params/3","ref":"PlausibleWeb.Live.Installation.html#handle_params/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.mount/3","ref":"PlausibleWeb.Live.Installation.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.render/1","ref":"PlausibleWeb.Live.Installation.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.render_snippet_404/0","ref":"PlausibleWeb.Live.Installation.html#render_snippet_404/0"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.render_snippet_404/1","ref":"PlausibleWeb.Live.Installation.html#render_snippet_404/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Installation.script_extension_params/0","ref":"PlausibleWeb.Live.Installation.html#script_extension_params/0"},{"type":"module","doc":"LiveView allowing listing, creating and revoking Plugins API tokens.","title":"PlausibleWeb.Live.Plugins.API.Settings","ref":"PlausibleWeb.Live.Plugins.API.Settings.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.Settings.handle_event/3","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.Settings.handle_info/2","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.Settings.mount/3","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.Settings.render/1","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#render/1"},{"type":"module","doc":"Live view for the goal creation form","title":"PlausibleWeb.Live.Plugins.API.TokenForm","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.TokenForm.handle_event/3","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.TokenForm.handle_info/2","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.TokenForm.mount/3","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Plugins.API.TokenForm.render/1","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#render/1"},{"type":"module","doc":"LiveView allowing listing, allowing and disallowing custom event properties.","title":"PlausibleWeb.Live.PropsSettings","ref":"PlausibleWeb.Live.PropsSettings.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.handle_event/3","ref":"PlausibleWeb.Live.PropsSettings.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.handle_info/2","ref":"PlausibleWeb.Live.PropsSettings.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.mount/3","ref":"PlausibleWeb.Live.PropsSettings.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.render/1","ref":"PlausibleWeb.Live.PropsSettings.html#render/1"},{"type":"module","doc":"Live view for the custom props creation form","title":"PlausibleWeb.Live.PropsSettings.Form","ref":"PlausibleWeb.Live.PropsSettings.Form.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.Form.handle_event/3","ref":"PlausibleWeb.Live.PropsSettings.Form.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.Form.handle_info/2","ref":"PlausibleWeb.Live.PropsSettings.Form.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.Form.mount/3","ref":"PlausibleWeb.Live.PropsSettings.Form.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.Form.render/1","ref":"PlausibleWeb.Live.PropsSettings.Form.html#render/1"},{"type":"module","doc":"Phoenix LiveComponent module that renders a list of custom properties","title":"PlausibleWeb.Live.PropsSettings.List","ref":"PlausibleWeb.Live.PropsSettings.List.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.PropsSettings.List.render/1","ref":"PlausibleWeb.Live.PropsSettings.List.html#render/1"},{"type":"function","doc":"* `props` (`:list`) (required)\n* `domain` (`:string`) (required)\n* `filter_text` (`:string`)","title":"Attributes - PlausibleWeb.Live.PropsSettings.List.render/1","ref":"PlausibleWeb.Live.PropsSettings.List.html#render/1-attributes"},{"type":"module","doc":"LiveView for registration form.","title":"PlausibleWeb.Live.RegisterForm","ref":"PlausibleWeb.Live.RegisterForm.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.RegisterForm.handle_event/3","ref":"PlausibleWeb.Live.RegisterForm.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.RegisterForm.handle_info/2","ref":"PlausibleWeb.Live.RegisterForm.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.RegisterForm.mount/3","ref":"PlausibleWeb.Live.RegisterForm.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.RegisterForm.render/1","ref":"PlausibleWeb.Live.RegisterForm.html#render/1"},{"type":"module","doc":"LiveView for password reset form.","title":"PlausibleWeb.Live.ResetPasswordForm","ref":"PlausibleWeb.Live.ResetPasswordForm.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ResetPasswordForm.handle_event/3","ref":"PlausibleWeb.Live.ResetPasswordForm.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ResetPasswordForm.handle_info/2","ref":"PlausibleWeb.Live.ResetPasswordForm.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ResetPasswordForm.mount/3","ref":"PlausibleWeb.Live.ResetPasswordForm.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.ResetPasswordForm.render/1","ref":"PlausibleWeb.Live.ResetPasswordForm.html#render/1"},{"type":"module","doc":"This module tries to supply LiveViews with some common Sentry context\n(without it, there is practically none).\n\nUse via `use PlausibleWeb.Live.SentryContext` in your LiveView module,\nor preferably via `use PlausibleWeb, :live_view`.\n\nIn case you have multiple LiveViews, there is `use PlausibleWeb, live_view: :no_sentry_context`\nexposed that allows you to skip using this module. This is because\nonly the root LiveView has access to `connect_info` and an exception will be\nthrown otherwise.","title":"PlausibleWeb.Live.SentryContext","ref":"PlausibleWeb.Live.SentryContext.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.SentryContext.on_mount/4","ref":"PlausibleWeb.Live.SentryContext.html#on_mount/4"},{"type":"module","doc":"LiveView for IP Addresses Shield","title":"PlausibleWeb.Live.Shields.Countries","ref":"PlausibleWeb.Live.Shields.Countries.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Countries.handle_info/2","ref":"PlausibleWeb.Live.Shields.Countries.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Countries.mount/3","ref":"PlausibleWeb.Live.Shields.Countries.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Countries.render/1","ref":"PlausibleWeb.Live.Shields.Countries.html#render/1"},{"type":"module","doc":"LiveView allowing Country Rules management","title":"PlausibleWeb.Live.Shields.CountryRules","ref":"PlausibleWeb.Live.Shields.CountryRules.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.CountryRules.handle_event/3","ref":"PlausibleWeb.Live.Shields.CountryRules.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.CountryRules.render/1","ref":"PlausibleWeb.Live.Shields.CountryRules.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.CountryRules.send_flash/2","ref":"PlausibleWeb.Live.Shields.CountryRules.html#send_flash/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.CountryRules.update/2","ref":"PlausibleWeb.Live.Shields.CountryRules.html#update/2"},{"type":"module","doc":"LiveView allowing hostname Rules management","title":"PlausibleWeb.Live.Shields.HostnameRules","ref":"PlausibleWeb.Live.Shields.HostnameRules.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.HostnameRules.handle_event/3","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.HostnameRules.render/1","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.HostnameRules.send_flash/2","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#send_flash/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.HostnameRules.suggest_hostnames/3","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#suggest_hostnames/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.HostnameRules.update/2","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#update/2"},{"type":"module","doc":"LiveView for Hostnames Shield","title":"PlausibleWeb.Live.Shields.Hostnames","ref":"PlausibleWeb.Live.Shields.Hostnames.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Hostnames.handle_info/2","ref":"PlausibleWeb.Live.Shields.Hostnames.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Hostnames.mount/3","ref":"PlausibleWeb.Live.Shields.Hostnames.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Hostnames.render/1","ref":"PlausibleWeb.Live.Shields.Hostnames.html#render/1"},{"type":"module","doc":"LiveView for IP Addresses Shield","title":"PlausibleWeb.Live.Shields.IPAddresses","ref":"PlausibleWeb.Live.Shields.IPAddresses.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPAddresses.handle_info/2","ref":"PlausibleWeb.Live.Shields.IPAddresses.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPAddresses.mount/3","ref":"PlausibleWeb.Live.Shields.IPAddresses.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPAddresses.render/1","ref":"PlausibleWeb.Live.Shields.IPAddresses.html#render/1"},{"type":"module","doc":"LiveView allowing IP Rules management","title":"PlausibleWeb.Live.Shields.IPRules","ref":"PlausibleWeb.Live.Shields.IPRules.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPRules.handle_event/3","ref":"PlausibleWeb.Live.Shields.IPRules.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPRules.render/1","ref":"PlausibleWeb.Live.Shields.IPRules.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPRules.send_flash/2","ref":"PlausibleWeb.Live.Shields.IPRules.html#send_flash/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.IPRules.update/2","ref":"PlausibleWeb.Live.Shields.IPRules.html#update/2"},{"type":"module","doc":"LiveView allowing page Rules management","title":"PlausibleWeb.Live.Shields.PageRules","ref":"PlausibleWeb.Live.Shields.PageRules.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.PageRules.handle_event/3","ref":"PlausibleWeb.Live.Shields.PageRules.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.PageRules.render/1","ref":"PlausibleWeb.Live.Shields.PageRules.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.PageRules.send_flash/2","ref":"PlausibleWeb.Live.Shields.PageRules.html#send_flash/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.PageRules.suggest_page_paths/3","ref":"PlausibleWeb.Live.Shields.PageRules.html#suggest_page_paths/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.PageRules.update/2","ref":"PlausibleWeb.Live.Shields.PageRules.html#update/2"},{"type":"module","doc":"LiveView for IP Addresses Shield","title":"PlausibleWeb.Live.Shields.Pages","ref":"PlausibleWeb.Live.Shields.Pages.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Pages.handle_info/2","ref":"PlausibleWeb.Live.Shields.Pages.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Pages.mount/3","ref":"PlausibleWeb.Live.Shields.Pages.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Shields.Pages.render/1","ref":"PlausibleWeb.Live.Shields.Pages.html#render/1"},{"type":"module","doc":"LiveView for sites index.","title":"PlausibleWeb.Live.Sites","ref":"PlausibleWeb.Live.Sites.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.ellipsis_menu/1","ref":"PlausibleWeb.Live.Sites.html#ellipsis_menu/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.favicon/1","ref":"PlausibleWeb.Live.Sites.html#favicon/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.handle_event/3","ref":"PlausibleWeb.Live.Sites.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.handle_info/2","ref":"PlausibleWeb.Live.Sites.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.handle_params/3","ref":"PlausibleWeb.Live.Sites.html#handle_params/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.icon_pin/1","ref":"PlausibleWeb.Live.Sites.html#icon_pin/1"},{"type":"function","doc":"* Global attributes are accepted.","title":"Attributes - PlausibleWeb.Live.Sites.icon_pin/1","ref":"PlausibleWeb.Live.Sites.html#icon_pin/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.invitation/1","ref":"PlausibleWeb.Live.Sites.html#invitation/1"},{"type":"function","doc":"* `site` (`Plausible.Site`) (required)\n* `invitation` (`Plausible.Auth.Invitation`) (required)\n* `hourly_stats` (`:map`) (required)","title":"Attributes - PlausibleWeb.Live.Sites.invitation/1","ref":"PlausibleWeb.Live.Sites.html#invitation/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.invitation_modal/1","ref":"PlausibleWeb.Live.Sites.html#invitation_modal/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.mount/3","ref":"PlausibleWeb.Live.Sites.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.percentage_change/1","ref":"PlausibleWeb.Live.Sites.html#percentage_change/1"},{"type":"function","doc":"* `change` (`:integer`) (required)","title":"Attributes - PlausibleWeb.Live.Sites.percentage_change/1","ref":"PlausibleWeb.Live.Sites.html#percentage_change/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.render/1","ref":"PlausibleWeb.Live.Sites.html#render/1"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.search_form/1","ref":"PlausibleWeb.Live.Sites.html#search_form/1"},{"type":"function","doc":"* `filter_text` (`:string`) - Defaults to `\"\"`.\n* `uri` (`URI`) (required)","title":"Attributes - PlausibleWeb.Live.Sites.search_form/1","ref":"PlausibleWeb.Live.Sites.html#search_form/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.site/1","ref":"PlausibleWeb.Live.Sites.html#site/1"},{"type":"function","doc":"* `site` (`Plausible.Site`) (required)\n* `hourly_stats` (`:map`) (required)","title":"Attributes - PlausibleWeb.Live.Sites.site/1","ref":"PlausibleWeb.Live.Sites.html#site/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.site_stats/1","ref":"PlausibleWeb.Live.Sites.html#site_stats/1"},{"type":"function","doc":"* `hourly_stats` (`:map`) (required)","title":"Attributes - PlausibleWeb.Live.Sites.site_stats/1","ref":"PlausibleWeb.Live.Sites.html#site_stats/1-attributes"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Sites.upgrade_nag_screen/1","ref":"PlausibleWeb.Live.Sites.html#upgrade_nag_screen/1"},{"type":"module","doc":"LiveView coordinating the site verification process.\nOnboarding new sites, renders a standalone component. \nEmbedded modal variant is available for general site settings.","title":"PlausibleWeb.Live.Verification","ref":"PlausibleWeb.Live.Verification.html"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Verification.handle_event/3","ref":"PlausibleWeb.Live.Verification.html#handle_event/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Verification.handle_info/2","ref":"PlausibleWeb.Live.Verification.html#handle_info/2"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Verification.mount/3","ref":"PlausibleWeb.Live.Verification.html#mount/3"},{"type":"function","doc":"","title":"PlausibleWeb.Live.Verification.render/1","ref":"PlausibleWeb.Live.Verification.html#render/1"},{"type":"module","doc":"MJML rendered for the weekly or monthly report e-mail","title":"PlausibleWeb.MJML.StatsReport","ref":"PlausibleWeb.MJML.StatsReport.html"},{"type":"function","doc":"Returns the raw MJML template. Useful for debugging rendering issues.","title":"PlausibleWeb.MJML.StatsReport.debug_mjml_template/0","ref":"PlausibleWeb.MJML.StatsReport.html#debug_mjml_template/0"},{"type":"function","doc":"Safely render the MJML template using Phoenix.HTML","title":"PlausibleWeb.MJML.StatsReport.render/1","ref":"PlausibleWeb.MJML.StatsReport.html#render/1"},{"type":"module","doc":"","title":"PlausibleWeb.PageController","ref":"PlausibleWeb.PageController.html"},{"type":"function","doc":"The root path is never accessible in Plausible.Cloud because it is handled by the upstream reverse proxy.\n\nThis controller action is only ever triggered in self-hosted Plausible.","title":"PlausibleWeb.PageController.index/2","ref":"PlausibleWeb.PageController.html#index/2"},{"type":"module","doc":"","title":"PlausibleWeb.PageView","ref":"PlausibleWeb.PageView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.PageView.__resource__/0","ref":"PlausibleWeb.PageView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.PageView.index.html/1","ref":"PlausibleWeb.PageView.html#index.html/1"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.PageView.render/2","ref":"PlausibleWeb.PageView.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.PageView.template_not_found/2","ref":"PlausibleWeb.PageView.html#template_not_found/2"},{"type":"module","doc":"Controller for Plugins API Capabilities - doesn't enforce authentication,\nserves as a comprehensive health check","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.index/2","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#index/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.open_api_operation/1","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#open_api_operation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.shared_security/0","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#shared_security/0"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.shared_tags/0","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#shared_tags/0"},{"type":"module","doc":"Controller for the CustomProp resource under Plugins API","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.disable/2","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#disable/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.enable/2","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#enable/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.open_api_operation/1","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#open_api_operation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.shared_security/0","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#shared_security/0"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.shared_tags/0","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#shared_tags/0"},{"type":"module","doc":"Controller for the Funnel resource under Plugins API","title":"PlausibleWeb.Plugins.API.Controllers.Funnels","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.create/2","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#create/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.get/2","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#get/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.index/2","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#index/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.open_api_operation/1","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#open_api_operation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.shared_security/0","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#shared_security/0"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.shared_tags/0","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#shared_tags/0"},{"type":"module","doc":"Controller for the Goal resource under Plugins API","title":"PlausibleWeb.Plugins.API.Controllers.Goals","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.create/2","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#create/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.delete/2","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#delete/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.delete_bulk/2","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#delete_bulk/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.get/2","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#get/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.index/2","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#index/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.open_api_operation/1","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#open_api_operation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.shared_security/0","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#shared_security/0"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.Goals.shared_tags/0","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#shared_tags/0"},{"type":"module","doc":"Controller for the Shared Link resource under Plugins API","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.create/2","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#create/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.get/2","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#get/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.index/2","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#index/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.open_api_operation/1","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#open_api_operation/1"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.shared_security/0","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#shared_security/0"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.shared_tags/0","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#shared_tags/0"},{"type":"module","doc":"Common responses for Plugins API","title":"PlausibleWeb.Plugins.API.Errors","ref":"PlausibleWeb.Plugins.API.Errors.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Errors.error/3","ref":"PlausibleWeb.Plugins.API.Errors.html#error/3"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Errors.unauthorized/1","ref":"PlausibleWeb.Plugins.API.Errors.html#unauthorized/1"},{"type":"module","doc":"OpenAPI schema for Capabilities","title":"PlausibleWeb.Plugins.API.Schemas.Capabilities","ref":"PlausibleWeb.Plugins.API.Schemas.Capabilities.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Capabilities.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Capabilities.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Capabilities.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Capabilities.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Goal","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Custom Property disable request","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Custom Property creation request","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for SharedLink list response","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for an error included in a response","title":"PlausibleWeb.Plugins.API.Schemas.Error","ref":"PlausibleWeb.Plugins.API.Schemas.Error.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Error.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Error.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Error.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Error.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Funnel","title":"PlausibleWeb.Plugins.API.Schemas.Funnel","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Funnel creation request - get or creates goals along the way","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Funnel list response","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Goal","title":"PlausibleWeb.Plugins.API.Schemas.Goal","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Goal creation request","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Custom Event Goal creation request","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Pageview Goal creation request","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Custom Event Goal creation request","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Custom Event Goal object","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for bulk Goal deletion request","title":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Goals list response","title":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Pageview Goal object","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for Revenue Goal object","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for common Goal Type\n\nFuture-proof: funnels etc.","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Type","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.html#t:t/0"},{"type":"module","doc":"OpenAPI Link schema","title":"PlausibleWeb.Plugins.API.Schemas.Link","ref":"PlausibleWeb.Plugins.API.Schemas.Link.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Link.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Link.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Link.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Link.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for a generic 404 response","title":"PlausibleWeb.Plugins.API.Schemas.NotFound","ref":"PlausibleWeb.Plugins.API.Schemas.NotFound.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.NotFound.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.NotFound.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.NotFound.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.NotFound.html#t:t/0"},{"type":"module","doc":"Pagination metadata OpenAPI schema","title":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata","ref":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for a generic 402 response","title":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired","ref":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for SharedLink object","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for SharedLink creation request","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for SharedLink list response","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for a generic 401 response","title":"PlausibleWeb.Plugins.API.Schemas.Unauthorized","ref":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.html#t:t/0"},{"type":"module","doc":"OpenAPI schema for a generic 422 response","title":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity","ref":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.schema/0","ref":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.html#schema/0"},{"type":"type","doc":"","title":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.t/0","ref":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.html#t:t/0"},{"type":"module","doc":"OpenAPI specification for the Plugins API","title":"PlausibleWeb.Plugins.API.Spec","ref":"PlausibleWeb.Plugins.API.Spec.html"},{"type":"module","doc":"View for rendering Capabilities on the Plugins API","title":"PlausibleWeb.Plugins.API.Views.Capabilities","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.Capabilities.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.Capabilities.render/2","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.Capabilities.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html#template_not_found/2"},{"type":"module","doc":"View for rendering Custom Props in the Plugins API","title":"PlausibleWeb.Plugins.API.Views.CustomProp","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.CustomProp.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.CustomProp.render/2","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.CustomProp.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html#template_not_found/2"},{"type":"module","doc":"View for rendering Plugins REST API errors","title":"PlausibleWeb.Plugins.API.Views.Error","ref":"PlausibleWeb.Plugins.API.Views.Error.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.Error.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.Error.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.Error.render/2","ref":"PlausibleWeb.Plugins.API.Views.Error.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.Error.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.Error.html#template_not_found/2"},{"type":"module","doc":"View for rendering Funnels in the Plugins API","title":"PlausibleWeb.Plugins.API.Views.Funnel","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.Funnel.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.Funnel.render/2","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.Funnel.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html#template_not_found/2"},{"type":"module","doc":"View for rendering Goals in the Plugins API","title":"PlausibleWeb.Plugins.API.Views.Goal","ref":"PlausibleWeb.Plugins.API.Views.Goal.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.Goal.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.Goal.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.Goal.render/2","ref":"PlausibleWeb.Plugins.API.Views.Goal.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.Goal.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.Goal.html#template_not_found/2"},{"type":"module","doc":"A view capable of rendering pagination metadata included\nin responses containing lists of objects.","title":"PlausibleWeb.Plugins.API.Views.Pagination","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.Pagination.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.Pagination.render/2","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugins.API.Views.Pagination.render_metadata_links/4","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#render_metadata_links/4"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.Pagination.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#template_not_found/2"},{"type":"module","doc":"View for rendering Shared Links in the Plugins API","title":"PlausibleWeb.Plugins.API.Views.SharedLink","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Plugins.API.Views.SharedLink.__resource__/0","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Plugins.API.Views.SharedLink.render/2","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Plugins.API.Views.SharedLink.template_not_found/2","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html#template_not_found/2"},{"type":"module","doc":"This module defines the test case to be used by\ntests that require setting up a Plugins API connection.","title":"PlausibleWeb.PluginsAPICase","ref":"PlausibleWeb.PluginsAPICase.html"},{"type":"module","doc":"Plug for Basic HTTP Authentication using\nPlugins API Tokens lookup.","title":"PlausibleWeb.Plugs.AuthorizePluginsAPI","ref":"PlausibleWeb.Plugs.AuthorizePluginsAPI.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.AuthorizePluginsAPI.call/2","ref":"PlausibleWeb.Plugs.AuthorizePluginsAPI.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.AuthorizePluginsAPI.init/1","ref":"PlausibleWeb.Plugs.AuthorizePluginsAPI.html#init/1"},{"type":"module","doc":"Plug for authorizing access to Stats and Sites APIs.\n\nThe plug expects `:api_scope` to be provided in the assigns. The scope\nwill then be used to check for API key validity. The assign can be\nprovided in the router configuration in a following way:\n\n scope \"/api/v1/stats\", PlausibleWeb.Api, assigns: %{api_scope: \"some:scope:*\"} do\n pipe_through [:public_api, PlausibleWeb.Plugs.AuthorizePublicAPI]\n\n # route definitions follow\n # ...\n end\n\nThe scope from `:api_scope` is checked for match against all scopes from API key's\n`scopes` field. If the scope is among `@implicit_scopes`, it's considered to be\npresent for any valid API key. Scopes are checked for match by prefix, so if we have\n`some:scope:*` in matching route `:api_scope` and the API key has `some:*` in its\n`scopes` field, they will match.\n\nAfter a match is found, additional verification can be conducted, like in case of\n`stats:read:*`, where valid site ID is expected among parameters too.\n\nAll API requests are rate limited per API key, enforcing a given hourly request limit.","title":"PlausibleWeb.Plugs.AuthorizePublicAPI","ref":"PlausibleWeb.Plugs.AuthorizePublicAPI.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.AuthorizePublicAPI.call/2","ref":"PlausibleWeb.Plugs.AuthorizePublicAPI.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.AuthorizePublicAPI.init/1","ref":"PlausibleWeb.Plugs.AuthorizePublicAPI.html#init/1"},{"type":"module","doc":"A thin macro wrapper around Plug.ErrorHandler that adds Sentry context\n containing a readable support hash presented to the users.\n To be used in the user-facing APIs, so that we don't leak internal\n server errors.\n\n Usage: `use PlausibleWeb.Plugs.ErrorHandler`","title":"PlausibleWeb.Plugs.ErrorHandler","ref":"PlausibleWeb.Plugs.ErrorHandler.html"},{"type":"module","doc":"Plug toggling registration according to selfhosted state.","title":"PlausibleWeb.Plugs.MaybeDisableRegistration","ref":"PlausibleWeb.Plugs.MaybeDisableRegistration.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.MaybeDisableRegistration.call/2","ref":"PlausibleWeb.Plugs.MaybeDisableRegistration.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.MaybeDisableRegistration.init/1","ref":"PlausibleWeb.Plugs.MaybeDisableRegistration.html#init/1"},{"type":"module","doc":"Rejects bot requests by any means available.\n\nWe're adding `x-robots-tag` to the response header and annotate the conn\nwith \"noindex, nofollow\" under `private.robots` key.\n\nThe only exception is, if the request is trying to access our live demo\nat plausible.io/plausible.io - in which case we'll allow indexing, but deny\nfollowing links and skip the bot detection, in kind robots we trust.","title":"PlausibleWeb.Plugs.NoRobots","ref":"PlausibleWeb.Plugs.NoRobots.html"},{"type":"module","doc":"Plug for bumping timeout on user session on every dashboard request.","title":"PlausibleWeb.Plugs.UserSessionTouch","ref":"PlausibleWeb.Plugs.UserSessionTouch.html"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.UserSessionTouch.call/2","ref":"PlausibleWeb.Plugs.UserSessionTouch.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.Plugs.UserSessionTouch.init/1","ref":"PlausibleWeb.Plugs.UserSessionTouch.html#init/1"},{"type":"module","doc":"","title":"PlausibleWeb.RefInspector","ref":"PlausibleWeb.RefInspector.html"},{"type":"function","doc":"","title":"PlausibleWeb.RefInspector.format_referrer/1","ref":"PlausibleWeb.RefInspector.html#format_referrer/1"},{"type":"function","doc":"","title":"PlausibleWeb.RefInspector.parse/1","ref":"PlausibleWeb.RefInspector.html#parse/1"},{"type":"function","doc":"","title":"PlausibleWeb.RefInspector.right_uri?/1","ref":"PlausibleWeb.RefInspector.html#right_uri?/1"},{"type":"module","doc":"Implements the strategy of retrieving client's remote IP","title":"PlausibleWeb.RemoteIP","ref":"PlausibleWeb.RemoteIP.html"},{"type":"function","doc":"","title":"PlausibleWeb.RemoteIP.get/1","ref":"PlausibleWeb.RemoteIP.html#get/1"},{"type":"module","doc":"","title":"PlausibleWeb.RequireAccountPlug","ref":"PlausibleWeb.RequireAccountPlug.html"},{"type":"function","doc":"","title":"PlausibleWeb.RequireAccountPlug.call/2","ref":"PlausibleWeb.RequireAccountPlug.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.RequireAccountPlug.init/1","ref":"PlausibleWeb.RequireAccountPlug.html#init/1"},{"type":"module","doc":"","title":"PlausibleWeb.RequireLoggedOutPlug","ref":"PlausibleWeb.RequireLoggedOutPlug.html"},{"type":"function","doc":"","title":"PlausibleWeb.RequireLoggedOutPlug.call/2","ref":"PlausibleWeb.RequireLoggedOutPlug.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.RequireLoggedOutPlug.init/1","ref":"PlausibleWeb.RequireLoggedOutPlug.html#init/1"},{"type":"module","doc":"","title":"PlausibleWeb.Router","ref":"PlausibleWeb.Router.html"},{"type":"function","doc":"","title":"PlausibleWeb.Router.api/2","ref":"PlausibleWeb.Router.html#api/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.app_layout/2","ref":"PlausibleWeb.Router.html#app_layout/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.browser/2","ref":"PlausibleWeb.Router.html#browser/2"},{"type":"function","doc":"Callback invoked by Plug on every request.","title":"PlausibleWeb.Router.call/2","ref":"PlausibleWeb.Router.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.csrf/2","ref":"PlausibleWeb.Router.html#csrf/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.external_api/2","ref":"PlausibleWeb.Router.html#external_api/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.flags/2","ref":"PlausibleWeb.Router.html#flags/2"},{"type":"function","doc":"Callback required by Plug that initializes the router\nfor serving web requests.","title":"PlausibleWeb.Router.init/1","ref":"PlausibleWeb.Router.html#init/1"},{"type":"function","doc":"","title":"PlausibleWeb.Router.internal_stats_api/2","ref":"PlausibleWeb.Router.html#internal_stats_api/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.kaffy_browser/2","ref":"PlausibleWeb.Router.html#kaffy_browser/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.plugins_api/2","ref":"PlausibleWeb.Router.html#plugins_api/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.plugins_api_auth/2","ref":"PlausibleWeb.Router.html#plugins_api_auth/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.public_api/2","ref":"PlausibleWeb.Router.html#public_api/2"},{"type":"function","doc":"","title":"PlausibleWeb.Router.shared_link/2","ref":"PlausibleWeb.Router.html#shared_link/2"},{"type":"module","doc":"NOTE: This plug will be replaced with a different\nsession expiration mechanism once server-side persisted\nsessions are rolled out.","title":"PlausibleWeb.SessionTimeoutPlug","ref":"PlausibleWeb.SessionTimeoutPlug.html"},{"type":"function","doc":"","title":"PlausibleWeb.SessionTimeoutPlug.call/2","ref":"PlausibleWeb.SessionTimeoutPlug.html#call/2"},{"type":"function","doc":"","title":"PlausibleWeb.SessionTimeoutPlug.init/1","ref":"PlausibleWeb.SessionTimeoutPlug.html#init/1"},{"type":"module","doc":"This controller deals with user management via the UI in Site Settings -> People. It's important to enforce permissions in this controller.\n\n Owner - Can manage users, can trigger a 'transfer ownership' request\n Admin - Can manage users\n Viewer - Can not access user management settings\n Anyone - Can accept invitations\n\n Everything else should be explicitly disallowed.","title":"PlausibleWeb.Site.MembershipController","ref":"PlausibleWeb.Site.MembershipController.html"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipController.invite_member/2","ref":"PlausibleWeb.Site.MembershipController.html#invite_member/2"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipController.invite_member_form/2","ref":"PlausibleWeb.Site.MembershipController.html#invite_member_form/2"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipController.remove_member/2","ref":"PlausibleWeb.Site.MembershipController.html#remove_member/2"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipController.transfer_ownership/2","ref":"PlausibleWeb.Site.MembershipController.html#transfer_ownership/2"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipController.transfer_ownership_form/2","ref":"PlausibleWeb.Site.MembershipController.html#transfer_ownership_form/2"},{"type":"function","doc":"Updates the role of a user. The user being updated could be the same or different from the user taking\n the action. When updating the role, it's important to enforce permissions:\n\n Owner - Can update anyone's role except for themselves. If they want to change their own role, they have to use the 'transfer ownership' feature.\n Admin - Can update anyone's role except for owners. Can downgrade their own access to 'viewer'. Can promote a viewer to admin.","title":"PlausibleWeb.Site.MembershipController.update_role/2","ref":"PlausibleWeb.Site.MembershipController.html#update_role/2"},{"type":"module","doc":"","title":"PlausibleWeb.Site.MembershipView","ref":"PlausibleWeb.Site.MembershipView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.Site.MembershipView.__resource__/0","ref":"PlausibleWeb.Site.MembershipView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipView.invite_member_form.html/1","ref":"PlausibleWeb.Site.MembershipView.html#invite_member_form.html/1"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.Site.MembershipView.render/2","ref":"PlausibleWeb.Site.MembershipView.html#render/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.Site.MembershipView.template_not_found/2","ref":"PlausibleWeb.Site.MembershipView.html#template_not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.Site.MembershipView.transfer_ownership_form.html/1","ref":"PlausibleWeb.Site.MembershipView.html#transfer_ownership_form.html/1"},{"type":"module","doc":"","title":"PlausibleWeb.SiteController","ref":"PlausibleWeb.SiteController.html"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.add_monthly_report_recipient/2","ref":"PlausibleWeb.SiteController.html#add_monthly_report_recipient/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.add_traffic_change_notification_recipient/2","ref":"PlausibleWeb.SiteController.html#add_traffic_change_notification_recipient/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.add_weekly_report_recipient/2","ref":"PlausibleWeb.SiteController.html#add_weekly_report_recipient/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.change_domain/2","ref":"PlausibleWeb.SiteController.html#change_domain/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.change_domain_submit/2","ref":"PlausibleWeb.SiteController.html#change_domain_submit/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.create_shared_link/2","ref":"PlausibleWeb.SiteController.html#create_shared_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.create_site/2","ref":"PlausibleWeb.SiteController.html#create_site/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.csv_import/2","ref":"PlausibleWeb.SiteController.html#csv_import/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.delete_google_auth/2","ref":"PlausibleWeb.SiteController.html#delete_google_auth/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.delete_shared_link/2","ref":"PlausibleWeb.SiteController.html#delete_shared_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.delete_site/2","ref":"PlausibleWeb.SiteController.html#delete_site/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.disable_monthly_report/2","ref":"PlausibleWeb.SiteController.html#disable_monthly_report/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.disable_traffic_change_notification/2","ref":"PlausibleWeb.SiteController.html#disable_traffic_change_notification/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.disable_weekly_report/2","ref":"PlausibleWeb.SiteController.html#disable_weekly_report/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.download_export/2","ref":"PlausibleWeb.SiteController.html#download_export/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.edit_shared_link/2","ref":"PlausibleWeb.SiteController.html#edit_shared_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.enable_monthly_report/2","ref":"PlausibleWeb.SiteController.html#enable_monthly_report/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.enable_traffic_change_notification/2","ref":"PlausibleWeb.SiteController.html#enable_traffic_change_notification/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.enable_weekly_report/2","ref":"PlausibleWeb.SiteController.html#enable_weekly_report/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.forget_import/2","ref":"PlausibleWeb.SiteController.html#forget_import/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.forget_imported/2","ref":"PlausibleWeb.SiteController.html#forget_imported/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.make_private/2","ref":"PlausibleWeb.SiteController.html#make_private/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.make_public/2","ref":"PlausibleWeb.SiteController.html#make_public/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.new/2","ref":"PlausibleWeb.SiteController.html#new/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.new_shared_link/2","ref":"PlausibleWeb.SiteController.html#new_shared_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.remove_monthly_report_recipient/2","ref":"PlausibleWeb.SiteController.html#remove_monthly_report_recipient/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.remove_traffic_change_notification_recipient/2","ref":"PlausibleWeb.SiteController.html#remove_traffic_change_notification_recipient/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.remove_weekly_report_recipient/2","ref":"PlausibleWeb.SiteController.html#remove_weekly_report_recipient/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.reset_stats/2","ref":"PlausibleWeb.SiteController.html#reset_stats/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings/2","ref":"PlausibleWeb.SiteController.html#settings/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_danger_zone/2","ref":"PlausibleWeb.SiteController.html#settings_danger_zone/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_email_reports/2","ref":"PlausibleWeb.SiteController.html#settings_email_reports/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_funnels/2","ref":"PlausibleWeb.SiteController.html#settings_funnels/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_general/2","ref":"PlausibleWeb.SiteController.html#settings_general/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_goals/2","ref":"PlausibleWeb.SiteController.html#settings_goals/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_imports_exports/2","ref":"PlausibleWeb.SiteController.html#settings_imports_exports/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_integrations/2","ref":"PlausibleWeb.SiteController.html#settings_integrations/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_people/2","ref":"PlausibleWeb.SiteController.html#settings_people/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_props/2","ref":"PlausibleWeb.SiteController.html#settings_props/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_shields/2","ref":"PlausibleWeb.SiteController.html#settings_shields/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.settings_visibility/2","ref":"PlausibleWeb.SiteController.html#settings_visibility/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.update_feature_visibility/2","ref":"PlausibleWeb.SiteController.html#update_feature_visibility/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.update_google_auth/2","ref":"PlausibleWeb.SiteController.html#update_google_auth/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.update_settings/2","ref":"PlausibleWeb.SiteController.html#update_settings/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.update_shared_link/2","ref":"PlausibleWeb.SiteController.html#update_shared_link/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteController.update_traffic_change_notification/2","ref":"PlausibleWeb.SiteController.html#update_traffic_change_notification/2"},{"type":"module","doc":"","title":"PlausibleWeb.SiteView","ref":"PlausibleWeb.SiteView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.SiteView.__resource__/0","ref":"PlausibleWeb.SiteView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.change_domain.html/1","ref":"PlausibleWeb.SiteView.html#change_domain.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.csv_import.html/1","ref":"PlausibleWeb.SiteView.html#csv_import.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.edit_shared_link.html/1","ref":"PlausibleWeb.SiteView.html#edit_shared_link.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.new.html/1","ref":"PlausibleWeb.SiteView.html#new.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.new_shared_link.html/1","ref":"PlausibleWeb.SiteView.html#new_shared_link.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.plausible_url/0","ref":"PlausibleWeb.SiteView.html#plausible_url/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.SiteView.render/2","ref":"PlausibleWeb.SiteView.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.render_snippet/1","ref":"PlausibleWeb.SiteView.html#render_snippet/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_danger_zone.html/1","ref":"PlausibleWeb.SiteView.html#settings_danger_zone.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_email_reports.html/1","ref":"PlausibleWeb.SiteView.html#settings_email_reports.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_funnels.html/1","ref":"PlausibleWeb.SiteView.html#settings_funnels.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_general.html/1","ref":"PlausibleWeb.SiteView.html#settings_general.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_goals.html/1","ref":"PlausibleWeb.SiteView.html#settings_goals.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_imports_exports.html/1","ref":"PlausibleWeb.SiteView.html#settings_imports_exports.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_integrations.html/1","ref":"PlausibleWeb.SiteView.html#settings_integrations.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_people.html/1","ref":"PlausibleWeb.SiteView.html#settings_people.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_props.html/1","ref":"PlausibleWeb.SiteView.html#settings_props.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_search_console.html/1","ref":"PlausibleWeb.SiteView.html#settings_search_console.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_shields.html/1","ref":"PlausibleWeb.SiteView.html#settings_shields.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.settings_visibility.html/1","ref":"PlausibleWeb.SiteView.html#settings_visibility.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.shared_link_dest/2","ref":"PlausibleWeb.SiteView.html#shared_link_dest/2"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.SiteView.template_not_found/2","ref":"PlausibleWeb.SiteView.html#template_not_found/2"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.traffic_change_form/1","ref":"PlausibleWeb.SiteView.html#traffic_change_form/1"},{"type":"function","doc":"","title":"PlausibleWeb.SiteView.with_indefinite_article/1","ref":"PlausibleWeb.SiteView.html#with_indefinite_article/1"},{"type":"module","doc":"This controller is responsible for rendering stats dashboards.\n\nThe stats dashboards are currently the only part of the app that uses client-side\nrendering. Since the dashboards are heavily interactive, they are built with React\nwhich is an appropriate choice for highly interactive browser UIs.\n\n \nsequenceDiagram\n Browser->>StatsController: GET /mydomain.com\n StatsController-->>Browser: StatsView.render(\"stats.html\")\n Note left of Browser: ReactDom.render(Dashboard)\n\n Browser -) Api.StatsController: GET /api/stats/mydomain.com/top-stats\n Api.StatsController --) Browser: {\"top_stats\": [...]}\n Note left of Browser: TopStats.render()\n\n Browser -) Api.StatsController: GET /api/stats/mydomain.com/main-graph\n Api.StatsController --) Browser: [{\"plot\": [...], \"labels\": [...]}, ...]\n Note left of Browser: VisitorGraph.render()\n\n Browser -) Api.StatsController: GET /api/stats/mydomain.com/sources\n Api.StatsController --) Browser: [{\"name\": \"Google\", \"visitors\": 292150}, ...]\n Note left of Browser: Sources.render()\n\n Note over Browser,StatsController: And so on, for all reports in the viewport\n \n\nThis reasoning for this sequence is as follows:\n 1. First paint is fast because it doesn't do any data aggregation yet - good UX\n 2. The basic structure of the dashboard is rendered with spinners before reports are ready - good UX\n 2. Rendering on the frontend allows for maximum interactivity. Re-rendering and re-fetching can be as granular as needed.\n 3. Routing on the frontend allows the user to navigate the dashboard without reloading the page and losing context\n 4. Rendering on the frontend allows caching results in the browser to reduce pressure on backends and storage\n 3.1 No client-side caching has been implemented yet. This is still theoretical. See https://github.com/plausible/analytics/discussions/1278\n 3.2 This is a big potential opportunity, because analytics data is mostly immutable. Clients can cache all historical data.\n 5. Since frontend rendering & navigation is harder to build and maintain than regular server-rendered HTML, we don't use SPA-style rendering anywhere else\n .The only place currently where the benefits outweigh the costs is the dashboard.","title":"PlausibleWeb.StatsController","ref":"PlausibleWeb.StatsController.html"},{"type":"function","doc":"","title":"PlausibleWeb.StatsController.authenticate_shared_link/2","ref":"PlausibleWeb.StatsController.html#authenticate_shared_link/2"},{"type":"function","doc":"The export is limited to 300 entries for other reports and 100 entries for pages because bigger result sets\nstart causing failures. Since we request data like time on page or bounce_rate for pages in a separate query\nusing the IN filter, it causes the requests to balloon in payload size.","title":"PlausibleWeb.StatsController.csv_export/2","ref":"PlausibleWeb.StatsController.html#csv_export/2"},{"type":"function","doc":"Authorizes and renders a shared link:\n 1. Shared link with no password protection: needs to just make sure the shared link entry is still\n in our database. This check makes sure shared link access can be revoked by the site admins. If the\n shared link exists, render it directly.\n\n 2. Shared link with password protection: Same checks as without the password, but an extra step is taken to\n protect the page with a password. When the user passes the password challenge, a cookie is set with Plausible.Auth.Token.sign_shared_link().\n The cookie allows the user to access the dashboard for 24 hours without entering the password again.\n\n #","title":"PlausibleWeb.StatsController.shared_link/2","ref":"PlausibleWeb.StatsController.html#shared_link/2"},{"type":"function","doc":"The URL format for shared links was changed in [this pull request](https://github.com/plausible/analytics/pull/752) in order\n to make the URLs easier to bookmark. The old format is supported along with the new in order to not break old links.\n\n See: https://plausible.io/docs/shared-links","title":"Backwards compatibility - PlausibleWeb.StatsController.shared_link/2","ref":"PlausibleWeb.StatsController.html#shared_link/2-backwards-compatibility"},{"type":"function","doc":"","title":"PlausibleWeb.StatsController.stats/2","ref":"PlausibleWeb.StatsController.html#stats/2"},{"type":"module","doc":"","title":"PlausibleWeb.StatsView","ref":"PlausibleWeb.StatsView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.StatsView.__resource__/0","ref":"PlausibleWeb.StatsView.html#__resource__/0"},{"type":"function","doc":"","title":"PlausibleWeb.StatsView.large_number_format/1","ref":"PlausibleWeb.StatsView.html#large_number_format/1"},{"type":"function","doc":"","title":"PlausibleWeb.StatsView.plausible_url/0","ref":"PlausibleWeb.StatsView.html#plausible_url/0"},{"type":"function","doc":"Returns a readable stats URL.\n\nNative Phoenix router functions percent-encode all diacritics, resulting in\nugly URLs, e.g. `https://plausible.io/café.com` transforms into\n`https://plausible.io/caf%C3%A9.com`.\n\nThis function encodes only the slash (`/`) character from the site's domain.","title":"PlausibleWeb.StatsView.pretty_stats_url/1","ref":"PlausibleWeb.StatsView.html#pretty_stats_url/1"},{"type":"function","doc":"iex> PlausibleWeb.StatsView.pretty_stats_url(%Plausible.Site{domain: \"user.gittea.io/repo\"})\n \"http://localhost:8000/user.gittea.io%2Frepo\"\n\n iex> PlausibleWeb.StatsView.pretty_stats_url(%Plausible.Site{domain: \"anakin.test\"})\n \"http://localhost:8000/anakin.test\"\n\n iex> PlausibleWeb.StatsView.pretty_stats_url(%Plausible.Site{domain: \"café.test\"})\n \"http://localhost:8000/café.test\"","title":"Examples - PlausibleWeb.StatsView.pretty_stats_url/1","ref":"PlausibleWeb.StatsView.html#pretty_stats_url/1-examples"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.StatsView.render/2","ref":"PlausibleWeb.StatsView.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.StatsView.shared_link_password.html/1","ref":"PlausibleWeb.StatsView.html#shared_link_password.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.StatsView.site_locked.html/1","ref":"PlausibleWeb.StatsView.html#site_locked.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.StatsView.stats.html/1","ref":"PlausibleWeb.StatsView.html#stats.html/1"},{"type":"function","doc":"","title":"PlausibleWeb.StatsView.stats_container_class/1","ref":"PlausibleWeb.StatsView.html#stats_container_class/1"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.StatsView.template_not_found/2","ref":"PlausibleWeb.StatsView.html#template_not_found/2"},{"type":"module","doc":"","title":"PlausibleWeb.Tracker","ref":"PlausibleWeb.Tracker.html"},{"type":"function","doc":"","title":"PlausibleWeb.Tracker.call/2","ref":"PlausibleWeb.Tracker.html#call/2"},{"type":"function","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","title":"PlausibleWeb.Tracker.child_spec/1","ref":"PlausibleWeb.Tracker.html#child_spec/1"},{"type":"function","doc":"","title":"PlausibleWeb.Tracker.init/1","ref":"PlausibleWeb.Tracker.html#init/1"},{"type":"module","doc":"Functions for managing session data related to Two-Factor\nAuthentication.","title":"PlausibleWeb.TwoFactor.Session","ref":"PlausibleWeb.TwoFactor.Session.html"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.clear_2fa_user/1","ref":"PlausibleWeb.TwoFactor.Session.html#clear_2fa_user/1"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.clear_remember_2fa/1","ref":"PlausibleWeb.TwoFactor.Session.html#clear_remember_2fa/1"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.get_2fa_user/1","ref":"PlausibleWeb.TwoFactor.Session.html#get_2fa_user/1"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.maybe_set_remember_2fa/3","ref":"PlausibleWeb.TwoFactor.Session.html#maybe_set_remember_2fa/3"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.remember_2fa?/2","ref":"PlausibleWeb.TwoFactor.Session.html#remember_2fa?/2"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.remember_2fa_days/0","ref":"PlausibleWeb.TwoFactor.Session.html#remember_2fa_days/0"},{"type":"function","doc":"","title":"PlausibleWeb.TwoFactor.Session.set_2fa_user/2","ref":"PlausibleWeb.TwoFactor.Session.html#set_2fa_user/2"},{"type":"module","doc":"","title":"PlausibleWeb.UnsubscribeController","ref":"PlausibleWeb.UnsubscribeController.html"},{"type":"function","doc":"","title":"PlausibleWeb.UnsubscribeController.monthly_report/2","ref":"PlausibleWeb.UnsubscribeController.html#monthly_report/2"},{"type":"function","doc":"","title":"PlausibleWeb.UnsubscribeController.weekly_report/2","ref":"PlausibleWeb.UnsubscribeController.html#weekly_report/2"},{"type":"module","doc":"","title":"PlausibleWeb.UnsubscribeView","ref":"PlausibleWeb.UnsubscribeView.html"},{"type":"function","doc":"The resource name, as an atom, for this view","title":"PlausibleWeb.UnsubscribeView.__resource__/0","ref":"PlausibleWeb.UnsubscribeView.html#__resource__/0"},{"type":"function","doc":"Renders the given template locally.","title":"PlausibleWeb.UnsubscribeView.render/2","ref":"PlausibleWeb.UnsubscribeView.html#render/2"},{"type":"function","doc":"","title":"PlausibleWeb.UnsubscribeView.success.html/1","ref":"PlausibleWeb.UnsubscribeView.html#success.html/1"},{"type":"function","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","title":"PlausibleWeb.UnsubscribeView.template_not_found/2","ref":"PlausibleWeb.UnsubscribeView.html#template_not_found/2"},{"type":"module","doc":"Functions for user session management.\n\nIn it's current shape, both current (legacy) and soon to be implemented (new)\nuser sessions are supported side by side.\n\nThe legacy token is still accepted from the session cookie. Once 14 days\npass (the current time window for which session cookie is valid without\nany activity), the legacy cookies won't be accepted anymore (legacy token\nretrieval is tracked with logging) and the logic will be cleaned of branching\nfor legacy session.","title":"PlausibleWeb.UserAuth","ref":"PlausibleWeb.UserAuth.html"},{"type":"function","doc":"","title":"PlausibleWeb.UserAuth.convert_legacy_session/1","ref":"PlausibleWeb.UserAuth.html#convert_legacy_session/1"},{"type":"function","doc":"","title":"PlausibleWeb.UserAuth.get_user_session/1","ref":"PlausibleWeb.UserAuth.html#get_user_session/1"},{"type":"function","doc":"","title":"PlausibleWeb.UserAuth.log_in_user/3","ref":"PlausibleWeb.UserAuth.html#log_in_user/3"},{"type":"function","doc":"","title":"PlausibleWeb.UserAuth.log_out_user/1","ref":"PlausibleWeb.UserAuth.html#log_out_user/1"},{"type":"function","doc":"Sets the `logged_in` cookie share with the static site for determining\nwhether client is authenticated.\n\nAs it's a separate cookie, there's a chance it might fall out of sync\nwith session cookie state due to manual deletion or premature expiration.","title":"PlausibleWeb.UserAuth.set_logged_in_cookie/1","ref":"PlausibleWeb.UserAuth.html#set_logged_in_cookie/1"},{"type":"function","doc":"","title":"PlausibleWeb.UserAuth.touch_user_session/1","ref":"PlausibleWeb.UserAuth.html#touch_user_session/1"},{"type":"function","doc":"","title":"PlausibleWeb.UserAuth.touch_user_session/2","ref":"PlausibleWeb.UserAuth.html#touch_user_session/2"},{"type":"extras","doc":"# Plausible Analytics\n\n \n \n \n \n \n \n Simple Metrics |\n Lightweight Script |\n Privacy Focused |\n Open Source |\n Docs |\n Contributing \n \n \n\n[Plausible Analytics](https://plausible.io/) is an easy to use, lightweight (< 1 KB), open source and privacy-friendly alternative to Google Analytics. It doesn’t use cookies and is fully compliant with GDPR, CCPA and PECR. You can self-host Plausible Community Edition or have us manage Plausible Analytics for you in the cloud. Here's [the live demo of our own website stats](https://plausible.io/plausible.io). Made and hosted in the EU 🇪🇺\n\nWe are dedicated to making web analytics more privacy-friendly. Our mission is to reduce corporate surveillance by providing an alternative web analytics tool which doesn’t come from the AdTech world. We are completely independent and solely funded by our subscribers.\n\n![Plausible Analytics](https://plausible.io/docs/img/plausible-analytics.png)","title":"Introduction","ref":"readme.html"},{"type":"extras","doc":"Here's what makes Plausible a great Google Analytics alternative and why we're trusted by 12,000+ paying subscribers to deliver their website and business insights:\n\n- **Clutter Free**: Plausible Analytics provides [simple web analytics](https://plausible.io/simple-web-analytics) and it cuts through the noise. No layers of menus, no need for custom reports. Get all the important insights on one single page. No training necessary.\n- **GDPR/CCPA/PECR compliant**: Measure traffic, not individuals. No personal data or IP addresses are ever stored in our database. We don't use cookies or any other persistent identifiers. [Read more about our data policy](https://plausible.io/data-policy)\n- **Lightweight**: Plausible Analytics works by loading a script on your website, like Google Analytics. Our script is [45x smaller](https://plausible.io/lightweight-web-analytics), making your website quicker to load. You can also send events directly to our [events API](https://plausible.io/docs/events-api).\n- **Email or Slack reports**: Keep an eye on your traffic with weekly and/or monthly email or Slack reports. You can also get traffic spike notifications.\n- **Invite team members and share stats**: You have the option to be transparent and open your web analytics to everyone. Your website stats are private by default but you can choose to make them public so anyone with your custom link can view them. You can [invite team members](https://plausible.io/docs/users-roles) and assign user roles too.\n- **Define key goals and track conversions**: Create custom events with custom dimensions to track conversions and attribution to understand and identify the trends that matter. Includes easy ways to track outbound link clicks, file downloads and 404 error pages.\n- **Search keywords**: Integrate your dashboard with Google Search Console to get the most accurate reporting on your search keywords.\n- **SPA support**: Plausible is built with modern web frameworks in mind and it works automatically with any pushState based router on the frontend. We also support frameworks that use the URL hash for routing. See [our documentation](https://plausible.io/docs/hash-based-routing).\n- **Smooth transition from Google Analytics**: There's a realtime dashboard, entry pages report and integration with Search Console. You can track your paid campaigns and conversions. You can invite team members. You can even [import your historical Google Analytics stats](https://plausible.io/docs/google-analytics-import). Learn how to [get the most out of your Plausible experience](https://plausible.io/docs/your-plausible-experience) and join thousands who have already migrated from Google Analytics.\n\nInterested to learn more? [Read more on our website](https://plausible.io), learn more about the team and the goals of the project on [our about page](https://plausible.io/about) or explore [the documentation](https://plausible.io/docs).","title":"Why Plausible? - Introduction","ref":"readme.html#why-plausible"},{"type":"extras","doc":"Plausible Analytics is an independently owned and actively developed project. To keep the project development going, to stay in business, to continue putting effort into building a better product and to cover our costs, we need to charge a fee.\n\nGoogle Analytics is free because Google has built their company and their wealth by collecting and analyzing huge amounts of personal information from web users and using these personal and behavioral insights to sell advertisements.\n\nPlausible has no part in that business model. No personal data is being collected and analyzed either. With Plausible, you 100% own and control all of your website data. This data is not being shared with or sold to any third-parties.\n\nWe choose the subscription business model rather than the business model of surveillance capitalism. See reasons why we believe you should [stop using Google Analytics on your website](https://plausible.io/blog/remove-google-analytics).","title":"Why is Plausible Analytics Cloud not free like Google Analytics? - Introduction","ref":"readme.html#why-is-plausible-analytics-cloud-not-free-like-google-analytics"},{"type":"extras","doc":"The easiest way to get started with Plausible Analytics is with [our official managed service in the cloud](https://plausible.io/#pricing). It takes 2 minutes to start counting your stats with a worldwide CDN, high availability, backups, security and maintenance all done for you by us.\n\nIn order to be compliant with the GDPR and the Schrems II ruling, all visitor data for our managed service in the cloud is exclusively processed on servers and cloud infrastructure owned and operated by European providers. Your website data never leaves the EU.\n\nOur managed hosting can save a substantial amount of developer time and resources. For most sites this ends up being the best value option and the revenue goes to funding the maintenance and further development of Plausible. So you’ll be supporting open source software and getting a great service!\n\n#","title":"Getting started with Plausible - Introduction","ref":"readme.html#getting-started-with-plausible"},{"type":"extras","doc":"Plausible is [open source web analytics](https://plausible.io/open-source-website-analytics) and we have a free as in beer and self-hosted solution called [Plausible Community Edition (CE)](https://plausible.io/self-hosted-web-analytics). Here are the differences between Plausible Analytics managed hosting in the cloud and the Plausible CE:\n\n| | Plausible Analytics Cloud | Plausible Community Edition |\n| ------------- | ------------- | ------------- |\n| **Infrastructure management** | Easy and convenient. It takes 2 minutes to start counting your stats with a worldwide CDN, high availability, backups, security and maintenance all done for you by us. We manage everything so you don’t have to worry about anything and can focus on your stats. | You do it all yourself. You need to get a server and you need to manage your infrastructure. You are responsible for installation, maintenance, upgrades, server capacity, uptime, backup, security, stability, consistency, loading time and so on.|\n| **Release schedule** | Continuously developed and improved with new features and updates multiple times per week. | [It's a long term release](https://plausible.io/blog/building-open-source) published twice per year so latest features and improvements won't be immediately available.|\n| **Premium features** | All features available as listed in [our pricing plans](https://plausible.io/#pricing). | Selected premium features such as funnels and ecommerce revenue goals are not available as we aim to ensure a [protective barrier around our cloud offering](https://plausible.io/blog/community-edition).|\n| **Bot filtering** | Advanced bot filtering for more accurate stats. Our algorithm detects and excludes non-human traffic patterns. We also exclude known bots by the User-Agent header and filter out traffic from data centers and referrer spam domains. | Basic bot filtering that targets the most common non-human traffic based on the User-Agent header and referrer spam domains.|\n| **Server location** | All visitor data is exclusively processed on EU-owned cloud infrastructure. We keep your site data on a secure, encrypted and green energy powered server in Germany. This ensures that your site data is protected by the strict European Union data privacy laws and ensures compliance with GDPR. Your website data never leaves the EU. | You have full control and can host your instance on any server in any country that you wish. Host it on a server in your basement or host it with any cloud provider wherever you want, even those that are not GDPR compliant.|\n| **Data portability** | You see all your site stats and metrics on our modern-looking, simple to use and fast loading dashboard. You can only see the stats aggregated in the dashboard. You can download the stats using the [CSV export](https://plausible.io/docs/export-stats), [stats API](https://plausible.io/docs/stats-api) or tools such as the [Data Studio Connector](https://plausible.io/docs/integration-guides#google-data-studio). | Do you want access to the raw data? Self-hosting gives you that option. You can take the data directly from the ClickHouse database. |\n| **Premium support** | Real support delivered by real human beings who build and maintain Plausible. | Premium support is not included. CE is community supported only.|\n| **Costs** | There's a cost associated with providing an analytics service so we charge a subscription fee. We choose the subscription business model rather than the business model of surveillance capitalism. Your money funds further development of Plausible. | You need to pay for your server, CDN, backups and whatever other cost there is associated with running the infrastructure. You never have to pay any fees to us. Your money goes to 3rd party companies with no connection to us.|\n\nInterested in self-hosting Plausible CE on your server? Take a look at our [Plausible CE installation instructions](https://github.com/plausible/community-edition/).\n\nPlausible CE is a community supported project and there are no guarantees that you will get support from the creators of Plausible to troubleshoot your self-hosting issues. There is a [community supported forum](https://github.com/plausible/analytics/discussions/categories/self-hosted-support) where you can ask for help.\n\nOur only source of funding is our premium, managed service for running Plausible in the cloud.","title":"Can Plausible be self-hosted? - Introduction","ref":"readme.html#can-plausible-be-self-hosted"},{"type":"extras","doc":"Plausible Analytics is a standard Elixir/Phoenix application backed by a PostgreSQL database for general data and a Clickhouse\ndatabase for stats. On the frontend we use [TailwindCSS](https://tailwindcss.com/) for styling and React to make the dashboard interactive.","title":"Technology - Introduction","ref":"readme.html#technology"},{"type":"extras","doc":"For anyone wishing to contribute to Plausible, we recommend taking a look at [our contributor guide](https://github.com/plausible/analytics/blob/master/CONTRIBUTING.md).","title":"Contributors - Introduction","ref":"readme.html#contributors"},{"type":"extras","doc":"We welcome feedback from our community. We have a public roadmap driven by the features suggested by the community members. Take a look at our [feedback board](https://plausible.io/feedback). Please let us know if you have any requests and vote on open issues so we can better prioritize.\n\nTo stay up to date with all the latest news and product updates, make sure to follow us on [X (formerly Twitter)](https://twitter.com/plausiblehq), [LinkedIn](https://www.linkedin.com/company/plausible-analytics/) or [Mastodon](https://fosstodon.org/@plausible).","title":"Feedback & Roadmap - Introduction","ref":"readme.html#feedback-roadmap"},{"type":"extras","doc":"Plausible CE is open source under the GNU Affero General Public License Version 3 (AGPLv3) or any later version. You can [find it here](https://github.com/plausible/analytics/blob/master/LICENSE.md).\n\nTo avoid issues with AGPL virality, we've released the JavaScript tracker which gets included on your website under the MIT license. You can [find it here](https://github.com/plausible/analytics/blob/master/tracker/LICENSE.md).\n\nCopyright (c) 2018-present Plausible Insights OÜ. Plausible Analytics name and logo are trademarks of Plausible Insights OÜ. Please see our [trademark guidelines](https://plausible.io/trademark) for info on acceptable usage.","title":"License & Trademarks - Introduction","ref":"readme.html#license-trademarks"},{"type":"extras","doc":"# Contributing\n\nWe welcome everyone to contribute to Plausible. This document is to help you on setting up your environment, finding a task, and opening pull requests.","title":"Contributing","ref":"contributing.html"},{"type":"extras","doc":"The easiest way to get up and running is to [install](https://docs.docker.com/get-docker/) and use Docker for running both Postgres and Clickhouse.\n\nMake sure Docker, Elixir, Erlang and Node.js are all installed on your development machine. The [`.tool-versions`](https://github.com/plausible/analytics/blob/master/.tool-versions) file is available to use with [asdf](https://github.com/asdf-vm/asdf) or similar tools.\n\n#","title":"Development setup - Contributing","ref":"contributing.html#development-setup"},{"type":"extras","doc":"1. Run both `make postgres` and `make clickhouse`.\n2. You can set up everything with `make install`, alternatively run each command separately:\n 1. Run `mix deps.get`. This will download the required Elixir dependencies.\n 2. Run `mix ecto.create`. This will create the required databases in both Postgres and Clickhouse.\n 3. Run `mix ecto.migrate` to build the database schema.\n 4. Run `mix run priv/repo/seeds.exs` to seed the database. Check the [Seeds](#Seeds) section for more.\n 5. Run `npm ci --prefix assets` to install the required client-side dependencies.\n 6. Run `npm ci --prefix tracker` to install the required tracker dependencies.\n 7. Run `mix assets.setup` to install Tailwind and Esbuild\n 8. Run `npm run deploy --prefix tracker` to generate tracker files in `priv/tracker/js`\n 9. Run `mix download_country_database` to fetch geolocation database\n3. Run `make server` or `mix phx.server` to start the Phoenix server.\n4. The system is now available on `localhost:8000`.\n\n#","title":"Start the environment - Contributing","ref":"contributing.html#start-the-environment"},{"type":"extras","doc":"You can optionally seed your database to automatically create an account and a site with stats:\n\n1. Run `mix run priv/repo/seeds.exs` to seed the database.\n2. Start the server with `make server` and navigate to `http://localhost:8000/login`.\n3. Log in with the following e-mail and password combination: `user@plausible.test` and `plausible`.\n4. You should now have a `dummy.site` site with generated stats.\n\nAlternatively, you can manually create a new account:\n\n1. Navigate to `http://localhost:8000/register` and fill in the form.\n2. Fill in the rest of the forms and for the domain use `dummy.site`\n3. Skip the JS snippet and click start collecting data.\n4. Run `mix send_pageview` from the terminal to generate a fake pageview event for the dummy site.\n5. You should now be all set!\n\n#","title":"Seeds - Contributing","ref":"contributing.html#seeds"},{"type":"extras","doc":"1. Stop and remove the Postgres container with `make postgres-stop`.\n2. Stop and remove the Clickhouse container with `make clickhouse-stop`.\n\nVolumes are preserved. You'll find that the Postgres and Clickhouse state are retained when you bring them up again the next time: no need to re-register and so on.\n\nNote: Since we are deleting the containers, be careful when deleting volumes with `docker volume prune`. You might accidentally delete the database and would have to go through re-registration process.\n\n#","title":"Stopping Docker containers - Contributing","ref":"contributing.html#stopping-docker-containers"},{"type":"extras","doc":"`pre-commit` requires Python to be available locally and covers Elixir, JavaScript, and CSS. Set up with `pip install --user pre-commit` followed by `pre-commit install`. Conversely, if the prompts are far too bothersome, remove with `pre-commit uninstall`.","title":"Pre-commit hooks - Contributing","ref":"contributing.html#pre-commit-hooks"},{"type":"extras","doc":"Bugs can be found in our [issue tracker](https://github.com/plausible/analytics/issues). Issues are usually up for grabs.\n\nNew features need to be discussed with the core team and the community first. If you're tackling a feature, please make sure it has been already discussed in the [Discussions tab](https://github.com/plausible/analytics/discussions). We kindly ask contributors to use the discussion comment section to propose a solution before opening a pull request.\n\nPull requests without an associated issue or discussion may still be merged, but we will focus on changes that have already been talked through.","title":"Finding a task - Contributing","ref":"contributing.html#finding-a-task"}],"content_type":"text/markdown"} \ No newline at end of file diff --git a/dist/search_data-E784FFA1.js b/dist/search_data-E784FFA1.js deleted file mode 100644 index e8b975db82..0000000000 --- a/dist/search_data-E784FFA1.js +++ /dev/null @@ -1 +0,0 @@ -searchData={"items":[{"type":"task","title":"mix cancel_subscription","doc":"This task is meant to replicate the behavior of cancelling\na subscription. On production, this action is initiated by\na Paddle webhook. Currently, only the subscription status\nis changed with that action.","ref":"Mix.Tasks.CancelSubscription.html"},{"type":"function","title":"Mix.Tasks.CancelSubscription.run/1","doc":"","ref":"Mix.Tasks.CancelSubscription.html#run/1"},{"type":"task","title":"mix clean_clickhouse","doc":"","ref":"Mix.Tasks.CleanClickhouse.html"},{"type":"function","title":"Mix.Tasks.CleanClickhouse.run/1","doc":"","ref":"Mix.Tasks.CleanClickhouse.html#run/1"},{"type":"task","title":"mix create_free_subscription","doc":"","ref":"Mix.Tasks.CreateFreeSubscription.html"},{"type":"function","title":"Mix.Tasks.CreateFreeSubscription.execute/1","doc":"","ref":"Mix.Tasks.CreateFreeSubscription.html#execute/1"},{"type":"function","title":"Mix.Tasks.CreateFreeSubscription.run/1","doc":"","ref":"Mix.Tasks.CreateFreeSubscription.html#run/1"},{"type":"task","title":"mix download_country_database","doc":"This task downloads the Country Lite database from DB-IP for self-hosted or development purposes.\nPlausible Cloud runs a paid version of DB-IP with more detailed geolocation data.","ref":"Mix.Tasks.DownloadCountryDatabase.html"},{"type":"function","title":"Mix.Tasks.DownloadCountryDatabase.run/1","doc":"","ref":"Mix.Tasks.DownloadCountryDatabase.html#run/1"},{"type":"task","title":"mix generate_referrer_favicons","doc":"","ref":"Mix.Tasks.GenerateReferrerFavicons.html"},{"type":"function","title":"Mix.Tasks.GenerateReferrerFavicons.run/1","doc":"","ref":"Mix.Tasks.GenerateReferrerFavicons.html#run/1"},{"type":"task","title":"mix pull_sandbox_subscription","doc":"","ref":"Mix.Tasks.PullSandboxSubscription.html"},{"type":"function","title":"Mix.Tasks.PullSandboxSubscription.run/1","doc":"","ref":"Mix.Tasks.PullSandboxSubscription.html#run/1"},{"type":"task","title":"mix send_pageview","doc":"It's often necessary to generate fake events for development and testing purposes. This Mix Task provides a quick and easy\nway to generate a pageview or custom event, either in your development environment or a remote Plausible instance.\n\nSee Mix.Tasks.SendPageview.usage/1 for more detailed documentation.","ref":"Mix.Tasks.SendPageview.html"},{"type":"function","title":"Mix.Tasks.SendPageview.run/1","doc":"","ref":"Mix.Tasks.SendPageview.html#run/1"},{"type":"module","title":"ObanErrorReporter","doc":"","ref":"ObanErrorReporter.html"},{"type":"function","title":"ObanErrorReporter.handle_event/4","doc":"","ref":"ObanErrorReporter.html#handle_event/4"},{"type":"module","title":"Plausible","doc":"Build-related macros","ref":"Plausible.html"},{"type":"function","title":"Plausible.ce?/0","doc":"","ref":"Plausible.html#ce?/0"},{"type":"function","title":"Plausible.ee?/0","doc":"","ref":"Plausible.html#ee?/0"},{"type":"macro","title":"Plausible.on_ce/1","doc":"","ref":"Plausible.html#on_ce/1"},{"type":"macro","title":"Plausible.on_ee/1","doc":"","ref":"Plausible.html#on_ee/1"},{"type":"function","title":"Plausible.product_name/0","doc":"","ref":"Plausible.html#product_name/0"},{"type":"module","title":"Plausible.AsyncInsertRepo","doc":"Clickhouse access with async inserts enabled","ref":"Plausible.AsyncInsertRepo.html"},{"type":"function","title":"Plausible.AsyncInsertRepo.aggregate/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#aggregate/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.aggregate/4","doc":"","ref":"Plausible.AsyncInsertRepo.html#aggregate/4"},{"type":"function","title":"Plausible.AsyncInsertRepo.all/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#all/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.alter_update_all/3","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","ref":"Plausible.AsyncInsertRepo.html#alter_update_all/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.checked_out?/0","doc":"","ref":"Plausible.AsyncInsertRepo.html#checked_out?/0"},{"type":"function","title":"Plausible.AsyncInsertRepo.checkout/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#checkout/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.child_spec/1","doc":"","ref":"Plausible.AsyncInsertRepo.html#child_spec/1"},{"type":"function","title":"Plausible.AsyncInsertRepo.config/0","doc":"","ref":"Plausible.AsyncInsertRepo.html#config/0"},{"type":"function","title":"Plausible.AsyncInsertRepo.default_options/1","doc":"","ref":"Plausible.AsyncInsertRepo.html#default_options/1"},{"type":"function","title":"Plausible.AsyncInsertRepo.delete/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#delete/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.delete!/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#delete!/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.delete_all/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#delete_all/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","ref":"Plausible.AsyncInsertRepo.html#disconnect_all/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.exists?/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#exists?/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.get/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#get/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.get!/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#get!/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.get_by/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#get_by/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.get_by!/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#get_by!/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.get_dynamic_repo/0","doc":"","ref":"Plausible.AsyncInsertRepo.html#get_dynamic_repo/0"},{"type":"function","title":"Plausible.AsyncInsertRepo.insert/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#insert/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.insert!/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#insert!/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.insert_all/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#insert_all/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.insert_or_update/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#insert_or_update/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.insert_or_update!/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#insert_or_update!/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.insert_stream/3","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","ref":"Plausible.AsyncInsertRepo.html#insert_stream/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.load/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#load/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.one/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#one/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.one!/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#one!/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.preload/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#preload/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.prepare_query/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#prepare_query/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.put_dynamic_repo/1","doc":"","ref":"Plausible.AsyncInsertRepo.html#put_dynamic_repo/1"},{"type":"function","title":"Plausible.AsyncInsertRepo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","ref":"Plausible.AsyncInsertRepo.html#query/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","ref":"Plausible.AsyncInsertRepo.html#query!/3"},{"type":"function","title":"Plausible.AsyncInsertRepo.reload/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#reload/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.reload!/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#reload!/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.start_link/1","doc":"","ref":"Plausible.AsyncInsertRepo.html#start_link/1"},{"type":"function","title":"Plausible.AsyncInsertRepo.stop/1","doc":"","ref":"Plausible.AsyncInsertRepo.html#stop/1"},{"type":"function","title":"Plausible.AsyncInsertRepo.stream/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#stream/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.to_inline_sql/2","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","ref":"Plausible.AsyncInsertRepo.html#to_inline_sql/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","ref":"Plausible.AsyncInsertRepo.html#to_sql/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.update/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#update/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.update!/2","doc":"","ref":"Plausible.AsyncInsertRepo.html#update!/2"},{"type":"function","title":"Plausible.AsyncInsertRepo.update_all/3","doc":"","ref":"Plausible.AsyncInsertRepo.html#update_all/3"},{"type":"module","title":"Plausible.Auth","doc":"Functions for user authentication context.","ref":"Plausible.Auth.html"},{"type":"function","title":"Plausible.Auth.check_password/2","doc":"","ref":"Plausible.Auth.html#check_password/2"},{"type":"function","title":"Plausible.Auth.create_api_key/3","doc":"","ref":"Plausible.Auth.html#create_api_key/3"},{"type":"function","title":"Plausible.Auth.create_user/3","doc":"","ref":"Plausible.Auth.html#create_user/3"},{"type":"function","title":"Plausible.Auth.delete_api_key/2","doc":"","ref":"Plausible.Auth.html#delete_api_key/2"},{"type":"function","title":"Plausible.Auth.delete_user/1","doc":"","ref":"Plausible.Auth.html#delete_user/1"},{"type":"function","title":"Plausible.Auth.enterprise_configured?/1","doc":"","ref":"Plausible.Auth.html#enterprise_configured?/1"},{"type":"function","title":"Plausible.Auth.find_api_key/1","doc":"","ref":"Plausible.Auth.html#find_api_key/1"},{"type":"function","title":"Plausible.Auth.find_user_by/1","doc":"","ref":"Plausible.Auth.html#find_user_by/1"},{"type":"function","title":"Plausible.Auth.get_user_by/1","doc":"","ref":"Plausible.Auth.html#get_user_by/1"},{"type":"function","title":"Plausible.Auth.has_active_sites?/2","doc":"","ref":"Plausible.Auth.html#has_active_sites?/2"},{"type":"function","title":"Plausible.Auth.is_super_admin?/1","doc":"","ref":"Plausible.Auth.html#is_super_admin?/1"},{"type":"function","title":"Plausible.Auth.rate_limit/2","doc":"","ref":"Plausible.Auth.html#rate_limit/2"},{"type":"function","title":"Plausible.Auth.rate_limits/0","doc":"","ref":"Plausible.Auth.html#rate_limits/0"},{"type":"function","title":"Plausible.Auth.user_owns_sites?/1","doc":"","ref":"Plausible.Auth.html#user_owns_sites?/1"},{"type":"type","title":"Plausible.Auth.rate_limit_type/0","doc":"","ref":"Plausible.Auth.html#t:rate_limit_type/0"},{"type":"module","title":"Plausible.Auth.ApiKey","doc":"","ref":"Plausible.Auth.ApiKey.html"},{"type":"function","title":"Plausible.Auth.ApiKey.changeset/2","doc":"","ref":"Plausible.Auth.ApiKey.html#changeset/2"},{"type":"function","title":"Plausible.Auth.ApiKey.do_hash/1","doc":"","ref":"Plausible.Auth.ApiKey.html#do_hash/1"},{"type":"function","title":"Plausible.Auth.ApiKey.process_key/1","doc":"","ref":"Plausible.Auth.ApiKey.html#process_key/1"},{"type":"function","title":"Plausible.Auth.ApiKey.update/2","doc":"","ref":"Plausible.Auth.ApiKey.html#update/2"},{"type":"type","title":"Plausible.Auth.ApiKey.t/0","doc":"","ref":"Plausible.Auth.ApiKey.html#t:t/0"},{"type":"module","title":"Plausible.Auth.ApiKeyAdmin","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html"},{"type":"function","title":"Plausible.Auth.ApiKeyAdmin.create_changeset/2","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html#create_changeset/2"},{"type":"function","title":"Plausible.Auth.ApiKeyAdmin.custom_index_query/3","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html#custom_index_query/3"},{"type":"function","title":"Plausible.Auth.ApiKeyAdmin.form_fields/1","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html#form_fields/1"},{"type":"function","title":"Plausible.Auth.ApiKeyAdmin.index/1","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html#index/1"},{"type":"function","title":"Plausible.Auth.ApiKeyAdmin.search_fields/1","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html#search_fields/1"},{"type":"function","title":"Plausible.Auth.ApiKeyAdmin.update_changeset/2","doc":"","ref":"Plausible.Auth.ApiKeyAdmin.html#update_changeset/2"},{"type":"module","title":"Plausible.Auth.EmailActivationCode","doc":"Schema for email activation codes.","ref":"Plausible.Auth.EmailActivationCode.html"},{"type":"function","title":"Plausible.Auth.EmailActivationCode.generate_code/0","doc":"","ref":"Plausible.Auth.EmailActivationCode.html#generate_code/0"},{"type":"function","title":"Plausible.Auth.EmailActivationCode.new/2","doc":"","ref":"Plausible.Auth.EmailActivationCode.html#new/2"},{"type":"type","title":"Plausible.Auth.EmailActivationCode.t/0","doc":"","ref":"Plausible.Auth.EmailActivationCode.html#t:t/0"},{"type":"module","title":"Plausible.Auth.EmailVerification","doc":"API for verifying emails.","ref":"Plausible.Auth.EmailVerification.html"},{"type":"function","title":"Plausible.Auth.EmailVerification.any?/1","doc":"","ref":"Plausible.Auth.EmailVerification.html#any?/1"},{"type":"function","title":"Plausible.Auth.EmailVerification.expired?/1","doc":"","ref":"Plausible.Auth.EmailVerification.html#expired?/1"},{"type":"function","title":"Plausible.Auth.EmailVerification.issue_code/2","doc":"","ref":"Plausible.Auth.EmailVerification.html#issue_code/2"},{"type":"function","title":"Plausible.Auth.EmailVerification.verify_code/2","doc":"","ref":"Plausible.Auth.EmailVerification.html#verify_code/2"},{"type":"module","title":"Plausible.Auth.GracePeriod","doc":"This embedded schema stores information about the account locking grace\nperiod.\n\nUsers are given this 7-day grace period to upgrade their account after\noutgrowing their subscriptions. The actual account locking happens in\nbackground with `Plausible.Workers.LockSites`.\n\nThe grace period can also be manual, without an end date, being controlled\nmanually from the CRM, and not by the background site locker job. This is\nuseful for enterprise subscriptions.","ref":"Plausible.Auth.GracePeriod.html"},{"type":"function","title":"Plausible.Auth.GracePeriod.active?/1","doc":"Returns whether the grace period is still active for a User. Defaults to\nfalse if the user is nil or there is no grace period.","ref":"Plausible.Auth.GracePeriod.html#active?/1"},{"type":"function","title":"Plausible.Auth.GracePeriod.end_changeset/1","doc":"Ends an existing grace period by `setting users.grace_period.is_over` to true.\nThis means the grace period has expired.","ref":"Plausible.Auth.GracePeriod.html#end_changeset/1"},{"type":"function","title":"Plausible.Auth.GracePeriod.expired?/1","doc":"Returns whether the grace period has already expired for a User. Defaults to\nfalse if the user is nil or there is no grace period.","ref":"Plausible.Auth.GracePeriod.html#expired?/1"},{"type":"function","title":"Plausible.Auth.GracePeriod.remove_changeset/1","doc":"Removes the grace period from the User completely.","ref":"Plausible.Auth.GracePeriod.html#remove_changeset/1"},{"type":"function","title":"Plausible.Auth.GracePeriod.start_changeset/1","doc":"Starts a account locking grace period of 7 days by changing the User struct.","ref":"Plausible.Auth.GracePeriod.html#start_changeset/1"},{"type":"function","title":"Plausible.Auth.GracePeriod.start_manual_lock_changeset/1","doc":"Starts a manual account locking grace period by changing the User struct.\nManual locking means the grace period can only be removed manually from the\nCRM.","ref":"Plausible.Auth.GracePeriod.html#start_manual_lock_changeset/1"},{"type":"type","title":"Plausible.Auth.GracePeriod.t/0","doc":"","ref":"Plausible.Auth.GracePeriod.html#t:t/0"},{"type":"module","title":"Plausible.Auth.Invitation","doc":"","ref":"Plausible.Auth.Invitation.html"},{"type":"function","title":"Plausible.Auth.Invitation.new/1","doc":"","ref":"Plausible.Auth.Invitation.html#new/1"},{"type":"type","title":"Plausible.Auth.Invitation.t/0","doc":"","ref":"Plausible.Auth.Invitation.html#t:t/0"},{"type":"module","title":"Plausible.Auth.Password","doc":"","ref":"Plausible.Auth.Password.html"},{"type":"function","title":"Plausible.Auth.Password.dummy_calculation/0","doc":"","ref":"Plausible.Auth.Password.html#dummy_calculation/0"},{"type":"function","title":"Plausible.Auth.Password.hash/1","doc":"","ref":"Plausible.Auth.Password.html#hash/1"},{"type":"function","title":"Plausible.Auth.Password.match?/2","doc":"","ref":"Plausible.Auth.Password.html#match?/2"},{"type":"module","title":"Plausible.Auth.TOTP","doc":"TOTP auth context\n\nHandles all the aspects of TOTP setup, management and validation for users.","ref":"Plausible.Auth.TOTP.html"},{"type":"module","title":"Setup - Plausible.Auth.TOTP","doc":"TOTP setup is started with `initiate/1`. At this stage, a random secret\nbinary is generated for user and stored under `User.totp_secret`. The secret\nis additionally encrypted while stored in the database using `Cloak`. The\nvault for safe storage is configured in `Plausible.Auth.TOTP.Vault` via\na dedicated `Ecto` type defined in `Plausible.Auth.TOTP.EncryptedBinary`.\nThe function returns updated user along with TOTP URI and a readable form\nof secret. Both - the URI and readable secret - are meant for exposure\nin the user's setup screen. The URI should be encoded as a QR code.\n\nAfter initiation, user is expected to confirm valid setup with `enable/2`,\nproviding TOTP code from their authenticator app. After code validation\npasses successfully, the `User.totp_enabled` flag is set to `true`.\nFinally, the user must be immediately presented with a list of recovery codes\nreturned by the same call of `enable/2`. The codes should be presented\nin copy/paste friendly form, ideally also with a print-friendly view option.\n\nThe `initiate/1` and `enable/1` functions can be safely called multiple\ntimes, allowing user to abort and restart setup up to these stages.","ref":"Plausible.Auth.TOTP.html#module-setup"},{"type":"module","title":"Management - Plausible.Auth.TOTP","doc":"The state of TOTP for a particular user can be chcecked by calling\n`enabled?/1` or `initiated?/1`.\n\nTOTP can be disabled with `disable/2`. User is expected to provide their\ncurrent password for safety. Once disabled, all TOTP user settings are\ncleared and any remaining generated recovery codes are removed. The function\ncan be safely run more than once. There's also alternative call for forced\ndisabling of TOTP for a given user without sending any notification,\n`force_disable/1`. It's meant for use in situation where user lost both,\n2FA device and recovery codes and their identity is verified independently.\n\nIf the user needs to regenerate the recovery codes outside of setup procedure,\nthey must do it via `generate_recovery_codes/2`, providing their current\npassword for safety. They must be warned that any existing recovery codes\nwill be invalidated.","ref":"Plausible.Auth.TOTP.html#module-management"},{"type":"module","title":"Validation - Plausible.Auth.TOTP","doc":"After logging in, user's TOTP state must be checked with `enabled?/1`.\n\nIf enabled, user must be presented with TOTP code input form accepting\n6 digit characters. The code must be checked using `validate_code/2`.\n\nUser must have an option to alternatively input one of their recovery\ncodes. Those codes must be checked with `use_recovery_code/2`.","ref":"Plausible.Auth.TOTP.html#module-validation"},{"type":"module","title":"Code validity - Plausible.Auth.TOTP","doc":"In case of TOTP codes, a grace period of 30 seconds is applied, which\nallows user to use their current and previous TOTP code, assuming 30\nsecond validity window of each. This allows user to use code that was\nabout to expire before the submission. Regardless of that, each TOTP\ncode can be used only once. Validation procedure rejects repeat use\nof the same code for safety. It's done by tracking last time a TOTP\ncode was used successfully, stored under `User.totp_last_used_at`.\n\nIn case of recovery codes, each code is deleted immediately after use.\nThey are strictly one-time use only.","ref":"Plausible.Auth.TOTP.html#module-code-validity"},{"type":"module","title":"TOTP Token - Plausible.Auth.TOTP","doc":"TOTP token is an alternate method of authenticating user session.\nIt's main use case is \"trust this device\" functionality, where user\ncan decide to skip 2FA verification for a particular browser session\nfor next N days. The token should then be stored in an encrypted,\nsigned cookie with a proper expiration timestamp.\n\nThe token should be reset each time it either fails to match\nor when other credentials (like password) are reset. This should\neffectively invalidate all trusted devices for a given user.","ref":"Plausible.Auth.TOTP.html#module-totp-token"},{"type":"function","title":"Plausible.Auth.TOTP.disable/2","doc":"","ref":"Plausible.Auth.TOTP.html#disable/2"},{"type":"function","title":"Plausible.Auth.TOTP.enable/3","doc":"","ref":"Plausible.Auth.TOTP.html#enable/3"},{"type":"function","title":"Plausible.Auth.TOTP.enabled?/1","doc":"","ref":"Plausible.Auth.TOTP.html#enabled?/1"},{"type":"function","title":"Plausible.Auth.TOTP.force_disable/1","doc":"","ref":"Plausible.Auth.TOTP.html#force_disable/1"},{"type":"function","title":"Plausible.Auth.TOTP.generate_recovery_codes/1","doc":"","ref":"Plausible.Auth.TOTP.html#generate_recovery_codes/1"},{"type":"function","title":"Plausible.Auth.TOTP.generate_recovery_codes/2","doc":"","ref":"Plausible.Auth.TOTP.html#generate_recovery_codes/2"},{"type":"function","title":"Plausible.Auth.TOTP.initiate/1","doc":"","ref":"Plausible.Auth.TOTP.html#initiate/1"},{"type":"function","title":"Plausible.Auth.TOTP.initiated?/1","doc":"","ref":"Plausible.Auth.TOTP.html#initiated?/1"},{"type":"function","title":"Plausible.Auth.TOTP.reset_token/1","doc":"","ref":"Plausible.Auth.TOTP.html#reset_token/1"},{"type":"function","title":"Plausible.Auth.TOTP.use_recovery_code/2","doc":"","ref":"Plausible.Auth.TOTP.html#use_recovery_code/2"},{"type":"function","title":"Plausible.Auth.TOTP.validate_code/3","doc":"","ref":"Plausible.Auth.TOTP.html#validate_code/3"},{"type":"module","title":"Plausible.Auth.TOTP.EncryptedBinary","doc":"Defines an Ecto type so Cloak.Ecto can encrypt/decrypt a binary field.","ref":"Plausible.Auth.TOTP.EncryptedBinary.html"},{"type":"module","title":"Plausible.Auth.TOTP.RecoveryCode","doc":"Schema for TOTP recovery codes.","ref":"Plausible.Auth.TOTP.RecoveryCode.html"},{"type":"function","title":"Plausible.Auth.TOTP.RecoveryCode.changeset/2","doc":"","ref":"Plausible.Auth.TOTP.RecoveryCode.html#changeset/2"},{"type":"function","title":"Plausible.Auth.TOTP.RecoveryCode.changeset_to_map/2","doc":"","ref":"Plausible.Auth.TOTP.RecoveryCode.html#changeset_to_map/2"},{"type":"function","title":"Plausible.Auth.TOTP.RecoveryCode.generate_codes/1","doc":"Generates `count` unique recovery codes, each alphanumeric\nand 10 characters long.","ref":"Plausible.Auth.TOTP.RecoveryCode.html#generate_codes/1"},{"type":"function","title":"Plausible.Auth.TOTP.RecoveryCode.match?/2","doc":"","ref":"Plausible.Auth.TOTP.RecoveryCode.html#match?/2"},{"type":"type","title":"Plausible.Auth.TOTP.RecoveryCode.t/0","doc":"","ref":"Plausible.Auth.TOTP.RecoveryCode.html#t:t/0"},{"type":"module","title":"Plausible.Auth.TOTP.Vault","doc":"Provides a vault that will be used to encrypt/decrypt the TOTP secrets of users who enable it.","ref":"Plausible.Auth.TOTP.Vault.html"},{"type":"function","title":"Plausible.Auth.TOTP.Vault.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.Auth.TOTP.Vault.html#child_spec/1"},{"type":"function","title":"Plausible.Auth.TOTP.Vault.start_link/1","doc":"","ref":"Plausible.Auth.TOTP.Vault.html#start_link/1"},{"type":"module","title":"Plausible.Auth.Token","doc":"","ref":"Plausible.Auth.Token.html"},{"type":"function","title":"Plausible.Auth.Token.sign_password_reset/1","doc":"","ref":"Plausible.Auth.Token.html#sign_password_reset/1"},{"type":"function","title":"Plausible.Auth.Token.sign_shared_link/1","doc":"","ref":"Plausible.Auth.Token.html#sign_shared_link/1"},{"type":"function","title":"Plausible.Auth.Token.verify_password_reset/1","doc":"","ref":"Plausible.Auth.Token.html#verify_password_reset/1"},{"type":"function","title":"Plausible.Auth.Token.verify_shared_link/1","doc":"","ref":"Plausible.Auth.Token.html#verify_shared_link/1"},{"type":"module","title":"Plausible.Auth.User","doc":"","ref":"Plausible.Auth.User.html"},{"type":"function","title":"Plausible.Auth.User.cancel_email_changeset/1","doc":"","ref":"Plausible.Auth.User.html#cancel_email_changeset/1"},{"type":"function","title":"Plausible.Auth.User.changeset/2","doc":"","ref":"Plausible.Auth.User.html#changeset/2"},{"type":"function","title":"Plausible.Auth.User.email_changeset/2","doc":"","ref":"Plausible.Auth.User.html#email_changeset/2"},{"type":"function","title":"Plausible.Auth.User.end_trial/1","doc":"","ref":"Plausible.Auth.User.html#end_trial/1"},{"type":"function","title":"Plausible.Auth.User.hash_password/1","doc":"","ref":"Plausible.Auth.User.html#hash_password/1"},{"type":"function","title":"Plausible.Auth.User.new/1","doc":"","ref":"Plausible.Auth.User.html#new/1"},{"type":"function","title":"Plausible.Auth.User.password_strength/1","doc":"","ref":"Plausible.Auth.User.html#password_strength/1"},{"type":"function","title":"Plausible.Auth.User.profile_img_url/1","doc":"","ref":"Plausible.Auth.User.html#profile_img_url/1"},{"type":"function","title":"Plausible.Auth.User.remove_trial_expiry/1","doc":"","ref":"Plausible.Auth.User.html#remove_trial_expiry/1"},{"type":"function","title":"Plausible.Auth.User.set_password/2","doc":"","ref":"Plausible.Auth.User.html#set_password/2"},{"type":"function","title":"Plausible.Auth.User.settings_changeset/2","doc":"","ref":"Plausible.Auth.User.html#settings_changeset/2"},{"type":"function","title":"Plausible.Auth.User.start_trial/1","doc":"","ref":"Plausible.Auth.User.html#start_trial/1"},{"type":"function","title":"Plausible.Auth.User.subscription_accept_traffic_until_offset_days/0","doc":"","ref":"Plausible.Auth.User.html#subscription_accept_traffic_until_offset_days/0"},{"type":"function","title":"Plausible.Auth.User.trial_accept_traffic_until_offset_days/0","doc":"","ref":"Plausible.Auth.User.html#trial_accept_traffic_until_offset_days/0"},{"type":"type","title":"Plausible.Auth.User.t/0","doc":"","ref":"Plausible.Auth.User.html#t:t/0"},{"type":"module","title":"Plausible.Auth.UserAdmin","doc":"","ref":"Plausible.Auth.UserAdmin.html"},{"type":"function","title":"Plausible.Auth.UserAdmin.custom_index_query/3","doc":"","ref":"Plausible.Auth.UserAdmin.html#custom_index_query/3"},{"type":"function","title":"Plausible.Auth.UserAdmin.delete/2","doc":"","ref":"Plausible.Auth.UserAdmin.html#delete/2"},{"type":"function","title":"Plausible.Auth.UserAdmin.disable_2fa/1","doc":"","ref":"Plausible.Auth.UserAdmin.html#disable_2fa/1"},{"type":"function","title":"Plausible.Auth.UserAdmin.form_fields/1","doc":"","ref":"Plausible.Auth.UserAdmin.html#form_fields/1"},{"type":"function","title":"Plausible.Auth.UserAdmin.index/1","doc":"","ref":"Plausible.Auth.UserAdmin.html#index/1"},{"type":"function","title":"Plausible.Auth.UserAdmin.resource_actions/1","doc":"","ref":"Plausible.Auth.UserAdmin.html#resource_actions/1"},{"type":"module","title":"Plausible.Auth.UserSession","doc":"Schema for storing user session data.","ref":"Plausible.Auth.UserSession.html"},{"type":"function","title":"Plausible.Auth.UserSession.new_session/3","doc":"","ref":"Plausible.Auth.UserSession.html#new_session/3"},{"type":"function","title":"Plausible.Auth.UserSession.timeout_duration/0","doc":"","ref":"Plausible.Auth.UserSession.html#timeout_duration/0"},{"type":"function","title":"Plausible.Auth.UserSession.touch_session/2","doc":"","ref":"Plausible.Auth.UserSession.html#touch_session/2"},{"type":"type","title":"Plausible.Auth.UserSession.t/0","doc":"","ref":"Plausible.Auth.UserSession.html#t:t/0"},{"type":"module","title":"Plausible.Billing","doc":"","ref":"Plausible.Billing.html"},{"type":"function","title":"Plausible.Billing.active_subscription_for/1","doc":"","ref":"Plausible.Billing.html#active_subscription_for/1"},{"type":"function","title":"Plausible.Billing.cancelled_subscription_notice_dismiss_id/1","doc":"","ref":"Plausible.Billing.html#cancelled_subscription_notice_dismiss_id/1"},{"type":"function","title":"Plausible.Billing.change_plan/2","doc":"","ref":"Plausible.Billing.html#change_plan/2"},{"type":"function","title":"Plausible.Billing.change_plan_preview/2","doc":"","ref":"Plausible.Billing.html#change_plan_preview/2"},{"type":"function","title":"Plausible.Billing.check_needs_to_upgrade/1","doc":"","ref":"Plausible.Billing.html#check_needs_to_upgrade/1"},{"type":"function","title":"Plausible.Billing.format_price/1","doc":"","ref":"Plausible.Billing.html#format_price/1"},{"type":"function","title":"Plausible.Billing.has_active_subscription?/1","doc":"","ref":"Plausible.Billing.html#has_active_subscription?/1"},{"type":"function","title":"Plausible.Billing.paddle_api/0","doc":"","ref":"Plausible.Billing.html#paddle_api/0"},{"type":"function","title":"Plausible.Billing.subscription_cancelled/1","doc":"","ref":"Plausible.Billing.html#subscription_cancelled/1"},{"type":"function","title":"Plausible.Billing.subscription_created/1","doc":"","ref":"Plausible.Billing.html#subscription_created/1"},{"type":"function","title":"Plausible.Billing.subscription_payment_succeeded/1","doc":"","ref":"Plausible.Billing.html#subscription_payment_succeeded/1"},{"type":"function","title":"Plausible.Billing.subscription_updated/1","doc":"","ref":"Plausible.Billing.html#subscription_updated/1"},{"type":"module","title":"Plausible.Billing.Ecto.Feature","doc":"Ecto type representing a feature. Features are cast and stored in the\ndatabase as strings and loaded as modules, for example: `\"props\"` is loaded\nas `Plausible.Billing.Feature.Props`.","ref":"Plausible.Billing.Ecto.Feature.html"},{"type":"function","title":"Plausible.Billing.Ecto.Feature.cast/1","doc":"","ref":"Plausible.Billing.Ecto.Feature.html#cast/1"},{"type":"function","title":"Plausible.Billing.Ecto.Feature.dump/1","doc":"","ref":"Plausible.Billing.Ecto.Feature.html#dump/1"},{"type":"function","title":"Plausible.Billing.Ecto.Feature.embed_as/1","doc":"","ref":"Plausible.Billing.Ecto.Feature.html#embed_as/1"},{"type":"function","title":"Plausible.Billing.Ecto.Feature.equal?/2","doc":"","ref":"Plausible.Billing.Ecto.Feature.html#equal?/2"},{"type":"function","title":"Plausible.Billing.Ecto.Feature.load/1","doc":"","ref":"Plausible.Billing.Ecto.Feature.html#load/1"},{"type":"function","title":"Plausible.Billing.Ecto.Feature.type/0","doc":"","ref":"Plausible.Billing.Ecto.Feature.html#type/0"},{"type":"module","title":"Plausible.Billing.Ecto.FeatureList","doc":"Ecto type representing a list of features. This is a proxy for \n`{:array, Plausible.Billing.Ecto.Feature}` and is required for Kaffy to\nrender the HTML input correctly.","ref":"Plausible.Billing.Ecto.FeatureList.html"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.cast/1","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#cast/1"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.dump/1","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#dump/1"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.embed_as/1","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#embed_as/1"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.equal?/2","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#equal?/2"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.load/1","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#load/1"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.render_form/5","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#render_form/5"},{"type":"function","title":"Plausible.Billing.Ecto.FeatureList.type/0","doc":"","ref":"Plausible.Billing.Ecto.FeatureList.html#type/0"},{"type":"module","title":"Plausible.Billing.Ecto.Limit","doc":"Ecto type representing a limit, that can be either a number or unlimited.\nUnlimited is dumped to the database as `-1` and loaded as `:unlimited` to\nkeep compatibility with the rest of the codebase.","ref":"Plausible.Billing.Ecto.Limit.html"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.cast/1","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#cast/1"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.dump/1","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#dump/1"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.embed_as/1","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#embed_as/1"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.equal?/2","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#equal?/2"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.load/1","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#load/1"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.render_form/5","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#render_form/5"},{"type":"function","title":"Plausible.Billing.Ecto.Limit.type/0","doc":"","ref":"Plausible.Billing.Ecto.Limit.html#type/0"},{"type":"module","title":"Plausible.Billing.EnterprisePlan","doc":"","ref":"Plausible.Billing.EnterprisePlan.html"},{"type":"function","title":"Plausible.Billing.EnterprisePlan.changeset/2","doc":"","ref":"Plausible.Billing.EnterprisePlan.html#changeset/2"},{"type":"module","title":"Plausible.Billing.EnterprisePlanAdmin","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html"},{"type":"function","title":"Plausible.Billing.EnterprisePlanAdmin.create_changeset/2","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html#create_changeset/2"},{"type":"function","title":"Plausible.Billing.EnterprisePlanAdmin.custom_index_query/3","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html#custom_index_query/3"},{"type":"function","title":"Plausible.Billing.EnterprisePlanAdmin.form_fields/1","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html#form_fields/1"},{"type":"function","title":"Plausible.Billing.EnterprisePlanAdmin.index/1","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html#index/1"},{"type":"function","title":"Plausible.Billing.EnterprisePlanAdmin.search_fields/1","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html#search_fields/1"},{"type":"function","title":"Plausible.Billing.EnterprisePlanAdmin.update_changeset/2","doc":"","ref":"Plausible.Billing.EnterprisePlanAdmin.html#update_changeset/2"},{"type":"behaviour","title":"Plausible.Billing.Feature","doc":"This module provides an interface for managing features, e.g. Revenue Goals,\nFunnels and Custom Properties.\n\nFeature modules have functions for toggling the feature on/off and checking\nwhether the feature is available for a site/user.\n\nWhen defining new features, the following options are expected by the\n`__using__` macro:\n\n * `:name` - an atom representing the feature name in the plan JSON\n file (see also Plausible.Billing.Plan).\n\n * `:display_name` - human-readable display name of the feature\n\n * `:toggle_field` - the field in the %Plausible.Site{} schema that toggles\n the feature. If `nil` or not set, toggle/2 silently returns `:ok`\n\n * `:free` - if set to `true`, makes the `check_availability/1` function\n always return `:ok` (no matter the user's subscription status)\n\nFunctions defined by `__using__` can be overridden if needed.","ref":"Plausible.Billing.Feature.html"},{"type":"callback","title":"Plausible.Billing.Feature.check_availability/1","doc":"Checks whether the site owner or the user plan includes the given feature.","ref":"Plausible.Billing.Feature.html#c:check_availability/1"},{"type":"callback","title":"Plausible.Billing.Feature.display_name/0","doc":"Returns the human-readable display name of the feature.","ref":"Plausible.Billing.Feature.html#c:display_name/0"},{"type":"callback","title":"Plausible.Billing.Feature.enabled?/1","doc":"Checks whether a feature is enabled or not. Returns false when the feature is\ndisabled or the user does not have access to it.","ref":"Plausible.Billing.Feature.html#c:enabled?/1"},{"type":"callback","title":"Plausible.Billing.Feature.free?/0","doc":"Returns whether the feature is free to use or not.","ref":"Plausible.Billing.Feature.html#c:free?/0"},{"type":"function","title":"Plausible.Billing.Feature.list/0","doc":"Lists all available feature modules.","ref":"Plausible.Billing.Feature.html#list/0"},{"type":"macro","title":"Plausible.Billing.Feature.list_short_names/0","doc":"Lists all the feature short names, e.g. RevenueGoals","ref":"Plausible.Billing.Feature.html#list_short_names/0"},{"type":"callback","title":"Plausible.Billing.Feature.name/0","doc":"Returns the atom representing the feature name in the plan JSON file.","ref":"Plausible.Billing.Feature.html#c:name/0"},{"type":"callback","title":"Plausible.Billing.Feature.opted_out?/1","doc":"Returns whether the site explicitly opted out of the feature. This function\nis different from enabled/1, because enabled/1 returns false when the site\nowner does not have access to the feature.","ref":"Plausible.Billing.Feature.html#c:opted_out?/1"},{"type":"callback","title":"Plausible.Billing.Feature.toggle/2","doc":"Toggles the feature on and off for a site. Returns\n`{:error, :upgrade_required}` when toggling a feature the site owner does not\nhave access to.","ref":"Plausible.Billing.Feature.html#c:toggle/2"},{"type":"callback","title":"Plausible.Billing.Feature.toggle_field/0","doc":"Returns the %Plausible.Site{} field that toggles the feature on and off.","ref":"Plausible.Billing.Feature.html#c:toggle_field/0"},{"type":"type","title":"Plausible.Billing.Feature.t/0","doc":"","ref":"Plausible.Billing.Feature.html#t:t/0"},{"type":"module","title":"Plausible.Billing.PaddleApi","doc":"","ref":"Plausible.Billing.PaddleApi.html"},{"type":"function","title":"Plausible.Billing.PaddleApi.checkout_domain/0","doc":"","ref":"Plausible.Billing.PaddleApi.html#checkout_domain/0"},{"type":"function","title":"Plausible.Billing.PaddleApi.fetch_prices/2","doc":"","ref":"Plausible.Billing.PaddleApi.html#fetch_prices/2"},{"type":"function","title":"Plausible.Billing.PaddleApi.get_invoices/1","doc":"","ref":"Plausible.Billing.PaddleApi.html#get_invoices/1"},{"type":"function","title":"Plausible.Billing.PaddleApi.get_subscription/1","doc":"","ref":"Plausible.Billing.PaddleApi.html#get_subscription/1"},{"type":"function","title":"Plausible.Billing.PaddleApi.update_subscription/2","doc":"","ref":"Plausible.Billing.PaddleApi.html#update_subscription/2"},{"type":"function","title":"Plausible.Billing.PaddleApi.update_subscription_preview/2","doc":"","ref":"Plausible.Billing.PaddleApi.html#update_subscription_preview/2"},{"type":"function","title":"Plausible.Billing.PaddleApi.vendors_domain/0","doc":"","ref":"Plausible.Billing.PaddleApi.html#vendors_domain/0"},{"type":"module","title":"Plausible.Billing.Plans","doc":"","ref":"Plausible.Billing.Plans.html"},{"type":"function","title":"Plausible.Billing.Plans.all/0","doc":"","ref":"Plausible.Billing.Plans.html#all/0"},{"type":"function","title":"Plausible.Billing.Plans.available_plans_for/2","doc":"","ref":"Plausible.Billing.Plans.html#available_plans_for/2"},{"type":"function","title":"Plausible.Billing.Plans.business_plans_for/1","doc":"","ref":"Plausible.Billing.Plans.html#business_plans_for/1"},{"type":"function","title":"Plausible.Billing.Plans.business_tier?/1","doc":"","ref":"Plausible.Billing.Plans.html#business_tier?/1"},{"type":"function","title":"Plausible.Billing.Plans.business_tier_launch/0","doc":"","ref":"Plausible.Billing.Plans.html#business_tier_launch/0"},{"type":"function","title":"Plausible.Billing.Plans.find/1","doc":"","ref":"Plausible.Billing.Plans.html#find/1"},{"type":"function","title":"Plausible.Billing.Plans.get_price_for/2","doc":"","ref":"Plausible.Billing.Plans.html#get_price_for/2"},{"type":"function","title":"Plausible.Billing.Plans.get_regular_plan/2","doc":"","ref":"Plausible.Billing.Plans.html#get_regular_plan/2"},{"type":"function","title":"Plausible.Billing.Plans.get_subscription_plan/1","doc":"","ref":"Plausible.Billing.Plans.html#get_subscription_plan/1"},{"type":"function","title":"Plausible.Billing.Plans.growth_plans_for/1","doc":"Returns a list of growth plans available for the user to choose.\n\nAs new versions of plans are introduced, users who were on old plans can\nstill choose from old plans.","ref":"Plausible.Billing.Plans.html#growth_plans_for/1"},{"type":"function","title":"Plausible.Billing.Plans.latest_enterprise_plan_with_price/2","doc":"","ref":"Plausible.Billing.Plans.html#latest_enterprise_plan_with_price/2"},{"type":"function","title":"Plausible.Billing.Plans.subscription_interval/1","doc":"","ref":"Plausible.Billing.Plans.html#subscription_interval/1"},{"type":"function","title":"Plausible.Billing.Plans.suggest/2","doc":"Returns the most appropriate plan for a user based on their usage during a\ngiven cycle.\n\nIf the usage during the cycle exceeds the enterprise-level threshold, or if\nthe user already belongs to an enterprise plan, it suggests the :enterprise\nplan.\n\nOtherwise, it recommends the plan where the cycle usage falls just under the\nplan's limit from the available options for the user.","ref":"Plausible.Billing.Plans.html#suggest/2"},{"type":"function","title":"Plausible.Billing.Plans.with_prices/2","doc":"This function takes a list of plans as an argument, gathers all product\nIDs in a single list, and makes an API call to Paddle. After a successful\nresponse, fills in the `monthly_cost` and `yearly_cost` fields for each\ngiven plan and returns the new list of plans with completed information.","ref":"Plausible.Billing.Plans.html#with_prices/2"},{"type":"function","title":"Plausible.Billing.Plans.yearly_product_ids/0","doc":"List yearly plans product IDs.","ref":"Plausible.Billing.Plans.html#yearly_product_ids/0"},{"type":"module","title":"Plausible.Billing.Quota","doc":"This module provides functions to work with plans usage and limits.","ref":"Plausible.Billing.Quota.html"},{"type":"function","title":"Plausible.Billing.Quota.below_limit?/2","doc":"Returns whether the usage is below the limit or not.\nReturns false if usage is equal to the limit.","ref":"Plausible.Billing.Quota.html#below_limit?/2"},{"type":"function","title":"Plausible.Billing.Quota.eligible_for_upgrade?/1","doc":"","ref":"Plausible.Billing.Quota.html#eligible_for_upgrade?/1"},{"type":"function","title":"Plausible.Billing.Quota.ensure_can_add_new_site/1","doc":"Enterprise plans are always allowed to add more sites (even when\nover limit) to avoid service disruption. Their usage is checked\nin a background job instead (see `check_usage.ex`).","ref":"Plausible.Billing.Quota.html#ensure_can_add_new_site/1"},{"type":"function","title":"Plausible.Billing.Quota.ensure_feature_access/2","doc":"","ref":"Plausible.Billing.Quota.html#ensure_feature_access/2"},{"type":"function","title":"Plausible.Billing.Quota.ensure_within_plan_limits/3","doc":"Ensures that the given user (or the usage map) is within the limits\nof the given plan.\n\nAn `opts` argument can be passed with `ignore_pageview_limit: true`\nwhich bypasses the pageview limit check and returns `:ok` as long as\nthe other limits are not exceeded.","ref":"Plausible.Billing.Quota.html#ensure_within_plan_limits/3"},{"type":"function","title":"Plausible.Billing.Quota.exceeded_cycles/2","doc":"","ref":"Plausible.Billing.Quota.html#exceeded_cycles/2"},{"type":"function","title":"Plausible.Billing.Quota.exceeds_last_two_usage_cycles?/2","doc":"","ref":"Plausible.Billing.Quota.html#exceeds_last_two_usage_cycles?/2"},{"type":"function","title":"Plausible.Billing.Quota.suggest_tier/3","doc":"Suggests a suitable tier (Growth or Business) for the given usage map.\n\nIf even the highest Business plan does not accommodate the usage, then\n`:custom` is returned. This means that this kind of usage should get on\na custom plan.\n\n`nil` is returned if the usage is not eligible for upgrade.","ref":"Plausible.Billing.Quota.html#suggest_tier/3"},{"type":"function","title":"Plausible.Billing.Quota.within_limit?/2","doc":"Returns whether the usage is within the limit or not.\nReturns true if usage is equal to the limit.","ref":"Plausible.Billing.Quota.html#within_limit?/2"},{"type":"module","title":"Plausible.Billing.SiteLocker","doc":"","ref":"Plausible.Billing.SiteLocker.html"},{"type":"function","title":"Plausible.Billing.SiteLocker.send_grace_period_end_email/1","doc":"","ref":"Plausible.Billing.SiteLocker.html#send_grace_period_end_email/1"},{"type":"function","title":"Plausible.Billing.SiteLocker.set_lock_status_for/2","doc":"","ref":"Plausible.Billing.SiteLocker.html#set_lock_status_for/2"},{"type":"function","title":"Plausible.Billing.SiteLocker.update_sites_for/2","doc":"","ref":"Plausible.Billing.SiteLocker.html#update_sites_for/2"},{"type":"type","title":"Plausible.Billing.SiteLocker.lock_reason/0","doc":"","ref":"Plausible.Billing.SiteLocker.html#t:lock_reason/0"},{"type":"type","title":"Plausible.Billing.SiteLocker.update_opt/0","doc":"","ref":"Plausible.Billing.SiteLocker.html#t:update_opt/0"},{"type":"module","title":"Plausible.Billing.Subscription.Status","doc":"The subscription statuses are stored in Paddle. They can only be changed\nthrough Paddle webhooks, which always send the current subscription status\nvia the payload.\n\n* `active` - All good with the payments. Can access stats.\n\n* `past_due` - The payment has failed, but we're trying to charge the customer\n again. Access to stats is still granted. There will be three retries - after\n 3, 5, and 7 days have passed from the first failure. After a failure on the\n final retry, the subscription status will change to `paused`. As soon as the\n customer updates their billing details, Paddle will charge them again, and\n after a successful payment, the subscription will become `active` again.\n\n* `paused` - we've tried to charge the customer but all the retries have failed.\n Stats access restricted. As soon as the customer updates their billing details,\n Paddle will charge them again, and after a successful payment, the subscription\n will become `active` again.\n\n* `deleted` - The customer has triggered the cancel subscription action. Access\n to stats should be granted for the time the customer has already paid for. If\n they want to upgrade again, new billing details have to be provided.\n\nPaddle documentation links for reference:\n\n* Subscription statuses -\n https://developer.paddle.com/classic/reference/zg9joji1mzu0mdi2-subscription-status-reference\n\n* Payment failures -\n https://developer.paddle.com/classic/guides/zg9joji1mzu0mduy-payment-failures","ref":"Plausible.Billing.Subscription.Status.html"},{"type":"macro","title":"Plausible.Billing.Subscription.Status.active/0","doc":"","ref":"Plausible.Billing.Subscription.Status.html#active/0"},{"type":"function","title":"Plausible.Billing.Subscription.Status.active?/1","doc":"","ref":"Plausible.Billing.Subscription.Status.html#active?/1"},{"type":"macro","title":"Plausible.Billing.Subscription.Status.deleted/0","doc":"","ref":"Plausible.Billing.Subscription.Status.html#deleted/0"},{"type":"function","title":"Plausible.Billing.Subscription.Status.deleted?/1","doc":"","ref":"Plausible.Billing.Subscription.Status.html#deleted?/1"},{"type":"macro","title":"Plausible.Billing.Subscription.Status.in?/2","doc":"","ref":"Plausible.Billing.Subscription.Status.html#in?/2"},{"type":"macro","title":"Plausible.Billing.Subscription.Status.past_due/0","doc":"","ref":"Plausible.Billing.Subscription.Status.html#past_due/0"},{"type":"function","title":"Plausible.Billing.Subscription.Status.past_due?/1","doc":"","ref":"Plausible.Billing.Subscription.Status.html#past_due?/1"},{"type":"macro","title":"Plausible.Billing.Subscription.Status.paused/0","doc":"","ref":"Plausible.Billing.Subscription.Status.html#paused/0"},{"type":"function","title":"Plausible.Billing.Subscription.Status.paused?/1","doc":"","ref":"Plausible.Billing.Subscription.Status.html#paused?/1"},{"type":"function","title":"Plausible.Billing.Subscription.Status.valid_statuses/0","doc":"","ref":"Plausible.Billing.Subscription.Status.html#valid_statuses/0"},{"type":"type","title":"Plausible.Billing.Subscription.Status.status/0","doc":"","ref":"Plausible.Billing.Subscription.Status.html#t:status/0"},{"type":"behaviour","title":"Plausible.Cache","doc":"Caching interface specific for Plausible. Usage:\n\n use Plausible.Cache\n\n # - Implement the callbacks required\n # - Optionally override `unwrap_cache_keys/1`\n # - Populate the cache with `Plausible.Cache.Warmer`\n\nServes as a wrapper around `Plausible.Cache.Adapter`, where the underlying\nimplementation can be transparently swapped.\n\nEven though normally the relevant Adapter processes are started, cache access is disabled\nduring tests via the `:plausible, Elixir.Plausible.Cache, enabled: bool()` application env key.\nThis can be overridden on case by case basis, using the child specs options.\n\nThe `base_db_query/0` callback is used to generate the base query that is \nexecuted on every cache refresh. \n\nThere are two modes of refresh operation: `:all` and `:updated_recently`;\nthe former will invoke the query as is and clear all the existing entries,\nwhile the latter will attempt to limit the query to only the records that \nhave been updated in the last 15 minutes and try to merge the new results with\nexisting cache entries.\n\nBoth refresh modes are normally executed periodically from within a warmer process;\nsee: `Plausible.Cache.Warmer`. The reason for two modes is that the latter is lighter \non the database and can be executed more frequently.\n\nWhen Cache is disabled via application env, the `get/1` function\nfalls back to pure database lookups (implemented via `get_from_source/1` callback. \nThis should help with introducing cached lookups in existing code, \nso that no existing tests should break.\n\nRefreshing the cache emits telemetry event defined as per `telemetry_event_refresh/2`.","ref":"Plausible.Cache.html"},{"type":"callback","title":"Plausible.Cache.base_db_query/0","doc":"Returns the base Ecto query used to refresh the cache","ref":"Plausible.Cache.html#c:base_db_query/0"},{"type":"callback","title":"Plausible.Cache.child_id/0","doc":"Supervisor child id, must be unique within the supervision tree","ref":"Plausible.Cache.html#c:child_id/0"},{"type":"callback","title":"Plausible.Cache.count_all/0","doc":"Counts all items at the source, an aggregate query most likely","ref":"Plausible.Cache.html#c:count_all/0"},{"type":"function","title":"Plausible.Cache.enabled?/0","doc":"Looks for application env value at `:plausible, Elixir.Plausible.Cache, enabled: bool()`","ref":"Plausible.Cache.html#enabled?/0"},{"type":"callback","title":"Plausible.Cache.get_from_source/1","doc":"Retrieves the item from the source, in case the cache is disabled","ref":"Plausible.Cache.html#c:get_from_source/1"},{"type":"callback","title":"Plausible.Cache.name/0","doc":"Unique cache name, used by underlying implementation","ref":"Plausible.Cache.html#c:name/0"},{"type":"callback","title":"Plausible.Cache.unwrap_cache_keys/1","doc":"Optionally unwraps the keys of the cache items, in case one item is stored under multiple keys","ref":"Plausible.Cache.html#c:unwrap_cache_keys/1"},{"type":"module","title":"Plausible.Cache.Adapter","doc":"Interface for the underlying cache implementation.\nCurrently: ConCache\n\nUsing the Adapter module directly, the user must ensure that the relevant\nprocesses are available to use, which is normally done via the child specification.","ref":"Plausible.Cache.Adapter.html"},{"type":"function","title":"Plausible.Cache.Adapter.child_spec/3","doc":"","ref":"Plausible.Cache.Adapter.html#child_spec/3"},{"type":"function","title":"Plausible.Cache.Adapter.delete/2","doc":"","ref":"Plausible.Cache.Adapter.html#delete/2"},{"type":"function","title":"Plausible.Cache.Adapter.fetch/3","doc":"","ref":"Plausible.Cache.Adapter.html#fetch/3"},{"type":"function","title":"Plausible.Cache.Adapter.get/2","doc":"","ref":"Plausible.Cache.Adapter.html#get/2"},{"type":"function","title":"Plausible.Cache.Adapter.get/3","doc":"","ref":"Plausible.Cache.Adapter.html#get/3"},{"type":"function","title":"Plausible.Cache.Adapter.keys/1","doc":"","ref":"Plausible.Cache.Adapter.html#keys/1"},{"type":"function","title":"Plausible.Cache.Adapter.put/4","doc":"","ref":"Plausible.Cache.Adapter.html#put/4"},{"type":"function","title":"Plausible.Cache.Adapter.put_many/2","doc":"","ref":"Plausible.Cache.Adapter.html#put_many/2"},{"type":"function","title":"Plausible.Cache.Adapter.size/1","doc":"","ref":"Plausible.Cache.Adapter.html#size/1"},{"type":"function","title":"Plausible.Cache.Adapter.with_lock!/4","doc":"","ref":"Plausible.Cache.Adapter.html#with_lock!/4"},{"type":"module","title":"Plausible.Cache.Stats","doc":"Keeps track of hit/miss ratio for various caches.","ref":"Plausible.Cache.Stats.html"},{"type":"function","title":"Plausible.Cache.Stats.bump/2","doc":"","ref":"Plausible.Cache.Stats.html#bump/2"},{"type":"function","title":"Plausible.Cache.Stats.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.Cache.Stats.html#child_spec/1"},{"type":"function","title":"Plausible.Cache.Stats.gather/1","doc":"","ref":"Plausible.Cache.Stats.html#gather/1"},{"type":"function","title":"Plausible.Cache.Stats.handle_telemetry_event/4","doc":"","ref":"Plausible.Cache.Stats.html#handle_telemetry_event/4"},{"type":"function","title":"Plausible.Cache.Stats.hit_rate/1","doc":"","ref":"Plausible.Cache.Stats.html#hit_rate/1"},{"type":"function","title":"Plausible.Cache.Stats.init/1","doc":"","ref":"Plausible.Cache.Stats.html#init/1"},{"type":"function","title":"Plausible.Cache.Stats.size/1","doc":"","ref":"Plausible.Cache.Stats.html#size/1"},{"type":"function","title":"Plausible.Cache.Stats.start_link/1","doc":"","ref":"Plausible.Cache.Stats.html#start_link/1"},{"type":"module","title":"Plausible.Cache.Warmer","doc":"A periodic cache warmer.\n\nChild specification options available:\n\n * `cache_impl` - module expected to implement `Plausible.Cache` behaviour\n * `interval` - the number of milliseconds for each warm-up cycle\n * `cache_name` - defaults to cache_impl.name() but can be overridden for testing\n * `force_start?` - enforcess process startup for testing, even if it's barred\n by `Plausible.Cache.enabled?`. This is useful for avoiding issues with DB ownership\n and async tests.\n * `warmer_fn` - by convention, either `:refresh_all` or `:refresh_updated_recently`,\n both are automatically provided by `cache_impl` module. Technically any exported\n or captured function will work, if need be.\n\nSee tests for more comprehensive examples.","ref":"Plausible.Cache.Warmer.html"},{"type":"function","title":"Plausible.Cache.Warmer.child_spec/1","doc":"","ref":"Plausible.Cache.Warmer.html#child_spec/1"},{"type":"module","title":"Plausible.ChangesetHelpers","doc":"Helper function for working with Ecto changesets","ref":"Plausible.ChangesetHelpers.html"},{"type":"function","title":"Plausible.ChangesetHelpers.traverse_errors/1","doc":"","ref":"Plausible.ChangesetHelpers.html#traverse_errors/1"},{"type":"module","title":"Plausible.Cldr","doc":"Provides the core functions to retrieve and manage\nthe CLDR data that supports formatting and localisation.\n\nIt provides the core functions to access formatted\nCLDR data, set and retrieve a current locale and validate\ncertain core data types such as locales, currencies and\nterritories.","ref":"Plausible.Cldr.html"},{"type":"function","title":"Plausible.Cldr.available_locale_name?/1","doc":"","ref":"Plausible.Cldr.html#available_locale_name?/1"},{"type":"function","title":"Plausible.Cldr.default_locale/0","doc":"Returns the default `locale`.","ref":"Plausible.Cldr.html#default_locale/0"},{"type":"function","title":"Example - Plausible.Cldr.default_locale/0","doc":"iex> Plausible.Cldr.default_locale()\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en-001\",\n cldr_locale_name: :\"en-001\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en-001\",\n script: :Latn,\n territory: :\"001\",\n transform: %{},\n language_variants: []\n }","ref":"Plausible.Cldr.html#default_locale/0-example"},{"type":"function","title":"Plausible.Cldr.default_territory/0","doc":"Returns the default territory when a locale\ndoes not specify one and none can be inferred.","ref":"Plausible.Cldr.html#default_territory/0"},{"type":"function","title":"Example - Plausible.Cldr.default_territory/0","doc":"iex> Plausible.Cldr.default_territory()\n :\"001\"","ref":"Plausible.Cldr.html#default_territory/0-example"},{"type":"function","title":"Plausible.Cldr.ellipsis/2","doc":"Add locale-specific ellipsis to a string.","ref":"Plausible.Cldr.html#ellipsis/2"},{"type":"function","title":"Arguments - Plausible.Cldr.ellipsis/2","doc":"* `string` is any `String.t` or a 2-element list\n of `String.t` between which the ellipsis is inserted.\n\n* `backend` is any module that includes `use Cldr` and therefore\n is a `Cldr` backend module. The default is `Cldr.default_backend!/0`.\n Note that `Cldr.default_backend!/0` will raise an exception if\n no `:default_backend` is configured under the `:ex_cldr` key in\n `config.exs`.\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.html#ellipsis/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.ellipsis/2","doc":"* `:locale` is any valid locale name returned by `Cldr.known_locale_names/1`.\n The default is `Cldr.get_locale/0`.\n\n* `:location` determines where to place the ellipsis. The options are\n `:after` (the default for a single string argument), `:between`\n (the default and only valid location for an argument that is a list\n of two strings) and `:before`.\n\n* `:format` formats based upon whether the ellipsis\n is inserted between words or sentences. The valid options are\n `:word` or `:sentence`. The default is `:sentence`.","ref":"Plausible.Cldr.html#ellipsis/2-options"},{"type":"function","title":"Examples - Plausible.Cldr.ellipsis/2","doc":"iex> Plausible.Cldr.ellipsis(\"And furthermore\")\n \"And furthermore…\"\n\n iex> Plausible.Cldr.ellipsis([\"And furthermore\", \"there is much to be done\"], locale: :ja)\n \"And furthermore…there is much to be done\"\n\n iex> Plausible.Cldr.ellipsis(\"And furthermore\", format: :word)\n \"And furthermore …\"\n\n iex> Plausible.Cldr.ellipsis([\"And furthermore\", \"there is much to be done\"], locale: :ja, format: :word)\n \"And furthermore … there is much to be done\"","ref":"Plausible.Cldr.html#ellipsis/2-examples"},{"type":"function","title":"Plausible.Cldr.get_locale/0","doc":"Return the current locale to be used for `Cldr` functions that\ntake an optional locale parameter for which a locale is not supplied.","ref":"Plausible.Cldr.html#get_locale/0"},{"type":"function","title":"Example - Plausible.Cldr.get_locale/0","doc":"iex> Plausible.Cldr.put_locale(\"pl\")\n iex> Plausible.Cldr.get_locale()\n %Cldr.LanguageTag{\n backend: Elixir.Plausible.Cldr,\n canonical_locale_name: \"pl\",\n cldr_locale_name: :pl,\n extensions: %{},\n language: \"pl\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :pl,\n territory: :PL,\n requested_locale_name: \"pl\",\n script: :Latn,\n transform: %{},\n language_variants: []\n }","ref":"Plausible.Cldr.html#get_locale/0-example"},{"type":"function","title":"Plausible.Cldr.known_calendars/0","doc":"","ref":"Plausible.Cldr.html#known_calendars/0"},{"type":"function","title":"Plausible.Cldr.known_currencies/0","doc":"","ref":"Plausible.Cldr.html#known_currencies/0"},{"type":"function","title":"Plausible.Cldr.known_gettext_locale_name/1","doc":"Returns either the Gettext `locale_name` in Cldr format or\n`false` based upon whether the locale name is configured in\n`Gettext`.","ref":"Plausible.Cldr.html#known_gettext_locale_name/1"},{"type":"function","title":"Arguments - Plausible.Cldr.known_gettext_locale_name/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_gettext_locale_names/0`","ref":"Plausible.Cldr.html#known_gettext_locale_name/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.known_gettext_locale_name/1","doc":"iex> Plausible.Cldr.known_gettext_locale_name(\"en\")\n \"en\"\n\n iex> Plausible.Cldr.known_gettext_locale_name(\"en-SA\")\n false","ref":"Plausible.Cldr.html#known_gettext_locale_name/1-examples"},{"type":"function","title":"Plausible.Cldr.known_gettext_locale_name?/1","doc":"Returns a boolean indicating if the specified locale\nname is configured and available in Gettext.","ref":"Plausible.Cldr.html#known_gettext_locale_name?/1"},{"type":"function","title":"Arguments - Plausible.Cldr.known_gettext_locale_name?/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.html#known_gettext_locale_name?/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.known_gettext_locale_name?/1","doc":"iex> Plausible.Cldr.known_gettext_locale_name?(\"en\")\n true\n\n iex> Plausible.Cldr.known_gettext_locale_name?(\"!!\")\n false","ref":"Plausible.Cldr.html#known_gettext_locale_name?/1-examples"},{"type":"function","title":"Plausible.Cldr.known_gettext_locale_names/0","doc":"Returns a list of Gettext locale names but in CLDR format with\nunderscore replaced by hyphen in order to facilitate comparisons\nwith `Cldr` locale names.","ref":"Plausible.Cldr.html#known_gettext_locale_names/0"},{"type":"function","title":"Plausible.Cldr.known_locale_name/1","doc":"Returns either the `locale_name` or `false` based upon\nwhether the locale name is configured in `Cldr`.\n\nThis is helpful when building a list of `or` expressions\nto return the first known locale name from a list.","ref":"Plausible.Cldr.html#known_locale_name/1"},{"type":"function","title":"Arguments - Plausible.Cldr.known_locale_name/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.html#known_locale_name/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.known_locale_name/1","doc":"iex> Plausible.Cldr.known_locale_name(:\"en-AU\")\n :\"en-AU\"\n\n iex> Plausible.Cldr.known_locale_name(:\"en-SA\")\n false","ref":"Plausible.Cldr.html#known_locale_name/1-examples"},{"type":"function","title":"Plausible.Cldr.known_locale_name?/1","doc":"Returns a boolean indicating if the specified locale\nname is configured and available in Cldr.","ref":"Plausible.Cldr.html#known_locale_name?/1"},{"type":"function","title":"Arguments - Plausible.Cldr.known_locale_name?/1","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.html#known_locale_name?/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.known_locale_name?/1","doc":"iex> Plausible.Cldr.known_locale_name?(:en)\n true\n\n iex> Plausible.Cldr.known_locale_name?(:\"!!\")\n false","ref":"Plausible.Cldr.html#known_locale_name?/1-examples"},{"type":"function","title":"Plausible.Cldr.known_locale_names/0","doc":"Returns a list of the known locale names.\n\nKnown locales are those locales which\nare the subset of all CLDR locales that\nhave been configured for use either\nin this module or in `Gettext`.","ref":"Plausible.Cldr.html#known_locale_names/0"},{"type":"function","title":"Plausible.Cldr.known_number_system_types/0","doc":"Returns a list of atoms representing the number systems types known to `Cldr`.","ref":"Plausible.Cldr.html#known_number_system_types/0"},{"type":"function","title":"Example - Plausible.Cldr.known_number_system_types/0","doc":"iex> Plausible.Cldr.known_number_system_types()\n [:default, :finance, :native, :traditional]","ref":"Plausible.Cldr.html#known_number_system_types/0-example"},{"type":"function","title":"Plausible.Cldr.known_number_systems/0","doc":"","ref":"Plausible.Cldr.html#known_number_systems/0"},{"type":"function","title":"Plausible.Cldr.known_rbnf_locale_name/1","doc":"Returns either the RBNF `locale_name` or `false` based upon\nwhether the locale name is configured in `Cldr`\nand has RBNF rules defined.","ref":"Plausible.Cldr.html#known_rbnf_locale_name/1"},{"type":"function","title":"Arguments - Plausible.Cldr.known_rbnf_locale_name/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.html#known_rbnf_locale_name/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.known_rbnf_locale_name/1","doc":"iex> Plausible.Cldr.known_rbnf_locale_name(:en)\n :en\n\n iex> Plausible.Cldr.known_rbnf_locale_name(:\"en-SA\")\n false","ref":"Plausible.Cldr.html#known_rbnf_locale_name/1-examples"},{"type":"function","title":"Plausible.Cldr.known_rbnf_locale_name?/1","doc":"Returns a boolean indicating if the specified locale\nname is configured and available in Cldr and supports\nrules based number formats (RBNF).","ref":"Plausible.Cldr.html#known_rbnf_locale_name?/1"},{"type":"function","title":"Arguments - Plausible.Cldr.known_rbnf_locale_name?/1","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.html#known_rbnf_locale_name?/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.known_rbnf_locale_name?/1","doc":"iex> Plausible.Cldr.known_rbnf_locale_name?(:en)\n true\n\n iex> Plausible.Cldr.known_rbnf_locale_name?(:\"!!\")\n false","ref":"Plausible.Cldr.html#known_rbnf_locale_name?/1-examples"},{"type":"function","title":"Plausible.Cldr.known_rbnf_locale_names/0","doc":"Returns a list of locale names which have rules-based number\nformats (RBNF).","ref":"Plausible.Cldr.html#known_rbnf_locale_names/0"},{"type":"function","title":"Plausible.Cldr.known_territories/0","doc":"","ref":"Plausible.Cldr.html#known_territories/0"},{"type":"function","title":"Plausible.Cldr.normalize_lenient_parse/3","doc":"Normalizes a string by applying transliteration\nof common symbols in numbers, currencies and dates","ref":"Plausible.Cldr.html#normalize_lenient_parse/3"},{"type":"function","title":"Plausible.Cldr.put_gettext_locale/1","doc":"Set the current process's Gettext locale from a\n`t:Cldr.LanguageTag`.","ref":"Plausible.Cldr.html#put_gettext_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.put_gettext_locale/1","doc":"* `locale` is a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`.","ref":"Plausible.Cldr.html#put_gettext_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.put_gettext_locale/1","doc":"* `{:ok, gettext_locale_name}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.html#put_gettext_locale/1-returns"},{"type":"function","title":"Behaviour - Plausible.Cldr.put_gettext_locale/1","doc":"1. If the `locale.gettext_locale_name` is `nil` then an error\n is returned.\n\n2. The `gettext` locale for the `gettext_backend` configured for the\n CLDR backend defined by the `t:Cldr.LanguageTag` is set.","ref":"Plausible.Cldr.html#put_gettext_locale/1-behaviour"},{"type":"function","title":"Examples - Plausible.Cldr.put_gettext_locale/1","doc":"iex> import Cldr.LanguageTag.Sigil\n iex> Plausible.Cldr.put_gettext_locale(~l\"en\")\n {:ok, \"en\"}\n\n iex> import Cldr.LanguageTag.Sigil\n iex> Plausible.Cldr.put_gettext_locale(~l\"de\")\n {\n :error,\n {\n Cldr.UnknownLocaleError,\n \"Locale TestBackend.Cldr.Locale.new!(\\\"de-DE\\\") does not map to a known gettext locale name\"\n }\n }","ref":"Plausible.Cldr.html#put_gettext_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.put_locale/1","doc":"Set the current locale to be used for `Cldr` functions that\ntake an optional locale parameter for which a locale is not supplied.","ref":"Plausible.Cldr.html#put_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.put_locale/1","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `t:Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`\n\nSee [rfc5646](https://tools.ietf.org/html/rfc5646) for the specification\nof a language tag.","ref":"Plausible.Cldr.html#put_locale/1-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.put_locale/1","doc":"iex> Plausible.Cldr.put_locale(\"en\")\n {:ok,\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en\",\n cldr_locale_name: :en,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en\",\n script: :Latn,\n territory: :US,\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.put_locale(\"invalid-locale!\")\n {:error, {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"!\\\" starting at position 15\"}}","ref":"Plausible.Cldr.html#put_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.quote/2","doc":"Add locale-specific quotation marks around a string.","ref":"Plausible.Cldr.html#quote/2"},{"type":"function","title":"Arguments - Plausible.Cldr.quote/2","doc":"* `string` is any valid Elixir string\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.html#quote/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.quote/2","doc":"* `locale` is any valid locale name returned by `Cldr.known_locale_names/1`.\n The default is `Cldr.get_locale/0`","ref":"Plausible.Cldr.html#quote/2-options"},{"type":"function","title":"Examples - Plausible.Cldr.quote/2","doc":"iex> Plausible.Cldr.quote(\"Quoted String\")\n \"“Quoted String”\"\n\n iex> Plausible.Cldr.quote(\"Quoted String\", locale: :ja)\n \"「Quoted String」\"","ref":"Plausible.Cldr.html#quote/2-examples"},{"type":"function","title":"Plausible.Cldr.unknown_locale_names/0","doc":"Returns a list of the locales names that are configured,\nbut not known in CLDR.\n\nSince there is a compile-time exception raised if there are\nany unknown locales this function should always\nreturn an empty list.","ref":"Plausible.Cldr.html#unknown_locale_names/0"},{"type":"function","title":"Plausible.Cldr.validate_calendar/1","doc":"","ref":"Plausible.Cldr.html#validate_calendar/1"},{"type":"function","title":"Plausible.Cldr.validate_currency/1","doc":"","ref":"Plausible.Cldr.html#validate_currency/1"},{"type":"function","title":"Plausible.Cldr.validate_locale/1","doc":"Normalise and validate a locale name.","ref":"Plausible.Cldr.html#validate_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.validate_locale/1","doc":"* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`","ref":"Plausible.Cldr.html#validate_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.validate_locale/1","doc":"* `{:ok, language_tag}`\n\n* `{:error, reason}`","ref":"Plausible.Cldr.html#validate_locale/1-returns"},{"type":"function","title":"Notes - Plausible.Cldr.validate_locale/1","doc":"See [rfc5646](https://tools.ietf.org/html/rfc5646) for the specification\nof a language tag.","ref":"Plausible.Cldr.html#validate_locale/1-notes"},{"type":"function","title":"Examples - Plausible.Cldr.validate_locale/1","doc":"iex> Plausible.Cldr.validate_locale(:en)\n {:ok,\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en\",\n cldr_locale_name: :en,\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en\",\n script: :Latn,\n territory: :US,\n transform: %{},\n language_variants: []\n }}\n\n\n iex> Plausible.Cldr.validate_locale Plausible.Cldr.default_locale()\n {:ok,\n %Cldr.LanguageTag{\n backend: Plausible.Cldr,\n canonical_locale_name: \"en-001\",\n cldr_locale_name: :\"en-001\",\n extensions: %{},\n gettext_locale_name: \"en\",\n language: \"en\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :en,\n requested_locale_name: \"en-001\",\n script: :Latn,\n territory: :\"001\",\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.validate_locale(\"zzz\")\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zzz\\\" is invalid\"}}","ref":"Plausible.Cldr.html#validate_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.validate_number_system/1","doc":"","ref":"Plausible.Cldr.html#validate_number_system/1"},{"type":"function","title":"Plausible.Cldr.validate_number_system_type/1","doc":"Normalise and validate a number system type.","ref":"Plausible.Cldr.html#validate_number_system_type/1"},{"type":"function","title":"Arguments - Plausible.Cldr.validate_number_system_type/1","doc":"* `number_system_type` is any number system type returned by\n `Cldr.known_number_system_types/1`","ref":"Plausible.Cldr.html#validate_number_system_type/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.validate_number_system_type/1","doc":"* `{:ok, normalized_number_system_type}` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.html#validate_number_system_type/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.validate_number_system_type/1","doc":"iex> Plausible.Cldr.validate_number_system_type(:default)\n {:ok, :default}\n\n iex> Plausible.Cldr.validate_number_system_type(:traditional)\n {:ok, :traditional}\n\n iex> Plausible.Cldr.validate_number_system_type(:latn)\n {\n :error,\n {Cldr.UnknownNumberSystemTypeError, \"The number system type :latn is unknown\"}\n }\n\n iex> Plausible.Cldr.validate_number_system_type(\"bork\")\n {\n :error,\n {Cldr.UnknownNumberSystemTypeError, \"The number system type \\\"bork\\\" is invalid\"}\n }","ref":"Plausible.Cldr.html#validate_number_system_type/1-examples"},{"type":"function","title":"Plausible.Cldr.validate_territory/1","doc":"","ref":"Plausible.Cldr.html#validate_territory/1"},{"type":"function","title":"Plausible.Cldr.with_locale/2","doc":"Execute a function with a locale ensuring that the\ncurrent locale is restored after the function.","ref":"Plausible.Cldr.html#with_locale/2"},{"type":"function","title":"Arguments - Plausible.Cldr.with_locale/2","doc":"* `locale` is any valid locale name returned by `Cldr.known_locale_names/1`.\n\n* `fun` is any 0-arity function or function capture.","ref":"Plausible.Cldr.html#with_locale/2-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.with_locale/2","doc":"* The value returned by the function `fun/0` or\n\n* `{:error, {exception, reason}}` if the locale is invalid or\n\n* raises an exception if the current locale cannot be\n identified.","ref":"Plausible.Cldr.html#with_locale/2-returns"},{"type":"module","title":"Plausible.Cldr.AcceptLanguage","doc":"Parses HTTP `Accept-Language` header values as defined in\n[rfc2616](https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4).\n\nThe Accept-Language request-header field is similar to Accept, but restricts\nthe set of natural languages that are preferred as a response to the request.\nLanguage tags function are provided in `Cldr.LanguageTag`.\n\nThe format of an `Accept-Language` header is as follows in `ABNF` format:\n\n Accept-Language = \"Accept-Language\" \":\"\n 1#( language-range [ \";\" \"q\" \"=\" qvalue ] )\n language-range = ( ( 1*8ALPHA *( \"-\" 1*8ALPHA ) ) | \"*\" )\n\nEach language-range MAY be given an associated quality value which represents an\nestimate of the user's preference for the languages specified by that range. The\nquality value defaults to \"q=1\". For example,\n\n Accept-Language: da, en-gb;q=0.8, en;q=0.7\n\nwould mean: \"I prefer Danish, but will accept British English and other types of English.\"","ref":"Plausible.Cldr.AcceptLanguage.html"},{"type":"function","title":"Plausible.Cldr.AcceptLanguage.best_match/1","doc":"Parse an `Accept-Language` string and return the best match for\na configured `Cldr` locale.","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1"},{"type":"function","title":"Arguments - Plausible.Cldr.AcceptLanguage.best_match/1","doc":"* `accept_language` is a string representing an accept language header","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.AcceptLanguage.best_match/1","doc":"* `{:ok, language_tag}` or\n\n* `{:error, reason}`","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.AcceptLanguage.best_match/1","doc":"iex> Plausible.Cldr.AcceptLanguage.best_match(\"da;q=0.1,zh-TW;q=0.3\", TestBackend.Cldr)\n {:ok,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.AcceptLanguage.best_match(\"da;q=0.1,zh-TW;q=0.3\", TestBackend.Cldr)\n {:ok,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n\n iex> Plausible.Cldr.AcceptLanguage.best_match(\"xx,yy;q=0.3\")\n {:error,\n {Cldr.NoMatchingLocale,\n \"No configured locale could be matched to \\\"xx,yy;q=0.3\\\"\"}}\n\n iex> Plausible.Cldr.AcceptLanguage.best_match(\"invalid_tag\")\n {:error, {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}","ref":"Plausible.Cldr.AcceptLanguage.html#best_match/1-examples"},{"type":"function","title":"Plausible.Cldr.AcceptLanguage.parse/1","doc":"Parses an `Accept-Language` header value in its string\nor tokenized form to return a tuple of the form\n`{:ok, [{quality, %Cldr.LanguageTag{}}, ...]}` sorted by quality.","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1"},{"type":"function","title":"Arguments - Plausible.Cldr.AcceptLanguage.parse/1","doc":"* `accept-language` is any string in the format defined by\n [rfc2616](https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4)\n\n* `backend` is any module that includes `use Cldr` and therefore\n is a `Cldr` backend module","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.AcceptLanguage.parse/1","doc":"* `{:ok, [{quality, language_tag}, ...]}` or\n\n* `{:error, {Cldr.AcceptLanguageError, String.t}}`\n\nIf at least one valid language tag is found but errors are also\ndetected on one more more tags, an `{ok, list}` tuple is returned\nwith an error tuple for each invalid tag added at the end of the list.","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1-returns"},{"type":"function","title":"Example - Plausible.Cldr.AcceptLanguage.parse/1","doc":"iex> Cldr.AcceptLanguage.parse(\"da,zh-TW;q=0.3\", TestBackend.Cldr)\n {:ok,\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n ]}\n\n iex> Plausible.Cldr.AcceptLanguage.parse(\"invalid_tag\")\n {:error,\n {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}\n\n iex> Plausible.Cldr.AcceptLanguage.parse(\"da,zh-TW;q=0.3,invalid_tag\")\n {:ok,\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }},\n {:error,\n {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}\n ]}","ref":"Plausible.Cldr.AcceptLanguage.html#parse/1-example"},{"type":"function","title":"Plausible.Cldr.AcceptLanguage.parse!/1","doc":"Parses an `Accept-Language` header value in its string\nor tokenized form to produce a list of tuples of the form\n`[{quality, %Cldr.LanguageTag{}}, ...]` sorted by quality\nin descending order.","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1"},{"type":"function","title":"Arguments - Plausible.Cldr.AcceptLanguage.parse!/1","doc":"* `accept-language` is any string in the format defined by [rfc2616](https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.4)","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.AcceptLanguage.parse!/1","doc":"* `{:ok, [{quality, language_tag}, ...]}` or\n\n* raises a `Cldr.AcceptLanguageError` exception\n\nIf at least one valid language tag is found but errors are also\ndetected on one more more tags, an `{ok, list}` tuple is returned\nwith an error tuple for each invalid tag added at the end of the list.","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1-returns"},{"type":"function","title":"Example - Plausible.Cldr.AcceptLanguage.parse!/1","doc":"iex> Plausible.Cldr.AcceptLanguage.parse!(\"da,zh-TW;q=0.3\")\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }}\n ]\n\n Plausible.Cldr.AcceptLanguage.parse! \"invalid_tag\"\n ** (Cldr.AcceptLanguageError) \"Expected a BCP47 language tag. Could not parse the remaining \"g\" starting at position 11\n (ex_cldr) lib/cldr/accept_language.ex:304: Cldr.AcceptLanguage.parse!/1\n\n iex> Plausible.Cldr.AcceptLanguage.parse!(\"da,zh-TW;q=0.3,invalid_tag\")\n [\n {1.0,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"da\",\n cldr_locale_name: :da,\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"da\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :da,\n requested_locale_name: \"da\",\n script: :Latn,\n territory: :DK,\n transform: %{},\n language_variants: []\n }},\n {0.3,\n %Cldr.LanguageTag{\n backend: TestBackend.Cldr,\n canonical_locale_name: \"zh-TW\",\n cldr_locale_name: :\"zh-Hant\",\n language_subtags: [],\n extensions: %{},\n gettext_locale_name: nil,\n language: \"zh\",\n locale: %{},\n private_use: [],\n rbnf_locale_name: :\"zh-Hant\",\n requested_locale_name: \"zh-TW\",\n script: :Hant,\n territory: :TW,\n transform: %{},\n language_variants: []\n }},\n {:error,\n {Cldr.LanguageTag.ParseError,\n \"Expected a BCP47 language tag. Could not parse the remaining \\\"g\\\" starting at position 11\"}}\n ]","ref":"Plausible.Cldr.AcceptLanguage.html#parse!/1-example"},{"type":"module","title":"Plausible.Cldr.Currency","doc":"","ref":"Plausible.Cldr.Currency.html"},{"type":"function","title":"Plausible.Cldr.Currency.currencies_for_locale/3","doc":"Returns a map of the metadata for all currencies for\na given locale.","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currencies_for_locale/3","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currencies_for_locale/3","doc":"* `{:ok, currency_map}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.currencies_for_locale/3","doc":"MyApp.Cldr.Currency.currencies_for_locale(\"en\")\n => {:ok,\n %{\n FJD: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"FJD\",\n count: %{one: \"Fijian dollar\", other: \"Fijian dollars\"},\n digits: 2,\n from: nil,\n iso_digits: 2,\n name: \"Fijian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"FJD\",\n tender: true,\n to: nil\n },\n SUR: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"SUR\",\n count: %{one: \"Soviet rouble\", other: \"Soviet roubles\"},\n digits: 2,\n from: nil,\n iso_digits: nil,\n name: \"Soviet Rouble\",\n narrow_symbol: nil,\n rounding: 0,\n symbol: \"SUR\",\n tender: true,\n to: nil\n },\n ...\n }}","ref":"Plausible.Cldr.Currency.html#currencies_for_locale/3-example"},{"type":"function","title":"Plausible.Cldr.Currency.currencies_for_locale!/3","doc":"Returns a map of the metadata for all currencies for\na given locale and raises on error.","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currencies_for_locale!/3","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currencies_for_locale!/3","doc":"* `{:ok, currency_map}` or\n\n* raises an exception","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.currencies_for_locale!/3","doc":"MyApp.Cldr.Currency.currencies_for_locale!(\"en\")\n => %{\n FJD: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"FJD\",\n count: %{one: \"Fijian dollar\", other: \"Fijian dollars\"},\n digits: 2,\n from: nil,\n iso_digits: 2,\n name: \"Fijian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"FJD\",\n tender: true,\n to: nil\n },\n SUR: %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"SUR\",\n count: %{one: \"Soviet rouble\", other: \"Soviet roubles\"},\n digits: 2,\n from: nil,\n iso_digits: nil,\n name: \"Soviet Rouble\",\n narrow_symbol: nil,\n rounding: 0,\n symbol: \"SUR\",\n tender: true,\n to: nil\n },\n ...\n }","ref":"Plausible.Cldr.Currency.html#currencies_for_locale!/3-example"},{"type":"function","title":"Plausible.Cldr.Currency.currency_for_code/2","doc":"Returns the currency metadata for the requested currency code.","ref":"Plausible.Cldr.Currency.html#currency_for_code/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currency_for_code/2","doc":"* `currency_or_currency_code` is a `binary` or `atom` representation\n of an ISO 4217 currency code, or a `%Cldr.Currency{}` struct.","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Currency.currency_for_code/2","doc":"* `:locale` is any valid locale name returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currency_for_code/2","doc":"* A `{:ok, currency}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Currency.currency_for_code/2","doc":"iex> Plausible.Cldr.Currency.currency_for_code(\"AUD\")\n {:ok,\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"AUD\",\n count: %{one: \"Australian dollar\", other: \"Australian dollars\"},\n digits: 2,\n iso_digits: 2,\n name: \"Australian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"A$\",\n tender: true\n }}\n\n iex> Plausible.Cldr.Currency.currency_for_code(\"THB\")\n {:ok,\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"THB\",\n count: %{one: \"Thai baht\", other: \"Thai baht\"},\n digits: 2,\n iso_digits: 2,\n name: \"Thai Baht\",\n narrow_symbol: \"฿\",\n rounding: 0,\n symbol: \"THB\",\n tender: true\n }}","ref":"Plausible.Cldr.Currency.html#currency_for_code/2-examples"},{"type":"function","title":"Plausible.Cldr.Currency.currency_for_code!/2","doc":"Returns the currency metadata for the requested currency code.","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currency_for_code!/2","doc":"* `currency_or_currency_code` is a `binary` or `atom` representation\n of an ISO 4217 currency code, or a `%Cldr.Currency{}` struct.","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Currency.currency_for_code!/2","doc":"* `:locale` is any valid locale name returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currency_for_code!/2","doc":"* A `t:Cldr.Current.t/0` or\n\n* raises an exception","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Currency.currency_for_code!/2","doc":"iex> Plausible.Cldr.Currency.currency_for_code!(\"AUD\")\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"AUD\",\n count: %{one: \"Australian dollar\", other: \"Australian dollars\"},\n digits: 2,\n iso_digits: 2,\n name: \"Australian Dollar\",\n narrow_symbol: \"$\",\n rounding: 0,\n symbol: \"A$\",\n tender: true\n }\n\n iex> Plausible.Cldr.Currency.currency_for_code!(\"THB\")\n %Cldr.Currency{\n cash_digits: 2,\n cash_rounding: 0,\n code: \"THB\",\n count: %{one: \"Thai baht\", other: \"Thai baht\"},\n digits: 2,\n iso_digits: 2,\n name: \"Thai Baht\",\n narrow_symbol: \"฿\",\n rounding: 0,\n symbol: \"THB\",\n tender: true\n }","ref":"Plausible.Cldr.Currency.html#currency_for_code!/2-examples"},{"type":"function","title":"Plausible.Cldr.Currency.currency_from_locale/1","doc":"Returns the effective currency for a given locale","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currency_from_locale/1","doc":"* `locale` is a `Cldr.LanguageTag` struct returned by\n `Cldr.Locale.new!/2`","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currency_from_locale/1","doc":"* A ISO 4217 currency code as an upcased atom","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Currency.currency_from_locale/1","doc":"iex> {:ok, locale} = Plausible.Cldr.validate_locale(\"en\")\n iex> Plausible.Cldr.Currency.currency_from_locale locale\n :USD\n\n iex> {:ok, locale} = Plausible.Cldr.validate_locale(\"en-AU\")\n iex> Plausible.Cldr.Currency.currency_from_locale locale\n :AUD\n\n iex> Plausible.Cldr.Currency.currency_from_locale(\"en-GB\")\n :GBP","ref":"Plausible.Cldr.Currency.html#currency_from_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.Currency.currency_history_for_locale/1","doc":"Returns a list of historic and the current\ncurrency for a given locale.","ref":"Plausible.Cldr.Currency.html#currency_history_for_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currency_history_for_locale/1","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`","ref":"Plausible.Cldr.Currency.html#currency_history_for_locale/1-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Currency.currency_history_for_locale/1","doc":"iex> MyApp.Cldr.Currency.currency_history_for_locale(\"en\")\n {:ok,\n %{\n USD: %{from: ~D[1792-01-01], to: nil},\n USN: %{tender: false},\n USS: %{from: nil, tender: false, to: ~D[2014-03-01]}\n }\n }","ref":"Plausible.Cldr.Currency.html#currency_history_for_locale/1-example"},{"type":"function","title":"Plausible.Cldr.Currency.currency_strings/3","doc":"Returns a map that matches a currency string to a\ncurrency code.\n\nA currency string is a localised name or symbol\nrepresenting a currency in a locale-specific manner.","ref":"Plausible.Cldr.Currency.html#currency_strings/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currency_strings/3","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","ref":"Plausible.Cldr.Currency.html#currency_strings/3-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currency_strings/3","doc":"* `{:ok, currency_string_map}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Currency.html#currency_strings/3-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.currency_strings/3","doc":"MyApp.Cldr.Currency.currency_strings(\"en\")\n => {:ok,\n %{\n \"mexican silver pesos\" => :MXP,\n \"sudanese dinar\" => :SDD,\n \"bad\" => :BAD,\n \"rsd\" => :RSD,\n \"swazi lilangeni\" => :SZL,\n \"zairean new zaire\" => :ZRN,\n \"guyanaese dollars\" => :GYD,\n \"equatorial guinean ekwele\" => :GQE,\n ...\n }}","ref":"Plausible.Cldr.Currency.html#currency_strings/3-example"},{"type":"function","title":"Plausible.Cldr.Currency.currency_strings!/3","doc":"Returns a map that matches a currency string to a\ncurrency code or raises an exception.\n\nA currency string is a localised name or symbol\nrepresenting a currency in a locale-specific manner.","ref":"Plausible.Cldr.Currency.html#currency_strings!/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.currency_strings!/3","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`\n\n* `currency_status` is `:all`, `:current`, `:historic`,\n `unannotated` or `:tender`; or a list of one or more status.\n The default is `:all`. See `Cldr.Currency.currency_filter/2`.","ref":"Plausible.Cldr.Currency.html#currency_strings!/3-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.currency_strings!/3","doc":"* `{:ok, currency_string_map}` or\n\n* raises an exception","ref":"Plausible.Cldr.Currency.html#currency_strings!/3-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.currency_strings!/3","doc":"MyApp.Cldr.Currency.currency_strings!(\"en\")\n => %{\n \"mexican silver pesos\" => :MXP,\n \"sudanese dinar\" => :SDD,\n \"bad\" => :BAD,\n \"rsd\" => :RSD,\n \"swazi lilangeni\" => :SZL,\n \"zairean new zaire\" => :ZRN,\n \"guyanaese dollars\" => :GYD,\n \"equatorial guinean ekwele\" => :GQE,\n ...\n }","ref":"Plausible.Cldr.Currency.html#currency_strings!/3-example"},{"type":"function","title":"Plausible.Cldr.Currency.current_currency_from_locale/1","doc":"Returns the current currency for a given locale.\n\nThis function does not consider the `U` extenion\nparameters `cu` or `rg`. It is recommended to us\n`Cldr.Currency.currency_from_locale/1` in most\ncircumstances.","ref":"Plausible.Cldr.Currency.html#current_currency_from_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.current_currency_from_locale/1","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`","ref":"Plausible.Cldr.Currency.html#current_currency_from_locale/1-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Currency.current_currency_from_locale/1","doc":"iex> MyApp.Cldr.Currency.current_currency_from_locale(\"en\")\n :USD\n\n iex> MyApp.Cldr.Currency.current_currency_from_locale(\"en-AU\")\n :AUD","ref":"Plausible.Cldr.Currency.html#current_currency_from_locale/1-example"},{"type":"function","title":"Plausible.Cldr.Currency.current_territory_currencies/0","doc":"Returns a mapping from a territory code to its\ncurrent currency code.\n\nIf a territory has no current currency (like\nAntartica, territory code `:AQ`) then no\nmapping is returned for that territory.","ref":"Plausible.Cldr.Currency.html#current_territory_currencies/0"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.current_territory_currencies/0","doc":"* A map of `{territory_code => Cldr.Currency.t}`","ref":"Plausible.Cldr.Currency.html#current_territory_currencies/0-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.current_territory_currencies/0","doc":"iex> Plausible.Cldr.Currency.current_territory_currencies()","ref":"Plausible.Cldr.Currency.html#current_territory_currencies/0-example"},{"type":"function","title":"Plausible.Cldr.Currency.known_currencies/0","doc":"","ref":"Plausible.Cldr.Currency.html#known_currencies/0"},{"type":"function","title":"Plausible.Cldr.Currency.known_currency?/1","doc":"","ref":"Plausible.Cldr.Currency.html#known_currency?/1"},{"type":"function","title":"Plausible.Cldr.Currency.known_currency_code/1","doc":"Returns a 2-tuple indicating if the supplied currency code is known.","ref":"Plausible.Cldr.Currency.html#known_currency_code/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.known_currency_code/1","doc":"* `currency_code` is a `binary` or `atom` representing an ISO4217\n currency code","ref":"Plausible.Cldr.Currency.html#known_currency_code/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.known_currency_code/1","doc":"* `{:ok, currency_code}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Currency.html#known_currency_code/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Currency.known_currency_code/1","doc":"iex> Plausible.Cldr.Currency.known_currency_code(\"AUD\")\n {:ok, :AUD}\n\n iex> Plausible.Cldr.Currency.known_currency_code(\"GGG\")\n {:error, {Cldr.UnknownCurrencyError, \"The currency \\\"GGG\\\" is invalid\"}}","ref":"Plausible.Cldr.Currency.html#known_currency_code/1-examples"},{"type":"function","title":"Plausible.Cldr.Currency.known_currency_code?/1","doc":"Returns a boolean indicating if the supplied currency code is known.","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.known_currency_code?/1","doc":"* `currency_code` is a `binary` or `atom` representing an ISO4217\n currency code","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.known_currency_code?/1","doc":"* `true` or `false`","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Currency.known_currency_code?/1","doc":"iex> Plausible.Cldr.Currency.known_currency_code?(\"AUD\")\n true\n\n iex> Plausible.Cldr.Currency.known_currency_code?(\"GGG\")\n false\n\n iex> Plausible.Cldr.Currency.known_currency_code?(:XCV)\n false","ref":"Plausible.Cldr.Currency.html#known_currency_code?/1-examples"},{"type":"function","title":"Plausible.Cldr.Currency.known_currency_codes/0","doc":"Returns a list of all known currency codes.","ref":"Plausible.Cldr.Currency.html#known_currency_codes/0"},{"type":"function","title":"Example - Plausible.Cldr.Currency.known_currency_codes/0","doc":"iex> Plausible.Cldr.Currency.known_currency_codes()","ref":"Plausible.Cldr.Currency.html#known_currency_codes/0-example"},{"type":"function","title":"Plausible.Cldr.Currency.new/2","doc":"Returns a `Currency` struct created from the arguments.","ref":"Plausible.Cldr.Currency.html#new/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.new/2","doc":"* `currency` is a private use currency code in a format defined by\n [ISO4217](https://en.wikipedia.org/wiki/ISO_4217)\n which is `X` followed by two alphanumeric characters.\n\n* `options` is a map of options representing the optional elements of\n the `Cldr.Currency.t` struct.","ref":"Plausible.Cldr.Currency.html#new/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Currency.new/2","doc":"* `:name` is the name of the currency. Required.\n* `:digits` is the precision of the currency. Required.\n* `:symbol` is the currency symbol. Optional.\n* `:narrow_symbol` is an alternative narrow symbol. Optional.\n* `:round_nearest` is the rounding precision such as `0.05`. Optional.\n* `:alt_code` is an alternative currency code for application use.\n* `:cash_digits` is the precision of the currency when used as cash. Optional.\n* `:cash_rounding_nearest` is the rounding precision when used as cash\n such as `0.05`. Optional.","ref":"Plausible.Cldr.Currency.html#new/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.new/2","doc":"* `{:ok, Cldr.Currency.t}` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Currency.html#new/2-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.new/2","doc":"iex> Plausible.Cldr.Currency.new(:XAE, name: \"Custom Name\", digits: 0)\n {:ok,\n %Cldr.Currency{\n alt_code: :XAE,\n cash_digits: 0,\n cash_rounding: nil,\n code: :XAE,\n count: %{other: \"Custom Name\"},\n digits: 0,\n from: nil,\n iso_digits: 0,\n name: \"Custom Name\",\n narrow_symbol: nil,\n rounding: 0,\n symbol: \"XAE\",\n tender: false,\n to: nil\n }}\n iex> MyApp.Cldr.Currency.new(:XAH, name: \"Custom Name\")\n {:error, \"Required options are missing. Required options are [:name, :digits]\"}\n iex> Plausible.Cldr.Currency.new(:XAE, name: \"XAE\", digits: 0)\n {:error, {Cldr.CurrencyAlreadyDefined, \"Currency :XAE is already defined.\"}}","ref":"Plausible.Cldr.Currency.html#new/2-example"},{"type":"function","title":"Plausible.Cldr.Currency.pluralize/3","doc":"Returns the appropriate currency display name for the `currency`, based\non the plural rules in effect for the `locale`.","ref":"Plausible.Cldr.Currency.html#pluralize/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.pluralize/3","doc":"* `number` is an integer, float or `Decimal`\n\n* `currency` is any currency returned by `Cldr.Currency.known_currencies/0`\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Currency.html#pluralize/3-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Currency.pluralize/3","doc":"* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`. The\n default is `Plausible.Cldr.get_locale/0`","ref":"Plausible.Cldr.Currency.html#pluralize/3-options"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.pluralize/3","doc":"* `{:ok, plural_string}` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Currency.html#pluralize/3-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Currency.pluralize/3","doc":"iex> Plausible.Cldr.Currency.pluralize(1, :USD)\n {:ok, \"US dollar\"}\n\n iex> Plausible.Cldr.Currency.pluralize(3, :USD)\n {:ok, \"US dollars\"}\n\n iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: \"zh\")\n {:ok, \"美元\"}\n\n iex> Plausible.Cldr.Currency.pluralize(12, :USD, locale: \"fr\")\n {:ok, \"dollars des États-Unis\"}\n\n iex> Plausible.Cldr.Currency.pluralize(1, :USD, locale: \"fr\")\n {:ok, \"dollar des États-Unis\"}","ref":"Plausible.Cldr.Currency.html#pluralize/3-examples"},{"type":"function","title":"Plausible.Cldr.Currency.strings_for_currency/2","doc":"Returns the strings associated with a currency\nin a given locale.","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Currency.strings_for_currency/2","doc":"* `currency` is an ISO4217 currency code\n\n* `locale` is any valid locale name returned by `MyApp.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `MyApp.Cldr.Locale.new!/1`","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Currency.strings_for_currency/2","doc":"* A list of strings or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2-returns"},{"type":"function","title":"Example - Plausible.Cldr.Currency.strings_for_currency/2","doc":"iex> MyApp.Cldr.Currency.strings_for_currency :AUD,(\"en\")\n [\"a$\", \"australian dollars\", \"aud\", \"australian dollar\"]","ref":"Plausible.Cldr.Currency.html#strings_for_currency/2-example"},{"type":"module","title":"Plausible.Cldr.Locale","doc":"Backend module that provides functions\nto define new locales and display human-readable\nlocale names for presentation purposes.","ref":"Plausible.Cldr.Locale.html"},{"type":"function","title":"Plausible.Cldr.Locale.fallback_locale_names/1","doc":"Returns the list of fallback locale names, starting\nwith the provided locale name.\n\nFallbacks are a list of locate names which can\nbe used to resolve translation or other localization\ndata if such localised data does not exist for\nthis specific locale..","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.fallback_locale_names/1","doc":"* `locale_name` is any locale name returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.fallback_locale_names/1","doc":"* `{:ok, list_of_locale_names}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.fallback_locale_names/1","doc":"iex> Plausible.Cldr.Locale.fallback_locale_names(:\"fr-CA\")\n {:ok, [:\"fr-CA\", :fr, :und]}\n\n # Fallbacks are typically formed by progressively\n # stripping variant, territory and script from the\n # given locale name. But not always - there are\n # certain fallbacks that take a different path.\n\n iex> Plausible.Cldr.Locale.fallback_locale_names(:nb)\n {:ok, [:nb, :no, :und]}","ref":"Plausible.Cldr.Locale.html#fallback_locale_names/1-examples"},{"type":"function","title":"Plausible.Cldr.Locale.fallback_locales/1","doc":"Returns the list of fallback locales, starting\nwith the provided locale name.\n\nFallbacks are a list of locate names which can\nbe used to resolve translation or other localization\ndata if such localised data does not exist for\nthis specific locale.","ref":"Plausible.Cldr.Locale.html#fallback_locales/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.fallback_locales/1","doc":"* `locale_name` is any locale name returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.Locale.html#fallback_locales/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.fallback_locales/1","doc":"* `{:ok, list_of_locales}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Locale.html#fallback_locales/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.fallback_locales/1","doc":"Plausible.Cldr.Locale.fallback_locales(:\"fr-CA\")\n => {:ok,\n [#Cldr.LanguageTag , #Cldr.LanguageTag ,\n #Cldr.LanguageTag ]}\n\n # Fallbacks are typically formed by progressively\n # stripping variant, territory and script from the\n # given locale name. But not always - there are\n # certain fallbacks that take a different path.\n\n Plausible.Cldr.Locale.fallback_locales(:nb))\n => {:ok,\n [#Cldr.LanguageTag , #Cldr.LanguageTag ,\n #Cldr.LanguageTag ]}","ref":"Plausible.Cldr.Locale.html#fallback_locales/1-examples"},{"type":"function","title":"Plausible.Cldr.Locale.locale_for_territory/1","doc":"Returns the \"best fit\" locale for a given territory.\n\nUsing the population percentage data from CLDR, the\nlanguage most commonly spoken in the given territory\nis used to form a locale name which is then validated\nagainst the given backend.\n\nFirst a territory-specific locale is validated and if\nthat fails, the base language only is validate.\n\nFor example, if the territory is `AU` then then the\nlanguage most spoken is \"en\". First, the locale \"en-AU\"\nis validated and if that fails, \"en\" is validated.","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.locale_for_territory/1","doc":"* `territory` is any ISO 3166 Alpha-2 territory\n code that can be validated by `Cldr.validate_territory/1`","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.locale_for_territory/1","doc":"* `{:ok, language_tag}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.locale_for_territory/1","doc":"iex> Plausible.Cldr.Locale.locale_for_territory(:AU)\n Elixir.Plausible.Cldr.validate_locale(:\"en-AU\")\n\n iex> Plausible.Cldr.Locale.locale_for_territory(:US)\n Elixir.Plausible.Cldr.validate_locale(:\"en-US\")\n\n iex> Plausible.Cldr.Locale.locale_for_territory(:ZZ)\n {:error, {Cldr.UnknownTerritoryError, \"The territory :ZZ is unknown\"}}","ref":"Plausible.Cldr.Locale.html#locale_for_territory/1-examples"},{"type":"function","title":"Plausible.Cldr.Locale.locale_from_host/2","doc":"Returns a \"best fit\" locale for a host name.","ref":"Plausible.Cldr.Locale.html#locale_from_host/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.locale_from_host/2","doc":"* `host` is any valid host name\n\n* `options` is a keyword list of options. The default\n is `[]`.","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Locale.locale_from_host/2","doc":"* `:tlds` is a list of territory codes as upper-cased\n atoms that are to be considered as top-level domains.\n See `Cldr.Locale.locale_from_host/2` for the default\n list.","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.locale_from_host/2","doc":"* `{:ok, langauge_tag}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-returns"},{"type":"function","title":"Notes - Plausible.Cldr.Locale.locale_from_host/2","doc":"Certain top-level domains have become associated with content\nunderlated to the territory for who the domain is registered.\nTherefore Google (and perhaps others) do not associate these\nTLDs as belonging to the territory but rather are considered\ngeneric top-level domain names.","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-notes"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.locale_from_host/2","doc":"iex> Plausible.Cldr.Locale.locale_from_host \"a.b.com.au\"\n Elixir.Plausible.Cldr.validate_locale(:\"en-AU\")\n\n iex> Plausible.Cldr.Locale.locale_from_host(\"a.b.com.tv\")\n {:error,\n {Cldr.UnknownLocaleError, \"No locale was identified for territory \\\"tv\\\"\"}}\n\n iex> Plausible.Cldr.Locale.locale_from_host(\"a.b.com\")\n {:error,\n {Cldr.UnknownLocaleError, \"No locale was identified for territory \\\"com\\\"\"}}","ref":"Plausible.Cldr.Locale.html#locale_from_host/2-examples"},{"type":"function","title":"Plausible.Cldr.Locale.new/1","doc":"","ref":"Plausible.Cldr.Locale.html#new/1"},{"type":"function","title":"Plausible.Cldr.Locale.new!/1","doc":"","ref":"Plausible.Cldr.Locale.html#new!/1"},{"type":"function","title":"Plausible.Cldr.Locale.script_direction_from_locale/1","doc":"Returns the script direction for a locale.","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.script_direction_from_locale/1","doc":"* `language_tag` is any language tag returned by `Cldr.Locale.new/2`\n or any `locale_name` returned by `Cldr.known_locale_names/1`.","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.script_direction_from_locale/1","doc":"* The script direction which is either `:ltr` (for left-to-right\n scripts) or `:rtl` (for right-to-left scripts).","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.script_direction_from_locale/1","doc":"iex> Plausible.Cldr.Locale.script_direction_from_locale \"en-US\"\n :ltr\n\n iex> Plausible.Cldr.Locale.script_direction_from_locale :ar\n :rtl","ref":"Plausible.Cldr.Locale.html#script_direction_from_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.Locale.territory_from_host/1","doc":"Returns the last segment of a host that might\nbe a territory.","ref":"Plausible.Cldr.Locale.html#territory_from_host/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.territory_from_host/1","doc":"* `host` is any valid host name","ref":"Plausible.Cldr.Locale.html#territory_from_host/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.territory_from_host/1","doc":"* `{:ok, territory}` or\n\n* `{:error, {exception, reason}}`","ref":"Plausible.Cldr.Locale.html#territory_from_host/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.territory_from_host/1","doc":"iex> Cldr.Locale.territory_from_host(\"a.b.com.au\")\n {:ok, :AU}\n\n iex> Cldr.Locale.territory_from_host(\"a.b.com\")\n {:error,\n {Cldr.UnknownLocaleError, \"No locale was identified for territory \\\"com\\\"\"}}","ref":"Plausible.Cldr.Locale.html#territory_from_host/1-examples"},{"type":"function","title":"Plausible.Cldr.Locale.territory_from_locale/1","doc":"Returns the territory from a language tag or\nlocale name.","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.territory_from_locale/1","doc":"* `locale` is any language tag returned by\n `Plausible.Cldr.Locale.new/1`\n or a locale name in the list returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.territory_from_locale/1","doc":"* A territory code as an atom","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.territory_from_locale/1","doc":"iex> Plausible.Cldr.Locale.territory_from_locale \"en-US\"\n :US\n\n iex> Plausible.Cldr.Locale.territory_from_locale \"en-US-u-rg-GBzzzz\"\n :GB","ref":"Plausible.Cldr.Locale.html#territory_from_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.Locale.timezone_from_locale/1","doc":"Returns the time zone from a language tag or\nlocale name.","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Locale.timezone_from_locale/1","doc":"* `locale` is any language tag returned by\n `Plausible.Cldr.Locale.new/1`\n or a locale name in the list returned by\n `Plausible.Cldr.known_locale_names/0`","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Locale.timezone_from_locale/1","doc":"* A time zone ID as a string or\n\n* `:error` if no time zone can be determined","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Locale.timezone_from_locale/1","doc":"iex> Plausible.Cldr.Locale.timezone_from_locale \"en-US-u-tz-ausyd\"\n \"Australia/Sydney\"","ref":"Plausible.Cldr.Locale.html#timezone_from_locale/1-examples"},{"type":"module","title":"Plausible.Cldr.Number","doc":"Formats numbers and currencies based upon CLDR's decimal formats specification.\n\nThe format specification is documentated in [Unicode TR35](http://unicode.org/reports/tr35/tr35-numbers.html#Number_Formats).\nThere are several classes of formatting including non-scientific, scientific,\nrules based (for spelling and ordinal formats), compact formats that display `1k`\nrather than `1,000` and so on. See `Cldr.Number.to_string/2` for specific formatting\noptions.\n\n#","ref":"Plausible.Cldr.Number.html"},{"type":"module","title":"Non-Scientific Notation Formatting - Plausible.Cldr.Number","doc":"The following description applies to formats that do not use scientific\nnotation or significant digits:\n\n* If the number of actual integer digits exceeds the maximum integer digits,\n then only the least significant digits are shown. For example, 1997 is\n formatted as \"97\" if the maximum integer digits is set to 2.\n\n* If the number of actual integer digits is less than the minimum integer\n digits, then leading zeros are added. For example, 1997 is formatted as\n \"01997\" if the minimum integer digits is set to 5.\n\n* If the number of actual fraction digits exceeds the maximum fraction\n digits, then half-even rounding it performed to the maximum fraction\n digits. For example, 0.125 is formatted as \"0.12\" if the maximum fraction\n digits is 2. This behavior can be changed by specifying a rounding\n increment and a rounding mode.\n\n* If the number of actual fraction digits is less than the minimum fraction\n digits, then trailing zeros are added. For example, 0.125 is formatted as\n \"0.1250\" if the minimum fraction digits is set to 4.\n\n* Trailing fractional zeros are not displayed if they occur j positions after\n the decimal, where j is less than the maximum fraction digits. For example,\n 0.10004 is formatted as \"0.1\" if the maximum fraction digits is four or\n less.\n\n#","ref":"Plausible.Cldr.Number.html#module-non-scientific-notation-formatting"},{"type":"module","title":"Scientific Notation Formatting - Plausible.Cldr.Number","doc":"Numbers in scientific notation are expressed as the product of a mantissa and\na power of ten, for example, 1234 can be expressed as 1.234 x 10^3. The\nmantissa is typically in the half-open interval [1.0, 10.0) or sometimes\n[0.0, 1.0), but it need not be. In a pattern, the exponent character\nimmediately followed by one or more digit characters indicates scientific\nnotation. Example: \"0.###E0\" formats the number 1234 as \"1.234E3\".\n\n* The number of digit characters after the exponent character gives the\n minimum exponent digit count. There is no maximum. Negative exponents are\n formatted using the localized minus sign, not the prefix and suffix from\n the pattern. This allows patterns such as \"0.###E0 m/s\". To prefix positive\n exponents with a localized plus sign, specify '+' between the exponent and\n the digits: \"0.###E+0\" will produce formats \"1E+1\", \"1E+0\", \"1E-1\", and so\n on. (In localized patterns, use the localized plus sign rather than '+'.)\n\n* The minimum number of integer digits is achieved by adjusting the exponent.\n Example: 0.00123 formatted with \"00.###E0\" yields \"12.3E-4\". This only\n happens if there is no maximum number of integer digits. If there is a\n maximum, then the minimum number of integer digits is fixed at one.\n\n* The maximum number of integer digits, if present, specifies the exponent\n grouping. The most common use of this is to generate engineering notation,\n in which the exponent is a multiple of three, for example, \"##0.###E0\". The\n number 12345 is formatted using \"##0.####E0\" as \"12.345E3\".\n\n* When using scientific notation, the formatter controls the digit counts\n using significant digits logic. The maximum number of significant digits\n limits the total number of integer and fraction digits that will be shown\n in the mantissa; it does not affect parsing. For example, 12345 formatted\n with \"##0.##E0\" is \"12.3E3\". Exponential patterns may not contain grouping\n separators.\n\n#","ref":"Plausible.Cldr.Number.html#module-scientific-notation-formatting"},{"type":"module","title":"Significant Digits - Plausible.Cldr.Number","doc":"There are two ways of controlling how many digits are shows: (a)\nsignificant digits counts, or (b) integer and fraction digit counts. Integer\nand fraction digit counts are described above. When a formatter is using\nsignificant digits counts, it uses however many integer and fraction digits\nare required to display the specified number of significant digits. It may\nignore min/max integer/fraction digits, or it may use them to the extent\npossible.","ref":"Plausible.Cldr.Number.html#module-significant-digits"},{"type":"function","title":"Plausible.Cldr.Number.parse/2","doc":"Parse a string locale-aware manner and return\na number.","ref":"Plausible.Cldr.Number.html#parse/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.parse/2","doc":"* `string` is any `String.t`\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Number.html#parse/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.parse/2","doc":"* `:number` is one of `:integer`, `:float`,\n `:decimal` or `nil`. The default is `nil`\n meaning that the type auto-detected as either\n an `integer` or a `float`.\n\n* `:locale` is any locale returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag.t`. The default is\n `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.html#parse/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Number.parse/2","doc":"* A number of the requested or default type or\n\n* `{:error, {exception, error}}` if no number could be determined","ref":"Plausible.Cldr.Number.html#parse/2-returns"},{"type":"function","title":"Notes - Plausible.Cldr.Number.parse/2","doc":"This function parses a string to return a number but\nin a locale-aware manner. It will normalise grouping\ncharacters and decimal separators, different forms of\nthe `+` and `-` symbols that appear in Unicode and\nstrips any `_` characters that might be used for\nformatting in a string. It then parses the number\nusing the Elixir standard library functions.","ref":"Plausible.Cldr.Number.html#parse/2-notes"},{"type":"function","title":"Examples - Plausible.Cldr.Number.parse/2","doc":"iex> Plausible.Cldr.Number.parse(\"+1.000,34\", locale: \"de\")\n {:ok, 1000.34}\n\n iex> Plausible.Cldr.Number.parse(\"-1_000_000.34\")\n {:ok, -1000000.34}\n\n iex> Plausible.Cldr.Number.parse(\"1.000\", locale: \"de\", number: :integer)\n {:ok, 1000}\n\n iex> Plausible.Cldr.Number.parse(\"+1.000,34\", locale: \"de\", number: :integer)\n {:error,\n {Cldr.Number.ParseError,\n \"The string \\\"+1.000,34\\\" could not be parsed as a number\"}}","ref":"Plausible.Cldr.Number.html#parse/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.resolve_currencies/2","doc":"Resolve curencies from strings within\na list.","ref":"Plausible.Cldr.Number.html#resolve_currencies/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.resolve_currencies/2","doc":"* `list` is any list in which currency\n names and symbols are expected\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.resolve_currencies/2","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `Plausible.Cldr.get_locale()`\n\n* `:only` is an `atom` or list of `atoms` representing the\n currencies or currency types to be considered for a match.\n The equates to a list of acceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:except` is an `atom` or list of `atoms` representing the\n currencies or currency types to be not considered for a match.\n This equates to a list of unacceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:fuzzy` is a float greater than `0.0` and less than or\n equal to `1.0` which is used as input to\n `String.jaro_distance/2` to determine is the provided\n currency string is *close enough* to a known currency\n string for it to identify definitively a currency code.\n It is recommended to use numbers greater than `0.8` in\n order to reduce false positives.","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-options"},{"type":"function","title":"Notes - Plausible.Cldr.Number.resolve_currencies/2","doc":"The `:only` and `:except` options accept a list of\ncurrency codes and/or currency types. The following\ntypes are recognised.\n\nIf both `:only` and `:except` are specified,\nthe `:except` entries take priority - that means\nany entries in `:except` are removed from the `:only`\nentries.\n\n * `:all`, the default, considers all currencies\n\n * `:current` considers those currencies that have a `:to`\n date of nil and which also is a known ISO4217 currency\n\n * `:historic` is the opposite of `:current`\n\n * `:tender` considers currencies that are legal tender\n\n * `:unannotated` considers currencies that don't have\n \"(some string)\" in their names. These are usually\n financial instruments.","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-notes"},{"type":"function","title":"Examples - Plausible.Cldr.Number.resolve_currencies/2","doc":"iex> Plausible.Cldr.Number.scan(\"100 US dollars\")\n ...> |> Plausible.Cldr.Number.resolve_currencies\n [100, :USD]\n\n iex> Plausible.Cldr.Number.scan(\"100 eurosports\")\n ...> |> Plausible.Cldr.Number.resolve_currencies(fuzzy: 0.75)\n [100, :EUR]\n\n iex> Plausible.Cldr.Number.scan(\"100 dollars des États-Unis\")\n ...> |> Plausible.Cldr.Number.resolve_currencies(locale: \"fr\")\n [100, :USD]","ref":"Plausible.Cldr.Number.html#resolve_currencies/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.resolve_currency/2","doc":"Resolve a currency from a string","ref":"Plausible.Cldr.Number.html#resolve_currency/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.resolve_currency/2","doc":"* `list` is any list in which currency\n names and symbols are expected\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Number.html#resolve_currency/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.resolve_currency/2","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `Plausible.Cldr.get_locale()`\n\n* `:only` is an `atom` or list of `atoms` representing the\n currencies or currency types to be considered for a match.\n The equates to a list of acceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:except` is an `atom` or list of `atoms` representing the\n currencies or currency types to be not considered for a match.\n This equates to a list of unacceptable currencies for parsing.\n See the notes below for currency types.\n\n* `:fuzzy` is a float greater than `0.0` and less than or\n equal to `1.0` which is used as input to\n `String.jaro_distance/2` to determine is the provided\n currency string is *close enough* to a known currency\n string for it to identify definitively a currency code.\n It is recommended to use numbers greater than `0.8` in\n order to reduce false positives.","ref":"Plausible.Cldr.Number.html#resolve_currency/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Number.resolve_currency/2","doc":"* An ISO4217 currency code as an atom or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Number.html#resolve_currency/2-returns"},{"type":"function","title":"Notes - Plausible.Cldr.Number.resolve_currency/2","doc":"The `:only` and `:except` options accept a list of\ncurrency codes and/or currency types. The following\ntypes are recognised.\n\nIf both `:only` and `:except` are specified,\nthe `:except` entries take priority - that means\nany entries in `:except` are removed from the `:only`\nentries.\n\n * `:all`, the default, considers all currencies\n\n * `:current` considers those currencies that have a `:to`\n date of nil and which also is a known ISO4217 currency\n\n * `:historic` is the opposite of `:current`\n\n * `:tender` considers currencies that are legal tender\n\n * `:unannotated` considers currencies that don't have\n \"(some string)\" in their names. These are usually\n financial instruments.","ref":"Plausible.Cldr.Number.html#resolve_currency/2-notes"},{"type":"function","title":"Examples - Plausible.Cldr.Number.resolve_currency/2","doc":"iex> Plausible.Cldr.Number.resolve_currency(\"US dollars\")\n [:USD]\n\n iex> Plausible.Cldr.Number.resolve_currency(\"100 eurosports\", fuzzy: 0.75)\n [:EUR]\n\n iex> Plausible.Cldr.Number.resolve_currency(\"dollars des États-Unis\", locale: \"fr\")\n [:USD]\n\n iex> Plausible.Cldr.Number.resolve_currency(\"not a known currency\", locale: \"fr\")\n {:error,\n {Cldr.UnknownCurrencyError,\n \"The currency \\\"not a known currency\\\" is unknown or not supported\"}}","ref":"Plausible.Cldr.Number.html#resolve_currency/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.resolve_per/2","doc":"Resolve and tokenize percent or permille\nfrom the beginning and/or the end of a string","ref":"Plausible.Cldr.Number.html#resolve_per/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.resolve_per/2","doc":"* `list` is any list in which percent\n and permille symbols are expected\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Number.html#resolve_per/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.resolve_per/2","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `options[:backend].get_locale()`","ref":"Plausible.Cldr.Number.html#resolve_per/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Number.resolve_per/2","doc":"* An `:percent` or `permille` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Number.html#resolve_per/2-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.resolve_per/2","doc":"iex> Plausible.Cldr.Number.resolve_per \"11%\"\n [\"11\", :percent]\n\n iex> Plausible.Cldr.Number.resolve_per \"% of linguists\"\n [:percent, \" of linguists\"]\n\n iex> Plausible.Cldr.Number.resolve_per \"% of linguists %\"\n [:percent, \" of linguists \", :percent]","ref":"Plausible.Cldr.Number.html#resolve_per/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.resolve_pers/2","doc":"Resolve and tokenize percent and permille\nsybols from strings within a list.\n\nPercent and permille symbols can be identified\nat the beginning and/or the end of a string.","ref":"Plausible.Cldr.Number.html#resolve_pers/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.resolve_pers/2","doc":"* `list` is any list in which percent and\n permille symbols are expected\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Number.html#resolve_pers/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.resolve_pers/2","doc":"* `:locale` is any valid locale returned by `Cldr.known_locale_names/1`\n or a `t:Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n The default is `options[:backend].get_locale()`","ref":"Plausible.Cldr.Number.html#resolve_pers/2-options"},{"type":"function","title":"Examples - Plausible.Cldr.Number.resolve_pers/2","doc":"iex> Plausible.Cldr.Number.scan(\"100%\")\n ...> |> Plausible.Cldr.Number.resolve_pers()\n [100, :percent]","ref":"Plausible.Cldr.Number.html#resolve_pers/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.scan/2","doc":"Scans a string locale-aware manner and returns\na list of strings and numbers.","ref":"Plausible.Cldr.Number.html#scan/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.scan/2","doc":"* `string` is any `String.t`\n\n* `options` is a keyword list of options","ref":"Plausible.Cldr.Number.html#scan/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.scan/2","doc":"* `:number` is one of `:integer`, `:float`,\n `:decimal` or `nil`. The default is `nil`\n meaning that the type auto-detected as either\n an `integer` or a `float`.\n\n* `:locale` is any locale returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag.t`. The default is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.html#scan/2-options"},{"type":"function","title":"Returns - Plausible.Cldr.Number.scan/2","doc":"* A list of strings and numbers","ref":"Plausible.Cldr.Number.html#scan/2-returns"},{"type":"function","title":"Notes - Plausible.Cldr.Number.scan/2","doc":"Number parsing is performed by `Cldr.Number.Parser.parse/2`\nand any options provided are passed to that function.","ref":"Plausible.Cldr.Number.html#scan/2-notes"},{"type":"function","title":"Examples - Plausible.Cldr.Number.scan/2","doc":"iex> Plausible.Cldr.Number.scan(\"£1_000_000.34\")\n [\"£\", 1000000.34]\n\n iex> Plausible.Cldr.Number.scan(\"I want £1_000_000 dollars\")\n [\"I want £\", 1000000, \" dollars\"]\n\n iex> Plausible.Cldr.Number.scan(\"The prize is 23\")\n [\"The prize is \", 23]\n\n iex> Plausible.Cldr.Number.scan(\"The lottery number is 23 for the next draw\")\n [\"The lottery number is \", 23, \" for the next draw\"]\n\n iex> Plausible.Cldr.Number.scan(\"The loss is -1.000 euros\", locale: \"de\", number: :integer)\n [\"The loss is \", -1000, \" euros\"]","ref":"Plausible.Cldr.Number.html#scan/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.to_approx_string/2","doc":"Formats a number and applies the `:approximately` format for\na locale and number system.","ref":"Plausible.Cldr.Number.html#to_approx_string/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.to_approx_string/2","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Cldr.Number.to_string/3` for a description of the available\n options.","ref":"Plausible.Cldr.Number.html#to_approx_string/2-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.to_approx_string/2","doc":"iex> Plausible.Cldr.Number.to_approx_string 1234\n {:ok, \"~1,234\"}","ref":"Plausible.Cldr.Number.html#to_approx_string/2-example"},{"type":"function","title":"Plausible.Cldr.Number.to_at_least_string/2","doc":"Formats a number and applies the `:at_least` format for\na locale and number system.","ref":"Plausible.Cldr.Number.html#to_at_least_string/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.to_at_least_string/2","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Plausible.Cldr.Number.to_string/2` for a description of the available\n options.","ref":"Plausible.Cldr.Number.html#to_at_least_string/2-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.to_at_least_string/2","doc":"iex> Plausible.Cldr.Number.to_at_least_string 1234\n {:ok, \"1,234+\"}","ref":"Plausible.Cldr.Number.html#to_at_least_string/2-example"},{"type":"function","title":"Plausible.Cldr.Number.to_at_most_string/2","doc":"Formats a number and applies the `:at_most` format for\na locale and number system.","ref":"Plausible.Cldr.Number.html#to_at_most_string/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.to_at_most_string/2","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Cldr.Number.to_string/3` for a description of the available\n options.","ref":"Plausible.Cldr.Number.html#to_at_most_string/2-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.to_at_most_string/2","doc":"iex> Plausible.Cldr.Number.to_at_most_string 1234\n {:ok, \"≤1,234\"}","ref":"Plausible.Cldr.Number.html#to_at_most_string/2-example"},{"type":"function","title":"Plausible.Cldr.Number.to_range_string/2","doc":"Formats the first and last numbers of a range and applies\nthe `:range` format for a locale and number system.","ref":"Plausible.Cldr.Number.html#to_range_string/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.to_range_string/2","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.\n See `Cldr.Number.to_string/3` for a description of the available\n options.","ref":"Plausible.Cldr.Number.html#to_range_string/2-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.to_range_string/2","doc":"iex> Plausible.Cldr.Number.to_range_string 1234..5678\n {:ok, \"1,234–5,678\"}","ref":"Plausible.Cldr.Number.html#to_range_string/2-example"},{"type":"function","title":"Plausible.Cldr.Number.to_string/2","doc":"Returns a number formatted into a string according to a format pattern and options.","ref":"Plausible.Cldr.Number.html#to_string/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.to_string/2","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted.","ref":"Plausible.Cldr.Number.html#to_string/2-arguments"},{"type":"function","title":"Options - Plausible.Cldr.Number.to_string/2","doc":"* `format`: the format style or a format string defining how the number is\n formatted. See `Cldr.Number.Format` for how format strings can be constructed.\n See `Cldr.Number.Format.format_styles_for/3` to return available format styles\n for a locale. The default `format` is `:standard`.\n\n* If `:format` is set to `:long` or `:short` then the formatting depends on\n whether `:currency` is specified. If not specified then the number is\n formatted as `:decimal_long` or `:decimal_short`. If `:currency` is\n specified the number is formatted as `:currency_long` or\n `:currency_short` and `:fractional_digits` is set to 0 as a default.\n\n* `:format` may also be a format defined by CLDR's Rules Based Number\n Formats (RBNF). Further information is found in the module `Cldr.Rbnf`.\n The most commonly used formats in this category are to spell out the\n number in a the locales language. The applicable formats are `:spellout`,\n `:spellout_year`, `:ordinal`. A number can also be formatted as roman\n numbers by using the format `:roman` or `:roman_lower`.\n\n* `currency`: is the currency for which the number is formatted. For\n available currencies see `Cldr.Currency.known_currencies/0`. This option\n is required if `:format` is set to `:currency`. If `currency` is set\n and no `:format` is set, `:format` will be set to `:currency` as well.\n\n* `currency_symbol`: Allows overriding a currency symbol. The alternatives\n are:\n * `:iso` the ISO currency code will be used instead of the default\n currency symbol.\n * `:narrow` uses the narrow symbol defined for the locale. The same\n narrow symbol can be defined for more than one currency and therefore this\n should be used with care. If no narrow symbol is defined, the standard\n symbol is used.\n * `:symbol` uses the standard symbol defined in CLDR. A symbol is unique\n for each currency and can be safely used.\n * \"string\" uses `string` as the currency symbol\n * `:standard` (the default and recommended) uses the CLDR-defined symbol\n based upon the currency format for the locale.\n\n* `:cash`: a boolean which indicates whether a number being formatted as a\n `:currency` is to be considered a cash value or not. Currencies can be\n rounded differently depending on whether `:cash` is `true` or `false`.\n *This option is deprecated in favour of `currency_digits: :cash`.\n\n* `:currency_digits` indicates which of the rounding and digits should be\n used. The options are `:accounting` which is the default, `:cash` or\n `:iso`\n\n* `:rounding_mode`: determines how a number is rounded to meet the precision\n of the format requested. The available rounding modes are `:down`,\n :half_up, :half_even, :ceiling, :floor, :half_down, :up. The default is\n `:half_even`.\n\n* `:number_system`: determines which of the number systems for a locale\n should be used to define the separators and digits for the formatted\n number. If `number_system` is an `atom` then `number_system` is\n interpreted as a number system. If the `:number_system` is\n `binary` then it is interpreted as a number system name. See\n `Cldr.Number.System.number_system_names_for/2`. The default is `:default`.\n\n* `:locale`: determines the locale in which the number is formatted. See\n `Cldr.known_locale_names/0`. The default is`Cldr.get_locale/0` which is the\n locale currently in affect for this `Process` and which is set by\n `Cldr.put_locale/1`.\n\n* If `:fractional_digits` is set to a positive integer value then the number\n will be rounded to that number of digits and displayed accordingly - overriding\n settings that would be applied by default. For example, currencies have\n fractional digits defined reflecting each currencies minor unit. Setting\n `:fractional_digits` will override that setting.\n\n* If `:maximum_integer_digits` is set to a positive integer value then the\n number is left truncated before formatting. For example if the number `1234`\n is formatted with the option `maximum_integer_digits: 2`, the number is\n truncated to `34` and formatted.\n\n* If `:round_nearest` is set to a positive integer value then the number\n will be rounded to nearest increment of that value - overriding\n settings that would be applied by default.\n\n* `:minimum_grouping_digits` overrides the CLDR definition of minimum grouping\n digits. For example in the locale `es` the number `1234` is formatted by default\n as `1345` because the locale defines the `minimium_grouping_digits` as `2`. If\n `minimum_grouping_digits: 1` is set as an option the number is formatting as\n `1.345`. The `:minimum_grouping_digits` is added to the grouping defined by\n the number format. If the sum of these two digits is greater than the number\n of digits in the integer (or fractional) part of the number then no grouping\n is performed.\n\n* `:wrapper` is a 2-arity function that will be called for each number component\n with parameters `string` and `tag` where `tag` is one of `:number`,\n `:currency_symbol`, `:currency_space`, `:literal`, `:quote`, `:percent`,\n `:permille`, `:minus` or `:plus`. The function must return a string. The\n function can be used to wrap format elements in HTML or other tags.","ref":"Plausible.Cldr.Number.html#to_string/2-options"},{"type":"function","title":"Locale extensions affecting formatting - Plausible.Cldr.Number.to_string/2","doc":"A locale identifier can specify options that affect number formatting.\nThese options are:\n\n* `nu`: defines the number system to be used if none is specified by the `:number_system`\n option to `to_string/2`\n\nThis key is part of the [u extension](https://unicode.org/reports/tr35/#u_Extension) and\nthat document should be consulted for details on how to construct a locale identifier with these\nextensions.","ref":"Plausible.Cldr.Number.html#to_string/2-locale-extensions-affecting-formatting"},{"type":"function","title":"Wrapping format elements - Plausible.Cldr.Number.to_string/2","doc":"Wrapping elements is particularly useful when formatting a number with a\ncurrency symbol and the requirement is to have different HTML formatting\napplied to the symbol than the number. For example:\n\n iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn\n ...> string, :currency_symbol -> \" \" <> string <> \" \"\n ...> string, :number -> \" \" <> string <> \" \"\n ...> string, :currency_space -> \" \" <> string <> \" \"\n ...> string, _other -> string\n ...> end)\n {:ok, \" $ 100.00 \"}\n\nIt is also possible and recommended to use the `Phoenix.HTML.Tag.content_tag/3`\nfunction if wrapping HTML tags since these will ensure HTML entities are\ncorrectly encoded. For example:\n\n iex> Cldr.Number.to_string(100, format: :currency, currency: :USD, wrapper: fn\n ...> string, :currency_symbol -> Phoenix.HTML.Tag.content_tag(:span, string, class: \"symbol\")\n ...> string, :number -> Phoenix.HTML.Tag.content_tag(:span, string, class: \"number\")\n ...> string, :currency_space -> Phoenix.HTML.Tag.content_tag(:span, string)\n ...> string, _other -> string\n ...> end)\n {:ok, \" $ 100.00 \"}\n\nWhen formatting a number the format is parsed into format elements that might include\na currency symbol, a literal string, inserted text between a currency symbol and the\ncurrency amount, a percent sign, the number itself and several other elements. In\nsome cases it is helpful to be apply specific formatting to each element.\nThis can be achieved by specifying a `:wrapper` option. This option takes a 2-arity\nfunction as an argument. For each element of the format the wrapper function is called\nwith two parameters: the format element as a string and an atom representing the\nelement type. The wrapper function is required to return a string that is then\ninserted in the final formatted number.","ref":"Plausible.Cldr.Number.html#to_string/2-wrapping-format-elements"},{"type":"function","title":"Returns - Plausible.Cldr.Number.to_string/2","doc":"* `{:ok, string}` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Number.html#to_string/2-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.to_string/2","doc":"iex> Plausible.Cldr.Number.to_string 12345\n {:ok, \"12,345\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, locale: \"fr\"\n {:ok, \"12 345\"}\n\n iex> Plausible.Cldr.Number.to_string 1345.32, currency: :EUR, locale: \"es\", minimum_grouping_digits: 1\n {:ok, \"1.345,32 €\"}\n\n iex> Plausible.Cldr.Number.to_string 1345.32, currency: :EUR, locale: \"es\"\n {:ok, \"1345,32 €\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, locale: \"fr\", currency: \"USD\"\n {:ok, \"12 345,00 $US\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: \"#E0\"\n {:ok, \"1.2345E4\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: \"THB\"\n {:ok, \"THB 12,345.00\"}\n\n iex> Plausible.Cldr.Number.to_string -12345, format: :accounting, currency: \"THB\"\n {:ok, \"(THB 12,345.00)\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: \"THB\",\n ...> locale: \"th\"\n {:ok, \"฿12,345.00\"}\n\n iex> Plausible.Cldr.Number.to_string 12345, format: :accounting, currency: \"THB\",\n ...> locale: \"th\", number_system: :native\n {:ok, \"฿๑๒,๓๔๕.๐๐\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :long\n {:ok, \"1 thousand\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :long, currency: \"USD\"\n {:ok, \"1,244 US dollars\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :short\n {:ok, \"1K\"}\n\n iex> Plausible.Cldr.Number.to_string 1244.30, format: :short, currency: \"EUR\"\n {:ok, \"€1K\"}\n\n iex> Plausible.Cldr.Number.to_string 1234, format: :spellout\n {:ok, \"one thousand two hundred thirty-four\"}\n\n iex> Plausible.Cldr.Number.to_string 1234, format: :spellout_verbose\n {:ok, \"one thousand two hundred and thirty-four\"}\n\n iex> Plausible.Cldr.Number.to_string 1989, format: :spellout_year\n {:ok, \"nineteen eighty-nine\"}\n\n iex> Plausible.Cldr.Number.to_string 123, format: :ordinal\n {:ok, \"123rd\"}\n\n iex> Plausible.Cldr.Number.to_string 123, format: :roman\n {:ok, \"CXXIII\"}\n\n iex> Plausible.Cldr.Number.to_string 123, locale: \"th-u-nu-thai\"\n {:ok, \"๑๒๓\"}","ref":"Plausible.Cldr.Number.html#to_string/2-examples"},{"type":"function","title":"Errors - Plausible.Cldr.Number.to_string/2","doc":"An error tuple `{:error, reason}` will be returned if an error is detected.\nThe two most likely causes of an error return are:\n\n * A format cannot be compiled. In this case the error tuple will look like:\n\n```\n iex> Plausible.Cldr.Number.to_string(12345, format: \"0#\")\n {:error, {Cldr.FormatCompileError,\n \"Decimal format compiler: syntax error before: \\\"#\\\"\"}}\n```\n\n * The format style requested is not defined for the `locale` and\n `number_system`. This happens typically when the number system is\n `:algorithmic` rather than the more common `:numeric`. In this case the error\n return looks like:\n\n```\n iex> Plausible.Cldr.Number.to_string(1234, locale: \"he\", number_system: \"hebr\", format: :percent)\n {:error, {Cldr.UnknownFormatError,\n \"The locale :he with number system :hebr does not define a format :percent\"}}\n```","ref":"Plausible.Cldr.Number.html#to_string/2-errors"},{"type":"function","title":"Plausible.Cldr.Number.to_string!/2","doc":"Same as the execution of `to_string/2` but raises an exception if an error would be\nreturned.","ref":"Plausible.Cldr.Number.html#to_string!/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.to_string!/2","doc":"* `number` is an integer, float or Decimal to be formatted\n\n* `options` is a keyword list defining how the number is to be formatted. See\n `Plausible.Cldr.Number.to_string/2`","ref":"Plausible.Cldr.Number.html#to_string!/2-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.to_string!/2","doc":"* a formatted number as a string or\n\n* raises an exception","ref":"Plausible.Cldr.Number.html#to_string!/2-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.to_string!/2","doc":"iex> Plausible.Cldr.Number.to_string! 12345\n \"12,345\"\n\n iex> Plausible.Cldr.Number.to_string! 12345, locale: \"fr\"\n \"12 345\"","ref":"Plausible.Cldr.Number.html#to_string!/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.validate_number_system/2","doc":"Return a valid number system from a provided locale and number\nsystem name or type.\n\nThe number system or number system type must be valid for the\ngiven locale. If a number system type is provided, the\nunderlying number system is returned.","ref":"Plausible.Cldr.Number.html#validate_number_system/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.validate_number_system/2","doc":"* `locale` is any valid locale name returned by `Cldr.known_locale_names/1`\n or a `Cldr.LanguageTag` struct returned by `Cldr.Locale.new!/2`\n\n* `system_name` is any number system name returned by\n `Cldr.known_number_systems/0` or a number system type\n returned by `Cldr.known_number_system_types/0`","ref":"Plausible.Cldr.Number.html#validate_number_system/2-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.Number.validate_number_system/2","doc":"iex> Plausible.Cldr.Number.validate_number_system \"en\", :latn\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.validate_number_system \"en\", :default\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.validate_number_system \"en\", :unknown\n {:error,\n {Cldr.UnknownNumberSystemError, \"The number system :unknown is unknown\"}}\n\n iex> Plausible.Cldr.Number.validate_number_system \"zz\", :default\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zz\\\" is invalid\"}}","ref":"Plausible.Cldr.Number.html#validate_number_system/2-examples"},{"type":"module","title":"Plausible.Cldr.Number.Cardinal","doc":"Implements cardinal plural rules for numbers.","ref":"Plausible.Cldr.Number.Cardinal.html"},{"type":"function","title":"Plausible.Cldr.Number.Cardinal.available_locale_names/0","doc":"The locale names for which plural rules are defined.","ref":"Plausible.Cldr.Number.Cardinal.html#available_locale_names/0"},{"type":"function","title":"Plausible.Cldr.Number.Cardinal.known_locale_names/0","doc":"The configured locales for which plural rules are defined.\n\nReturns the intersection of `Plausible.Cldr.known_locale_names/0` and\nthe locales for which Cardinal plural rules are defined.\n\nThere are many `Cldr` locales which don't have their own plural\nrules so this list is the intersection of `Cldr`'s configured\nlocales and those that have rules.","ref":"Plausible.Cldr.Number.Cardinal.html#known_locale_names/0"},{"type":"function","title":"Plausible.Cldr.Number.Cardinal.plural_rule/3","doc":"Return the plural key for a given number in a given locale\n\nReturns which plural key (`:zero`, `:one`, `:two`, `:few`,\n`:many` or `:other`) a given number fits into within the\ncontext of a given locale.\n\nNote that these key names should not be interpreted\nliterally. For example, the key returned from\n`Cldr.Number.Ordinal.plural_rule(0, \"en\")` is actually\n`:other`, not `:zero`.\n\nThis key can then be used to format a number, date, time, unit,\nlist or other content in a plural-sensitive way.","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rule/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Cardinal.plural_rule/3","doc":"* `number` is any `integer`, `float` or `Decimal`\n\n* `locale` is any locale returned by `Cldr.Locale.new!/2` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `rounding` is one of `[:down, :up, :ceiling, :floor, :half_even, :half_up, :half_down]`. The\n default is `:half_even`.","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rule/3-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Cardinal.plural_rule/3","doc":"iex> Plausible.Cldr.Number.Cardinal.plural_rule 0, \"fr\"\n :one\n\n iex> Plausible.Cldr.Number.Cardinal.plural_rule 0, \"en\"\n :other","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rule/3-examples"},{"type":"function","title":"Plausible.Cldr.Number.Cardinal.plural_rules/0","doc":"Returns all the plural rules defined in CLDR.","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rules/0"},{"type":"function","title":"Plausible.Cldr.Number.Cardinal.plural_rules_for/1","doc":"Return the plural rules for a locale.","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rules_for/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Cardinal.plural_rules_for/1","doc":"* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\nThe rules are returned in AST form after parsing.","ref":"Plausible.Cldr.Number.Cardinal.html#plural_rules_for/1-arguments"},{"type":"function","title":"Plausible.Cldr.Number.Cardinal.pluralize/3","doc":"Pluralize a number using cardinal plural rules\nand a substitution map.","ref":"Plausible.Cldr.Number.Cardinal.html#pluralize/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Cardinal.pluralize/3","doc":"* `number` is an integer, float or Decimal\n\n* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `substitutions` is a map that maps plural keys to a string.\n The valid substitution keys are `:zero`, `:one`, `:two`,\n `:few`, `:many` and `:other`.\n\nSee also `Plausible.Cldr.Number.Cardinal.Cardinal.plural_rule/3`.","ref":"Plausible.Cldr.Number.Cardinal.html#pluralize/3-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Cardinal.pluralize/3","doc":"iex> Plausible.Cldr.Number.Cardinal.pluralize 1, \"en\", %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 2, \"en\", %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 2, \"en\", %{one: \"one\", two: \"two\", other: \"other\"}\n \"other\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 22, \"en\", %{one: \"one\", two: \"two\", other: \"other\"}\n \"other\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(1), \"en\", %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), \"en\", %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize Decimal.new(2), \"en\", %{one: \"one\", two: \"two\"}\n nil\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, \"ar\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"few\"\n\n iex> Plausible.Cldr.Number.Cardinal.pluralize 1..10, \"en\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"other\"","ref":"Plausible.Cldr.Number.Cardinal.html#pluralize/3-examples"},{"type":"module","title":"Plausible.Cldr.Number.Format","doc":"Functions to manage the collection of number patterns defined in Cldr.\n\nNumber patterns affect how numbers are interpreted in a localized context.\nHere are some examples, based on the French locale. The \".\" shows where the\ndecimal point should go. The \",\" shows where the thousands separator should\ngo. A \"0\" indicates zero-padding: if the number is too short, a zero (in the\nlocale's numeric set) will go there. A \"#\" indicates no padding: if the\nnumber is too short, nothing goes there. A \"¤\" shows where the currency sign\nwill go. The following illustrates the effects of different patterns for the\nFrench locale, with the number \"1234.567\". Notice how the pattern characters\n',' and '.' are replaced by the characters appropriate for the locale.","ref":"Plausible.Cldr.Number.Format.html"},{"type":"module","title":"Number Pattern Examples - Plausible.Cldr.Number.Format","doc":"| Pattern\t | Currency\t | Text |\n| ------------- | :-------------: | ----------: |\n| #,##0.##\t | n/a\t | 1 234,57 |\n| #,##0.###\t | n/a\t | 1 234,567 |\n| ###0.#####\t | n/a\t | 1234,567 |\n| ###0.0000#\t | n/a\t | 1234,5670 |\n| 00000.0000\t | n/a\t | 01234,5670 |\n| #,##0.00 ¤\t | EUR\t | 1 234,57 € |\n\nThe number of # placeholder characters before the decimal do not matter,\nsince no limit is placed on the maximum number of digits. There should,\nhowever, be at least one zero some place in the pattern. In currency formats,\nthe number of digits after the decimal also do not matter, since the\ninformation in the supplemental data (see Supplemental Currency Data) is used\nto override the number of decimal places — and the rounding — according to\nthe currency that is being formatted. That can be seen in the above chart,\nwith the difference between Yen and Euro formatting.\n\nDetails of the number formats are described in the\n[Unicode documentation](http://unicode.org/reports/tr35/tr35-numbers.html#Number_Format_Patterns)","ref":"Plausible.Cldr.Number.Format.html#module-number-pattern-examples"},{"type":"function","title":"Plausible.Cldr.Number.Format.all_formats_for/1","doc":"Returns the decimal formats defined for a given locale.","ref":"Plausible.Cldr.Number.Format.html#all_formats_for/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.all_formats_for/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Format.html#all_formats_for/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.Format.all_formats_for/1","doc":"* `{:ok, map}` where map is a map of decimal formats\n keyed by number system or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Number.Format.html#all_formats_for/1-returns"},{"type":"function","title":"Plausible.Cldr.Number.Format.all_formats_for!/1","doc":"Returns the decimal formats defined for a given locale.","ref":"Plausible.Cldr.Number.Format.html#all_formats_for!/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.all_formats_for!/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Format.html#all_formats_for!/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.Format.all_formats_for!/1","doc":"* `{:ok, map}` where map is a map of decimal formats\n keyed by number system or\n\n* raises an exception.\n\nSee `Plausible.Cldr.Number.Format.Number.Format.all_formats_for/1` for further information.","ref":"Plausible.Cldr.Number.Format.html#all_formats_for!/1-returns"},{"type":"function","title":"Plausible.Cldr.Number.Format.currency_spacing/2","doc":"Returns the currency space for a given locale and\nnumber system.","ref":"Plausible.Cldr.Number.Format.html#currency_spacing/2"},{"type":"function","title":"Plausible.Cldr.Number.Format.decimal_format_list/0","doc":"Returns the list of decimal formats in the configured locales including\nthe list of locales configured for precompilation in `config.exs`.\n\nThis function exists to allow the decimal formatter\nto precompile all the known formats at compile time.","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list/0"},{"type":"function","title":"Example - Plausible.Cldr.Number.Format.decimal_format_list/0","doc":"#=> Plausible.Cldr.Number.Format.Format.decimal_format_list\n [\"#\", \"#,##,##0%\",\n \"#,##,##0.###\", \"#,##,##0.00¤\", \"#,##,##0.00¤;(#,##,##0.00¤)\",\n \"#,##,##0 %\", \"#,##0%\", \"#,##0.###\", \"#,##0.00 ¤\",\n \"#,##0.00 ¤;(#,##0.00 ¤)\", \"#,##0.00¤\", \"#,##0.00¤;(#,##0.00¤)\",\n \"#,##0 %\", \"#0%\", \"#0.######\", \"#0.00 ¤\", \"#E0\", \"%#,##0\", \"% #,##0\",\n \"0\", \"0.000000E+000\", \"0000 M ¤\", \"0000¤\", \"000G ¤\", \"000K ¤\", \"000M ¤\",\n \"000T ¤\", \"000mM ¤\", \"000m ¤\", \"000 Bio'.' ¤\", \"000 Bln ¤\", \"000 Bn ¤\",\n \"000 B ¤\", \"000 E ¤\", \"000 K ¤\", \"000 MRD ¤\", \"000 Md ¤\", \"000 Mio'.' ¤\",\n \"000 Mio ¤\", \"000 Mld ¤\", \"000 Mln ¤\", \"000 Mn ¤\", \"000 Mrd'.' ¤\",\n \"000 Mrd ¤\", \"000 Mr ¤\", \"000 M ¤\", \"000 NT ¤\", \"000 N ¤\", \"000 Tn ¤\",\n \"000 Tr ¤\", ...]","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list/0-example"},{"type":"function","title":"Plausible.Cldr.Number.Format.decimal_format_list_for/1","doc":"Returns the list of decimal formats for a configured locale.","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list_for/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.decimal_format_list_for/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.\n\nThis function exists to allow the decimal formatter to precompile all\nthe known formats at compile time. Its use is not otherwise recommended.","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list_for/1-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.Format.decimal_format_list_for/1","doc":"iex> Plausible.Cldr.Number.Format.decimal_format_list_for(:en)\n {:ok, [\"#,##0%\", \"#,##0.###\", \"#,##0.00\", \"#,##0.00;(#,##0.00)\",\"#E0\",\n \"0 billion\", \"0 million\", \"0 thousand\",\n \"0 trillion\", \"00 billion\", \"00 million\", \"00 thousand\", \"00 trillion\",\n \"000 billion\", \"000 million\", \"000 thousand\", \"000 trillion\", \"000B\", \"000K\",\n \"000M\", \"000T\", \"00B\", \"00K\", \"00M\", \"00T\", \"0B\", \"0K\", \"0M\", \"0T\",\n \"¤#,##0.00\", \"¤#,##0.00;(¤#,##0.00)\", \"¤000B\", \"¤000K\", \"¤000M\",\n \"¤000T\", \"¤00B\", \"¤00K\", \"¤00M\", \"¤00T\", \"¤0B\", \"¤0K\", \"¤0M\", \"¤0T\",\n \"¤ #,##0.00\", \"¤ #,##0.00;(¤ #,##0.00)\", \"¤ 000B\", \"¤ 000K\", \"¤ 000M\",\n \"¤ 000T\", \"¤ 00B\", \"¤ 00K\", \"¤ 00M\", \"¤ 00T\", \"¤ 0B\", \"¤ 0K\", \"¤ 0M\", \"¤ 0T\"]}","ref":"Plausible.Cldr.Number.Format.html#decimal_format_list_for/1-example"},{"type":"function","title":"Plausible.Cldr.Number.Format.default_grouping_for/1","doc":"Returns the default grouping for a locale as a map.","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.default_grouping_for/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.Format.default_grouping_for/1","doc":"* `{:ok, grouping}` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Format.default_grouping_for/1","doc":"iex> Plausible.Cldr.Number.Format.default_grouping_for(:en)\n {:ok, %{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}}","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for/1-examples"},{"type":"function","title":"Plausible.Cldr.Number.Format.default_grouping_for!/1","doc":"Returns the default grouping for a locale\nor raises on error.","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.default_grouping_for!/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.Format.default_grouping_for!/1","doc":"* `grouping` as a map or\n\n* raises an exception.","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Format.default_grouping_for!/1","doc":"iex> Plausible.Cldr.Number.Format.default_grouping_for!(:en)\n %{fraction: %{first: 0, rest: 0}, integer: %{first: 3, rest: 3}}","ref":"Plausible.Cldr.Number.Format.html#default_grouping_for!/1-examples"},{"type":"function","title":"Plausible.Cldr.Number.Format.formats_for/2","doc":"Return the predfined formats for a given `locale` and `number_system`.","ref":"Plausible.Cldr.Number.Format.html#formats_for/2"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.formats_for/2","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.\n\n* `number_system` is any valid number system or number system type returned\n by `Plausible.Cldr.Number.System.number_systems_for/1`.","ref":"Plausible.Cldr.Number.Format.html#formats_for/2-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.Format.formats_for/2","doc":"Plausible.Cldr.Number.Format.formats_for :fr, :native\n #=> %Cldr.Number.Format{\n accounting: \"#,##0.00 ¤;(#,##0.00 ¤)\",\n currency: \"#,##0.00 ¤\",\n percent: \"#,##0 %\",\n scientific: \"#E0\",\n standard: \"#,##0.###\"\n currency_short: [{\"1000\", [one: \"0 k ¤\", other: \"0 k ¤\"]},\n {\"10000\", [one: \"00 k ¤\", other: \"00 k ¤\"]},\n {\"100000\", [one: \"000 k ¤\", other: \"000 k ¤\"]},\n {\"1000000\", [one: \"0 M ¤\", other: \"0 M ¤\"]},\n {\"10000000\", [one: \"00 M ¤\", other: \"00 M ¤\"]},\n {\"100000000\", [one: \"000 M ¤\", other: \"000 M ¤\"]},\n {\"1000000000\", [one: \"0 Md ¤\", other: \"0 Md ¤\"]},\n {\"10000000000\", [one: \"00 Md ¤\", other: \"00 Md ¤\"]},\n {\"100000000000\", [one: \"000 Md ¤\", other: \"000 Md ¤\"]},\n {\"1000000000000\", [one: \"0 Bn ¤\", other: \"0 Bn ¤\"]},\n {\"10000000000000\", [one: \"00 Bn ¤\", other: \"00 Bn ¤\"]},\n {\"100000000000000\", [one: \"000 Bn ¤\", other: \"000 Bn ¤\"]}],\n ...\n }","ref":"Plausible.Cldr.Number.Format.html#formats_for/2-example"},{"type":"function","title":"Plausible.Cldr.Number.Format.formats_for!/2","doc":"","ref":"Plausible.Cldr.Number.Format.html#formats_for!/2"},{"type":"function","title":"Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","doc":"Returns the minimum grouping digits for a locale.","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","doc":"* `{:ok, minumum_digits}` or\n\n* `{:error, {exception, message}}`","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Format.minimum_grouping_digits_for/1","doc":"iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for(\"en\")\n {:ok, 1}","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for/1-examples"},{"type":"function","title":"Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","doc":"Returns the minimum grouping digits for a locale\nor raises on error.","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","doc":"* `minumum_digits` or\n\n* raises an exception.","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Format.minimum_grouping_digits_for!/1","doc":"iex> Plausible.Cldr.Number.Format.minimum_grouping_digits_for!(\"en\")\n 1","ref":"Plausible.Cldr.Number.Format.html#minimum_grouping_digits_for!/1-examples"},{"type":"module","title":"Plausible.Cldr.Number.Formatter.Decimal","doc":"","ref":"Plausible.Cldr.Number.Formatter.Decimal.html"},{"type":"function","title":"Plausible.Cldr.Number.Formatter.Decimal.metadata!/1","doc":"","ref":"Plausible.Cldr.Number.Formatter.Decimal.html#metadata!/1"},{"type":"function","title":"Plausible.Cldr.Number.Formatter.Decimal.to_string/3","doc":"Formats a number according to a decimal format string.","ref":"Plausible.Cldr.Number.Formatter.Decimal.html#to_string/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Formatter.Decimal.to_string/3","doc":"* `number` is an integer, float or Decimal\n\n* `format` is a format string. See `Plausible.Cldr.Number` for further information.\n\n* `options` is a map of options. See `Plausible.Cldr.Number.to_string/2`\n for further information.","ref":"Plausible.Cldr.Number.Formatter.Decimal.html#to_string/3-arguments"},{"type":"module","title":"Plausible.Cldr.Number.Ordinal","doc":"Implements ordinal plural rules for numbers.","ref":"Plausible.Cldr.Number.Ordinal.html"},{"type":"function","title":"Plausible.Cldr.Number.Ordinal.available_locale_names/0","doc":"The locale names for which plural rules are defined.","ref":"Plausible.Cldr.Number.Ordinal.html#available_locale_names/0"},{"type":"function","title":"Plausible.Cldr.Number.Ordinal.known_locale_names/0","doc":"The configured locales for which plural rules are defined.\n\nReturns the intersection of `Plausible.Cldr.known_locale_names/0` and\nthe locales for which Ordinal plural rules are defined.\n\nThere are many `Cldr` locales which don't have their own plural\nrules so this list is the intersection of `Cldr`'s configured\nlocales and those that have rules.","ref":"Plausible.Cldr.Number.Ordinal.html#known_locale_names/0"},{"type":"function","title":"Plausible.Cldr.Number.Ordinal.plural_rule/3","doc":"Return the plural key for a given number in a given locale\n\nReturns which plural key (`:zero`, `:one`, `:two`, `:few`,\n`:many` or `:other`) a given number fits into within the\ncontext of a given locale.\n\nNote that these key names should not be interpreted\nliterally. For example, the key returned from\n`Cldr.Number.Ordinal.plural_rule(0, \"en\")` is actually\n`:other`, not `:zero`.\n\nThis key can then be used to format a number, date, time, unit,\nlist or other content in a plural-sensitive way.","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rule/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Ordinal.plural_rule/3","doc":"* `number` is any `integer`, `float` or `Decimal`\n\n* `locale` is any locale returned by `Cldr.Locale.new!/2` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `rounding` is one of `[:down, :up, :ceiling, :floor, :half_even, :half_up, :half_down]`. The\n default is `:half_even`.","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rule/3-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Ordinal.plural_rule/3","doc":"iex> Plausible.Cldr.Number.Ordinal.plural_rule 0, \"fr\"\n :other\n\n iex> Plausible.Cldr.Number.Ordinal.plural_rule 1, \"en\"\n :one","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rule/3-examples"},{"type":"function","title":"Plausible.Cldr.Number.Ordinal.plural_rules/0","doc":"Returns all the plural rules defined in CLDR.","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rules/0"},{"type":"function","title":"Plausible.Cldr.Number.Ordinal.plural_rules_for/1","doc":"Return the plural rules for a locale.","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rules_for/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Ordinal.plural_rules_for/1","doc":"* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\nThe rules are returned in AST form after parsing.","ref":"Plausible.Cldr.Number.Ordinal.html#plural_rules_for/1-arguments"},{"type":"function","title":"Plausible.Cldr.Number.Ordinal.pluralize/3","doc":"Pluralize a number using ordinal plural rules\nand a substitution map.","ref":"Plausible.Cldr.Number.Ordinal.html#pluralize/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Ordinal.pluralize/3","doc":"* `number` is an integer, float or Decimal or a `Range.t{}`. When a range, The\n is that in any usage, the start value is strictly less than the end value,\n and that no values are negative. Results for any cases that do not meet\n these criteria are undefined.\n\n* `locale` is any locale returned by `Plausible.Cldr.Locale.new!/1` or any\n `locale_name` returned by `Plausible.Cldr.known_locale_names/0`\n\n* `substitutions` is a map that maps plural keys to a string.\n The valid substitution keys are `:zero`, `:one`, `:two`,\n `:few`, `:many` and `:other`.\n\nSee also `Plausible.Cldr.Number.Ordinal.Ordinal.plural_rule/3`.","ref":"Plausible.Cldr.Number.Ordinal.html#pluralize/3-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Ordinal.pluralize/3","doc":"iex> Plausible.Cldr.Number.Ordinal.pluralize 1, :en, %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 2, :en, %{one: \"one\", two: \"two\"}\n \"two\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 22, :en, %{one: \"one\", two: \"two\", other: \"other\"}\n \"two\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(1), :en, %{one: \"one\"}\n \"one\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: \"one\"}\n nil\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize Decimal.new(2), :en, %{one: \"one\", two: \"two\"}\n \"two\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, \"ar\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"other\"\n\n iex> Plausible.Cldr.Number.Ordinal.pluralize 1..10, \"en\", %{one: \"one\", few: \"few\", other: \"other\"}\n \"other\"","ref":"Plausible.Cldr.Number.Ordinal.html#pluralize/3-examples"},{"type":"module","title":"Plausible.Cldr.Number.PluralRule.Range","doc":"Implements plural rules for ranges","ref":"Plausible.Cldr.Number.PluralRule.Range.html"},{"type":"function","title":"Plausible.Cldr.Number.PluralRule.Range.plural_rule/3","doc":"Returns a final plural type for a start-of-range plural\ntype, an end-of-range plural type and a locale.","ref":"Plausible.Cldr.Number.PluralRule.Range.html#plural_rule/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.PluralRule.Range.plural_rule/3","doc":"* `first` is a plural type for the start of a range\n\n* `last` is a plural type for the end of a range\n\n* `locale` is any `Cldr.LanguageTag.t` or a language name\n (not locale name)","ref":"Plausible.Cldr.Number.PluralRule.Range.html#plural_rule/3-arguments"},{"type":"function","title":"Example - Plausible.Cldr.Number.PluralRule.Range.plural_rule/3","doc":"iex> Plausible.Cldr.Number.PluralRule.Range.plural_rule :other, :few, \"ar\"\n :few","ref":"Plausible.Cldr.Number.PluralRule.Range.html#plural_rule/3-example"},{"type":"module","title":"Plausible.Cldr.Number.Symbol","doc":"","ref":"Plausible.Cldr.Number.Symbol.html"},{"type":"function","title":"Plausible.Cldr.Number.Symbol.all_decimal_symbols/0","doc":"Returns a list of all decimal symbols defined\nby the locales configured in this backend as\na list.","ref":"Plausible.Cldr.Number.Symbol.html#all_decimal_symbols/0"},{"type":"function","title":"Plausible.Cldr.Number.Symbol.all_decimal_symbols_class/0","doc":"Returns a list of all decimal symbols defined\nby the locales configured in this backend as\na string.\n\nThis string can be used as a character class\nwhen builing a regular expression.","ref":"Plausible.Cldr.Number.Symbol.html#all_decimal_symbols_class/0"},{"type":"function","title":"Plausible.Cldr.Number.Symbol.all_grouping_symbols/0","doc":"Returns a list of all grouping symbols defined\nby the locales configured in this backend as\na list.","ref":"Plausible.Cldr.Number.Symbol.html#all_grouping_symbols/0"},{"type":"function","title":"Plausible.Cldr.Number.Symbol.all_grouping_symbols_class/0","doc":"Returns a list of all grouping symbols defined\nby the locales configured in this backend as\na string.\n\nThis string can be used as a character class\nwhen builing a regular expression.","ref":"Plausible.Cldr.Number.Symbol.html#all_grouping_symbols_class/0"},{"type":"function","title":"Plausible.Cldr.Number.Symbol.number_symbols_for/1","doc":"Returns a map of `Cldr.Number.Symbol.t` structs of the number symbols for each\nof the number systems of a locale.","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/1"},{"type":"function","title":"Options - Plausible.Cldr.Number.Symbol.number_symbols_for/1","doc":"* `locale` is any valid locale name returned by\n `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by\n `Plausible.Cldr.Locale.new!/1`. The default\n is `Plausible.Cldr.get_locale/0`.","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/1-options"},{"type":"function","title":"Example: - Plausible.Cldr.Number.Symbol.number_symbols_for/1","doc":"iex> Plausible.Cldr.Number.Symbol.number_symbols_for(:th)\n {:ok, %{\n latn: %Cldr.Number.Symbol{\n decimal: \".\",\n exponential: \"E\",\n group: \",\",\n infinity: \"∞\",\n list: \";\",\n minus_sign: \"-\",\n nan: \"NaN\",\n per_mille: \"‰\",\n percent_sign: \"%\",\n plus_sign: \"+\",\n superscripting_exponent: \"×\",\n time_separator: \":\"\n },\n thai: %Cldr.Number.Symbol{\n decimal: \".\",\n exponential: \"E\",\n group: \",\",\n infinity: \"∞\",\n list: \";\",\n minus_sign: \"-\",\n nan: \"NaN\",\n per_mille: \"‰\",\n percent_sign: \"%\",\n plus_sign: \"+\",\n superscripting_exponent: \"×\",\n time_separator: \":\"\n }\n }}","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/1-example"},{"type":"function","title":"Plausible.Cldr.Number.Symbol.number_symbols_for/2","doc":"","ref":"Plausible.Cldr.Number.Symbol.html#number_symbols_for/2"},{"type":"module","title":"Plausible.Cldr.Number.System","doc":"","ref":"Plausible.Cldr.Number.System.html"},{"type":"function","title":"Plausible.Cldr.Number.System.number_system_for/2","doc":"Returns the actual number system from a number system type.\n\n* `locale` is any valid locale name returned by `Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by ``Cldr.Locale.new!/2``\n\n* `system_name` is any number system name returned by\n `Cldr.known_number_systems/0` or a number system type\n returned by `Cldr.known_number_system_types/0`\n\nThis function will decode a number system type into the actual\nnumber system. If the number system provided can't be decoded\nit is returned as is.","ref":"Plausible.Cldr.Number.System.html#number_system_for/2"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.number_system_for/2","doc":"iex> Plausible.Cldr.Number.System.number_system_for \"th\", :latn\n {:ok, %{digits: \"0123456789\", type: :numeric}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"en\", :default\n {:ok, %{digits: \"0123456789\", type: :numeric}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"he\", :traditional\n {:ok, %{rules: \"hebrew\", type: :algorithmic}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"en\", :native\n {:ok, %{digits: \"0123456789\", type: :numeric}}\n\n iex> Plausible.Cldr.Number.System.number_system_for \"en\", :finance\n {\n :error,\n {Cldr.UnknownNumberSystemError,\n \"The number system :finance is unknown for the locale named :en. Valid number systems are %{default: :latn, native: :latn}\"}\n }","ref":"Plausible.Cldr.Number.System.html#number_system_for/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.System.number_system_from_locale/1","doc":"Returns the number system from a language tag or\nlocale name.","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.System.number_system_from_locale/1","doc":"* `locale` is any language tag returned be `Cldr.Locale.new/2`\n or a locale name in the list returned by `Cldr.known_locale_names/1`","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1-arguments"},{"type":"function","title":"Returns - Plausible.Cldr.Number.System.number_system_from_locale/1","doc":"* A number system name as an atom","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1-returns"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.number_system_from_locale/1","doc":"iex> Plausible.Cldr.Number.System.number_system_from_locale \"en-US-u-nu-thai\"\n :thai\n\n iex> Plausible.Cldr.Number.System.number_system_from_locale \"en-US\"\n :latn","ref":"Plausible.Cldr.Number.System.html#number_system_from_locale/1-examples"},{"type":"function","title":"Plausible.Cldr.Number.System.number_system_names_for/1","doc":"Returns the number systems available for a locale\nor `{:error, message}` if the locale is not known.\n\n* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`","ref":"Plausible.Cldr.Number.System.html#number_system_names_for/1"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.number_system_names_for/1","doc":"iex> Plausible.Cldr.Number.System.number_system_names_for \"en\"\n {:ok, [:latn]}\n\n iex> Plausible.Cldr.Number.System.number_system_names_for \"zz\"\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zz\\\" is invalid\"}}","ref":"Plausible.Cldr.Number.System.html#number_system_names_for/1-examples"},{"type":"function","title":"Plausible.Cldr.Number.System.number_system_names_for!/1","doc":"","ref":"Plausible.Cldr.Number.System.html#number_system_names_for!/1"},{"type":"function","title":"Plausible.Cldr.Number.System.number_system_types_for/1","doc":"","ref":"Plausible.Cldr.Number.System.html#number_system_types_for/1"},{"type":"function","title":"Plausible.Cldr.Number.System.number_systems_for/1","doc":"Returns the number systems available for a locale\nor `{:error, message}` if the locale is not known.\n\n* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`","ref":"Plausible.Cldr.Number.System.html#number_systems_for/1"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.number_systems_for/1","doc":"iex> Plausible.Cldr.Number.System.number_systems_for \"en\"\n {:ok, %{default: :latn, native: :latn}}\n\n iex> Plausible.Cldr.Number.System.number_systems_for \"th\"\n {:ok, %{default: :latn, native: :thai}}\n\n iex> Plausible.Cldr.Number.System.number_systems_for \"zz\"\n {:error, {Cldr.InvalidLanguageError, \"The language \\\"zz\\\" is invalid\"}}","ref":"Plausible.Cldr.Number.System.html#number_systems_for/1-examples"},{"type":"function","title":"Plausible.Cldr.Number.System.number_systems_for!/1","doc":"","ref":"Plausible.Cldr.Number.System.html#number_systems_for!/1"},{"type":"function","title":"Plausible.Cldr.Number.System.number_systems_like/2","doc":"","ref":"Plausible.Cldr.Number.System.html#number_systems_like/2"},{"type":"function","title":"Plausible.Cldr.Number.System.system_name_from/2","doc":"Returns a number system name for a given locale and number system reference.\n\n* `system_name` is any number system name returned by\n `Plausible.Cldr.known_number_systems/0` or a number system type\n returned by `Plausible.Cldr.known_number_system_types/0`\n\n* `locale` is any valid locale name returned by `Plausible.Cldr.known_locale_names/0`\n or a `Cldr.LanguageTag` struct returned by `Plausible.Cldr.Locale.new!/1`\n\nNumber systems can be references in one of two ways:\n\n* As a number system type such as :default, :native, :traditional and\n :finance. This allows references to a number system for a locale in a\n consistent fashion for a given use\n\n* WIth the number system name directly, such as :latn, :arab or any of the\n other 70 or so\n\nThis function dereferences the supplied `system_name` and returns the\nactual system name.","ref":"Plausible.Cldr.Number.System.html#system_name_from/2"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.system_name_from/2","doc":"ex> Plausible.Cldr.Number.System.system_name_from(:default, \"en\")\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.System.system_name_from(\"latn\", \"en\")\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.System.system_name_from(:native, \"en\")\n {:ok, :latn}\n\n iex> Plausible.Cldr.Number.System.system_name_from(:nope, \"en\")\n {\n :error,\n {Cldr.UnknownNumberSystemError, \"The number system :nope is unknown\"}\n }\n\nNote that return value is not guaranteed to be a valid\nnumber system for the given locale as demonstrated in the third example.","ref":"Plausible.Cldr.Number.System.html#system_name_from/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.System.to_system/2","doc":"Converts a number into the representation of\na non-latin number system.\n\nThis function converts numbers to a known\nnumber system only, it does not provide number\nformatting.\n\n* `number` is a `float`, `integer` or `Decimal`\n\n* `system_name` is any number system name returned by\n `Cldr.known_number_systems/0` or a number system type\n returned by `Cldr.known_number_system_types/0`\n\nThere are two types of number systems in CLDR:\n\n* `:numeric` in which the number system defines\n a direct mapping between the latin digits `0..9`\n into a the number system equivalent. In this case,\n` to_system/2` invokes `Cldr.Number.Transliterate.transliterate_digits/3`\n for the given number.\n\n* `:algorithmic` in which the number system\n does not have the same structure as the `:latn`\n number system and therefore the conversion is\n done algorithmically. For CLDR the algorithm\n is implemented through `Cldr.Rbnf` rulesets.\n These rulesets are considered by CLDR to be\n less rigorous than the `:numeric` number systems\n and caution and testing for a specific use case\n is recommended.","ref":"Plausible.Cldr.Number.System.html#to_system/2"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.to_system/2","doc":"iex> Plausible.Cldr.Number.System.to_system 123456, :hebr\n {:ok, \"קכ״ג׳תנ״ו\"}\n\n iex> Plausible.Cldr.Number.System.to_system 123, :hans\n {:ok, \"一百二十三\"}\n\n iex> Plausible.Cldr.Number.System.to_system 123, :hant\n {:ok, \"一百二十三\"}\n\n iex> Plausible.Cldr.Number.System.to_system 123, :hansfin\n {:ok, \"壹佰贰拾叁\"}","ref":"Plausible.Cldr.Number.System.html#to_system/2-examples"},{"type":"function","title":"Plausible.Cldr.Number.System.to_system!/2","doc":"Converts a number into the representation of\na non-latin number system. Returns a converted\nstring or raises on error.\n\n* `number` is a `float`, `integer` or `Decimal`\n\n* `system_name` is any number system name returned by\n `Plausible.Cldr.known_number_systems/0` or a number system type\n returned by `Plausible.Cldr.known_number_system_types/0`\n\nSee `Plausible.Cldr.Number.System.to_system/2` for further\ninformation.","ref":"Plausible.Cldr.Number.System.html#to_system!/2"},{"type":"function","title":"Examples - Plausible.Cldr.Number.System.to_system!/2","doc":"iex> Plausible.Cldr.Number.System.to_system! 123, :hans\n \"一百二十三\"\n\n iex> Plausible.Cldr.Number.System.to_system! 123, :hant\n \"一百二十三\"\n\n iex> Plausible.Cldr.Number.System.to_system! 123, :hansfin\n \"壹佰贰拾叁\"","ref":"Plausible.Cldr.Number.System.html#to_system!/2-examples"},{"type":"module","title":"Plausible.Cldr.Number.Transliterate","doc":"Transliteration for digits and separators.\n\nTransliterating a string is an expensive business. First the string has to\nbe exploded into its component graphemes. Then for each grapheme we have\nto map to the equivalent in the other `{locale, number_system}`. Then we\nhave to reassemble the string.\n\nEffort is made to short circuit where possible. Transliteration is not\nrequired for any `{locale, number_system}` that is the same as `{\"en\",\n\"latn\"}` since the implementation uses this combination for the placeholders during\nformatting already. When short circuiting is possible (typically the en-*\nlocales with \"latn\" number_system - the total number of short circuited\nlocales is 211 of the 537 in CLDR) the overall number formatting is twice as\nfast than when formal transliteration is required.\n\n#","ref":"Plausible.Cldr.Number.Transliterate.html"},{"type":"module","title":"Configuring precompilation of digit transliterations - Plausible.Cldr.Number.Transliterate","doc":"This module includes `Cldr.Number.Transliterate.transliterate_digits/3` which transliterates\ndigits between number systems. For example from :arabic to :latn. Since generating a\ntransliteration map is slow, pairs of transliterations can be configured so that the\ntransliteration map is created at compile time and therefore speeding up transliteration at\nrun time.\n\nTo configure these transliteration pairs, add the to the `use Cldr` configuration\nin a backend module:\n\n defmodule MyApp.Cldr do\n use Cldr,\n locale: [\"en\", \"fr\", \"th\"],\n default_locale: \"en\",\n precompile_transliterations: [{:latn, :thai}, {:arab, :thai}]\n end\n\nWhere each tuple in the list configures one transliteration map. In this example, two maps are\nconfigured: from `:latn` to `:thai` and from `:arab` to `:thai`.\n\nA list of configurable number systems is returned by `Cldr.Number.System.numeric_systems/0`.\n\nIf a transliteration is requested between two number pairs that have not been configured for\nprecompilation, a warning is logged.","ref":"Plausible.Cldr.Number.Transliterate.html#module-configuring-precompilation-of-digit-transliterations"},{"type":"function","title":"Plausible.Cldr.Number.Transliterate.transliterate/3","doc":"Transliterates from latin digits to another number system's digits.\n\nTransliterates the latin digits 0..9 to their equivalents in\nanother number system. Also transliterates the decimal and grouping\nseparators as well as the plus, minus and exponent symbols. Any other character\nin the string will be returned \"as is\".","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate/3"},{"type":"function","title":"Arguments - Plausible.Cldr.Number.Transliterate.transliterate/3","doc":"* `sequence` is the string to be transliterated.\n\n* `locale` is any known locale, defaulting to `Plausible.Cldr.get_locale/0`.\n\n* `number_system` is any known number system. If expressed as a `string` it\n is the actual name of a known number system. If epressed as an `atom` it is\n used as a key to look up a number system for the locale (the usual keys are\n `:default` and `:native` but :traditional and :finance are also part of the\n standard). See `Plausible.Cldr.Number.System.number_systems_for/1` for a locale to\n see what number system types are defined. The default is `:default`.\n\nFor available number systems see `Cldr.Number.System.number_systems/0`\nand `Plausible.Cldr.Number.System.number_systems_for/1`. Also see\n`Plausible.Cldr.Number.Symbol.number_symbols_for/1`.","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate/3-arguments"},{"type":"function","title":"Examples - Plausible.Cldr.Number.Transliterate.transliterate/3","doc":"iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\")\n \"123556\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123,556.000\", \"fr\", :default)\n \"123 556,000\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\", \"th\", :default)\n \"123556\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\", \"th\", \"thai\")\n \"๑๒๓๕๕๖\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"123556\", \"th\", :native)\n \"๑๒๓๕๕๖\"\n\n iex> Plausible.Cldr.Number.Transliterate.transliterate(\"Some number is: 123556\", \"th\", \"thai\")\n \"Some number is: ๑๒๓๕๕๖\"","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate/3-examples"},{"type":"function","title":"Plausible.Cldr.Number.Transliterate.transliterate!/3","doc":"","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate!/3"},{"type":"function","title":"Plausible.Cldr.Number.Transliterate.transliterate_digits/3","doc":"Transliterates digits from one number system to another number system\n\n* `digits` is binary representation of a number\n\n* `from_system` and `to_system` are number system names in atom form. See\n`Cldr.Number.System.numeric_systems/0` for available number systems.","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate_digits/3"},{"type":"function","title":"Example - Plausible.Cldr.Number.Transliterate.transliterate_digits/3","doc":"iex> Plausible.Cldr.Number.Transliterate.transliterate_digits \"٠١٢٣٤٥٦٧٨٩\", :arab, :latn\n \"0123456789\"","ref":"Plausible.Cldr.Number.Transliterate.html#transliterate_digits/3-example"},{"type":"module","title":"Plausible.Cldr.Rbnf.NumberSystem","doc":"Functions to implement the number system rule-based-number-format rules of CLDR.\n\nThese rules are defined only on the \"und\" locale and represent specialised\nnumber formatting.\n\nThe standard public API for RBNF is via the `Cldr.Number.to_string/2` function.\n\nThe functions on this module are defined at compile time based upon the RBNF rules\ndefined in the Unicode CLDR data repository. Available rules are identified by:\n\n iex> Plausible.Cldr.Rbnf.NumberSystem.rule_sets(:und)\n ...> |> Enum.sort()\n [\n :armenian_lower,\n :armenian_upper,\n :cyrillic_lower,\n :ethiopic,\n :georgian,\n :greek_lower,\n :greek_upper,\n :hebrew,\n :hebrew_item,\n :roman_lower,\n :roman_upper,\n :tamil,\n :zz_default\n ]\n\nA rule can then be invoked on an available rule_set. For example\n\n iex> Plausible.Cldr.Rbnf.NumberSystem.roman_upper(123, :und)\n \"CXXIII\"\n\nThis particular call is equivalent to the call through the public API of:\n\n iex> Plausible.Cldr.Number.to_string(123, format: :roman)\n {:ok, \"CXXIII\"}","ref":"Plausible.Cldr.Rbnf.NumberSystem.html"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.all_rule_sets/0","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#all_rule_sets/0"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_lower/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_lower/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_lower/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_lower/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_upper/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_upper/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.armenian_upper/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#armenian_upper/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_1_10/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_1_10/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_final/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_final/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_post/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_post/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.cyrillic_lower_thousands/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#cyrillic_lower_thousands/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p1/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p1/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p2/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p2/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p3/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p3/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.ethiopic_p/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#ethiopic_p/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.georgian/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#georgian/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.georgian/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#georgian/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_lower/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_lower/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_lower/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_lower/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_numeral_majuscules/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_numeral_majuscules/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_numeral_minuscules/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_numeral_minuscules/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_upper/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_upper/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.greek_upper/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#greek_upper/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_0_99/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_0_99/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_item/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_item/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_item/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_item/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_item_hundreds/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_item_hundreds/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.hebrew_thousands/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#hebrew_thousands/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_lower/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_lower/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_lower/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_lower/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_upper/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_upper/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.roman_upper/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#roman_upper/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.rule_sets/0","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#rule_sets/0"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.rule_sets/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#rule_sets/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.tamil/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#tamil/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.tamil/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#tamil/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.tamil_thousands/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#tamil_thousands/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.zz_default/1","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#zz_default/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.NumberSystem.zz_default/2","doc":"","ref":"Plausible.Cldr.Rbnf.NumberSystem.html#zz_default/2"},{"type":"module","title":"Plausible.Cldr.Rbnf.Ordinal","doc":"Functions to implement the ordinal rule-based-number-format rules of CLDR.\n\nAs CLDR notes, the data is incomplete or non-existent for many languages. It\nis considered complete for English however.\n\nThe standard public API for RBNF is via the `Cldr.Number.to_string/2` function.\n\nThe functions on this module are defined at compile time based upon the RBNF rules\ndefined in the Unicode CLDR data repository. Available rules are identified by:\n\n iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets(:en)\n [:digits_ordinal]\n\n iex> Plausible.Cldr.Rbnf.Ordinal.rule_sets(\"fr\")\n ...> |> Enum.sort()\n [\n :digits_ordinal,\n :digits_ordinal_feminine,\n :digits_ordinal_feminine_plural,\n :digits_ordinal_masculine,\n :digits_ordinal_masculine_plural\n ]\n\nA rule can then be invoked on an available rule_set. For example\n\n iex> Plausible.Cldr.Rbnf.Ordinal.digits_ordinal(123, :en)\n \"123rd\"\n\nThis call is equivalent to the call through the public API of:\n\n iex> Plausible.Cldr.Number.to_string(123, format: :ordinal)\n {:ok, \"123rd\"}","ref":"Plausible.Cldr.Rbnf.Ordinal.html"},{"type":"function","title":"Plausible.Cldr.Rbnf.Ordinal.all_rule_sets/0","doc":"","ref":"Plausible.Cldr.Rbnf.Ordinal.html#all_rule_sets/0"},{"type":"function","title":"Plausible.Cldr.Rbnf.Ordinal.digits_ordinal/2","doc":"","ref":"Plausible.Cldr.Rbnf.Ordinal.html#digits_ordinal/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Ordinal.rule_sets/0","doc":"","ref":"Plausible.Cldr.Rbnf.Ordinal.html#rule_sets/0"},{"type":"function","title":"Plausible.Cldr.Rbnf.Ordinal.rule_sets/1","doc":"","ref":"Plausible.Cldr.Rbnf.Ordinal.html#rule_sets/1"},{"type":"module","title":"Plausible.Cldr.Rbnf.Spellout","doc":"Functions to implement the spellout rule-based-number-format rules of CLDR.\n\nAs CLDR notes, the data is incomplete or non-existent for many languages. It\nis considered complete for English however.\n\nThe standard public API for RBNF is via the `Cldr.Number.to_string/2` function.\n\nThe functions on this module are defined at compile time based upon the RBNF rules\ndefined in the Unicode CLDR data repository. Available rules are identified by:\n\n iex> Plausible.Cldr.Rbnf.Spellout.rule_sets(\"en\")\n ...> |> Enum.sort()\n [\n :spellout_cardinal,\n :spellout_cardinal_verbose,\n :spellout_numbering,\n :spellout_numbering_verbose,\n :spellout_numbering_year,\n :spellout_ordinal,\n :spellout_ordinal_verbose\n ]\n\nA rule can then be invoked on an available rule_set. For example:\n\n iex> Plausible.Cldr.Rbnf.Spellout.spellout_ordinal(123, \"en\")\n \"one hundred twenty-third\"\n\nThis call is equivalent to the call through the public API of:\n\n iex> Plausible.Cldr.Number.to_string(123, format: :spellout)\n {:ok, \"one hundred twenty-three\"}","ref":"Plausible.Cldr.Rbnf.Spellout.html"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.all_rule_sets/0","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#all_rule_sets/0"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.and/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#and/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.and_o/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#and_o/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.commas/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#commas/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.commas_o/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#commas_o/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.r2d_year/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#r2d_year/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.rule_sets/0","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#rule_sets/0"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.rule_sets/1","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#rule_sets/1"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_cardinal/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_cardinal/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_cardinal_verbose/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_cardinal_verbose/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_numbering/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_numbering/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_numbering_verbose/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_numbering_verbose/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_numbering_year/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_numbering_year/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_ordinal/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_ordinal/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.spellout_ordinal_verbose/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#spellout_ordinal_verbose/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.th/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#th/2"},{"type":"function","title":"Plausible.Cldr.Rbnf.Spellout.tieth/2","doc":"","ref":"Plausible.Cldr.Rbnf.Spellout.html#tieth/2"},{"type":"module","title":"Plausible.ClickhouseEventV2","doc":"Event schema for when NumericIDs migration is complete","ref":"Plausible.ClickhouseEventV2.html"},{"type":"function","title":"Plausible.ClickhouseEventV2.merge_session/2","doc":"","ref":"Plausible.ClickhouseEventV2.html#merge_session/2"},{"type":"function","title":"Plausible.ClickhouseEventV2.new/1","doc":"","ref":"Plausible.ClickhouseEventV2.html#new/1"},{"type":"module","title":"Plausible.ClickhouseLocationData","doc":"Schema for storing location id <-> translation mappings in ClickHouse\n\nIndirectly read via dictionary `location_data_dictionary` in ALIAS columns in\n`events_v2`, `sessions_v2` and `imported_locations` table.","ref":"Plausible.ClickhouseLocationData.html"},{"type":"module","title":"Plausible.ClickhouseRepo","doc":"","ref":"Plausible.ClickhouseRepo.html"},{"type":"function","title":"Plausible.ClickhouseRepo.aggregate/3","doc":"","ref":"Plausible.ClickhouseRepo.html#aggregate/3"},{"type":"function","title":"Plausible.ClickhouseRepo.aggregate/4","doc":"","ref":"Plausible.ClickhouseRepo.html#aggregate/4"},{"type":"function","title":"Plausible.ClickhouseRepo.all/2","doc":"","ref":"Plausible.ClickhouseRepo.html#all/2"},{"type":"function","title":"Plausible.ClickhouseRepo.alter_update_all/3","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","ref":"Plausible.ClickhouseRepo.html#alter_update_all/3"},{"type":"function","title":"Plausible.ClickhouseRepo.checked_out?/0","doc":"","ref":"Plausible.ClickhouseRepo.html#checked_out?/0"},{"type":"function","title":"Plausible.ClickhouseRepo.checkout/2","doc":"","ref":"Plausible.ClickhouseRepo.html#checkout/2"},{"type":"function","title":"Plausible.ClickhouseRepo.child_spec/1","doc":"","ref":"Plausible.ClickhouseRepo.html#child_spec/1"},{"type":"function","title":"Plausible.ClickhouseRepo.config/0","doc":"","ref":"Plausible.ClickhouseRepo.html#config/0"},{"type":"function","title":"Plausible.ClickhouseRepo.default_options/1","doc":"","ref":"Plausible.ClickhouseRepo.html#default_options/1"},{"type":"function","title":"Plausible.ClickhouseRepo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","ref":"Plausible.ClickhouseRepo.html#disconnect_all/2"},{"type":"function","title":"Plausible.ClickhouseRepo.exists?/2","doc":"","ref":"Plausible.ClickhouseRepo.html#exists?/2"},{"type":"function","title":"Plausible.ClickhouseRepo.get/3","doc":"","ref":"Plausible.ClickhouseRepo.html#get/3"},{"type":"function","title":"Plausible.ClickhouseRepo.get!/3","doc":"","ref":"Plausible.ClickhouseRepo.html#get!/3"},{"type":"function","title":"Plausible.ClickhouseRepo.get_by/3","doc":"","ref":"Plausible.ClickhouseRepo.html#get_by/3"},{"type":"function","title":"Plausible.ClickhouseRepo.get_by!/3","doc":"","ref":"Plausible.ClickhouseRepo.html#get_by!/3"},{"type":"function","title":"Plausible.ClickhouseRepo.get_dynamic_repo/0","doc":"","ref":"Plausible.ClickhouseRepo.html#get_dynamic_repo/0"},{"type":"function","title":"Plausible.ClickhouseRepo.insert_stream/3","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","ref":"Plausible.ClickhouseRepo.html#insert_stream/3"},{"type":"function","title":"Plausible.ClickhouseRepo.load/2","doc":"","ref":"Plausible.ClickhouseRepo.html#load/2"},{"type":"function","title":"Plausible.ClickhouseRepo.one/2","doc":"","ref":"Plausible.ClickhouseRepo.html#one/2"},{"type":"function","title":"Plausible.ClickhouseRepo.one!/2","doc":"","ref":"Plausible.ClickhouseRepo.html#one!/2"},{"type":"function","title":"Plausible.ClickhouseRepo.parallel_tasks/2","doc":"","ref":"Plausible.ClickhouseRepo.html#parallel_tasks/2"},{"type":"function","title":"Plausible.ClickhouseRepo.preload/3","doc":"","ref":"Plausible.ClickhouseRepo.html#preload/3"},{"type":"function","title":"Plausible.ClickhouseRepo.put_dynamic_repo/1","doc":"","ref":"Plausible.ClickhouseRepo.html#put_dynamic_repo/1"},{"type":"function","title":"Plausible.ClickhouseRepo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","ref":"Plausible.ClickhouseRepo.html#query/3"},{"type":"function","title":"Plausible.ClickhouseRepo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","ref":"Plausible.ClickhouseRepo.html#query!/3"},{"type":"function","title":"Plausible.ClickhouseRepo.reload/2","doc":"","ref":"Plausible.ClickhouseRepo.html#reload/2"},{"type":"function","title":"Plausible.ClickhouseRepo.reload!/2","doc":"","ref":"Plausible.ClickhouseRepo.html#reload!/2"},{"type":"function","title":"Plausible.ClickhouseRepo.start_link/1","doc":"","ref":"Plausible.ClickhouseRepo.html#start_link/1"},{"type":"function","title":"Plausible.ClickhouseRepo.stop/1","doc":"","ref":"Plausible.ClickhouseRepo.html#stop/1"},{"type":"function","title":"Plausible.ClickhouseRepo.stream/2","doc":"","ref":"Plausible.ClickhouseRepo.html#stream/2"},{"type":"function","title":"Plausible.ClickhouseRepo.to_inline_sql/2","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","ref":"Plausible.ClickhouseRepo.html#to_inline_sql/2"},{"type":"function","title":"Plausible.ClickhouseRepo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","ref":"Plausible.ClickhouseRepo.html#to_sql/2"},{"type":"module","title":"Plausible.ClickhouseSessionV2","doc":"Session schema for when NumericIDs migration is complete","ref":"Plausible.ClickhouseSessionV2.html"},{"type":"function","title":"Plausible.ClickhouseSessionV2.random_uint64/0","doc":"","ref":"Plausible.ClickhouseSessionV2.html#random_uint64/0"},{"type":"module","title":"Plausible.ClickhouseSessionV2.BoolUInt8","doc":"Custom type to cast Bool as UInt8","ref":"Plausible.ClickhouseSessionV2.BoolUInt8.html"},{"type":"function","title":"Plausible.ClickhouseSessionV2.BoolUInt8.embed_as/1","doc":"","ref":"Plausible.ClickhouseSessionV2.BoolUInt8.html#embed_as/1"},{"type":"function","title":"Plausible.ClickhouseSessionV2.BoolUInt8.equal?/2","doc":"","ref":"Plausible.ClickhouseSessionV2.BoolUInt8.html#equal?/2"},{"type":"module","title":"Plausible.ConfigHelpers","doc":"","ref":"Plausible.ConfigHelpers.html"},{"type":"function","title":"Plausible.ConfigHelpers.get_int_from_path_or_env/3","doc":"","ref":"Plausible.ConfigHelpers.html#get_int_from_path_or_env/3"},{"type":"function","title":"Plausible.ConfigHelpers.get_var_from_path_or_env/3","doc":"","ref":"Plausible.ConfigHelpers.html#get_var_from_path_or_env/3"},{"type":"module","title":"Plausible.CrmExtensions","doc":"Extensions for Kaffy CRM","ref":"Plausible.CrmExtensions.html"},{"type":"function","title":"Plausible.CrmExtensions.javascripts/1","doc":"","ref":"Plausible.CrmExtensions.html#javascripts/1"},{"type":"module","title":"Plausible.DataCase","doc":"This module defines the setup for tests requiring\naccess to the application's data layer.\n\nYou may define functions here to be used as helpers in\nyour tests.\n\nFinally, if the test case interacts with the database,\nit cannot be async. For this reason, every test runs\ninside a transaction which is reset at the beginning\nof the test unless the test case is marked as async.","ref":"Plausible.DataCase.html"},{"type":"module","title":"Plausible.DataMigration","doc":"Base module for coordinated Clickhouse data migrations\nrun via remote shell or otherwise (TBD).","ref":"Plausible.DataMigration.html"},{"type":"module","title":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource","doc":"Clean up referrer_source entries for demo site with\n`Direct / None` for value populated by dogfooding\nPlausible stats.","ref":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource.html"},{"type":"function","title":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource.run/1","doc":"","ref":"Plausible.DataMigration.CleanUpDemoSiteReferrerSource.html#run/1"},{"type":"module","title":"Plausible.DataMigration.LocationsSync","doc":"ClickHouse locations data migration for storing location names in ClickHouse.\n\nOnly run when `Location.version()` changes: either as a migration or in cron.\n\nThe migration:\n1. Truncates existing `location_data` table (if exists)\n2. Creates new table (if needed)\n3. Inserts new data from Location module\n4. (Re-)Creates dictionary to read location data from table\n5. Creates ALIAS columns in `events_v2`, `sessions_v2` and `imported_locations` table to make reading location names easy\n6. Updates table comment for `location_data` to indicate last version synced.\n\nNote that the dictionary is large enough to cache the whole dataset in memory, making lookups fast.\n\nThis migration is intended to be idempotent and rerunnable - if run multiple times, it should always set things to the same\nresult as if run once.\n\nSQL files available at: priv/data_migrations/LocationsSync/sql","ref":"Plausible.DataMigration.LocationsSync.html"},{"type":"function","title":"Plausible.DataMigration.LocationsSync.confirm/3","doc":"","ref":"Plausible.DataMigration.LocationsSync.html#confirm/3"},{"type":"function","title":"Plausible.DataMigration.LocationsSync.out_of_date?/0","doc":"","ref":"Plausible.DataMigration.LocationsSync.html#out_of_date?/0"},{"type":"function","title":"Plausible.DataMigration.LocationsSync.run/0","doc":"","ref":"Plausible.DataMigration.LocationsSync.html#run/0"},{"type":"function","title":"Plausible.DataMigration.LocationsSync.run_sql/3","doc":"","ref":"Plausible.DataMigration.LocationsSync.html#run_sql/3"},{"type":"function","title":"Plausible.DataMigration.LocationsSync.run_sql_confirm/2","doc":"","ref":"Plausible.DataMigration.LocationsSync.html#run_sql_confirm/2"},{"type":"module","title":"Plausible.DataMigration.NumericIDs","doc":"Numeric IDs migration, SQL files available at:\npriv/data_migrations/NumericIDs/sql","ref":"Plausible.DataMigration.NumericIDs.html"},{"type":"function","title":"Plausible.DataMigration.NumericIDs.confirm/3","doc":"","ref":"Plausible.DataMigration.NumericIDs.html#confirm/3"},{"type":"function","title":"Plausible.DataMigration.NumericIDs.run/1","doc":"","ref":"Plausible.DataMigration.NumericIDs.html#run/1"},{"type":"function","title":"Plausible.DataMigration.NumericIDs.run_sql/3","doc":"","ref":"Plausible.DataMigration.NumericIDs.html#run_sql/3"},{"type":"function","title":"Plausible.DataMigration.NumericIDs.run_sql_confirm/2","doc":"","ref":"Plausible.DataMigration.NumericIDs.html#run_sql_confirm/2"},{"type":"module","title":"Plausible.DataMigration.PopulateEventSessionColumns","doc":"Populates event session columns with data from sessions table.\n\nRun via: ./bin/plausible rpc \"Plausible.DataMigration.PopulateEventSessionColumns.run\"\nKill via: ./bin/plausible rpc \"Plausible.DataMigration.PopulateEventSessionColumns.kill\"\nMonitor via ./bin/plausible rpc \"Plausible.DataMigration.PopulateEventSessionColumns.report_progress\"\n\nSuggested to run in a screen/tmux session to be able to easily monitor\n\nSQL files available at: priv/data_migrations/PopulateEventSessionColumns/sql","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.confirm/3","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#confirm/3"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.kill/1","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#kill/1"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.report_progress/1","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#report_progress/1"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.run/1","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#run/1"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.run_sql/3","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#run_sql/3"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.run_sql_confirm/2","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#run_sql_confirm/2"},{"type":"function","title":"Plausible.DataMigration.PopulateEventSessionColumns.wait_until_mutations_complete/1","doc":"","ref":"Plausible.DataMigration.PopulateEventSessionColumns.html#wait_until_mutations_complete/1"},{"type":"module","title":"Plausible.DataMigration.Repo","doc":"Ecto.Repo for Clickhouse data migrations, to be started manually,\noutside of the main application supervision tree.","ref":"Plausible.DataMigration.Repo.html"},{"type":"function","title":"Plausible.DataMigration.Repo.aggregate/3","doc":"","ref":"Plausible.DataMigration.Repo.html#aggregate/3"},{"type":"function","title":"Plausible.DataMigration.Repo.aggregate/4","doc":"","ref":"Plausible.DataMigration.Repo.html#aggregate/4"},{"type":"function","title":"Plausible.DataMigration.Repo.all/2","doc":"","ref":"Plausible.DataMigration.Repo.html#all/2"},{"type":"function","title":"Plausible.DataMigration.Repo.alter_update_all/3","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","ref":"Plausible.DataMigration.Repo.html#alter_update_all/3"},{"type":"function","title":"Plausible.DataMigration.Repo.checked_out?/0","doc":"","ref":"Plausible.DataMigration.Repo.html#checked_out?/0"},{"type":"function","title":"Plausible.DataMigration.Repo.checkout/2","doc":"","ref":"Plausible.DataMigration.Repo.html#checkout/2"},{"type":"function","title":"Plausible.DataMigration.Repo.child_spec/1","doc":"","ref":"Plausible.DataMigration.Repo.html#child_spec/1"},{"type":"function","title":"Plausible.DataMigration.Repo.config/0","doc":"","ref":"Plausible.DataMigration.Repo.html#config/0"},{"type":"function","title":"Plausible.DataMigration.Repo.default_options/1","doc":"","ref":"Plausible.DataMigration.Repo.html#default_options/1"},{"type":"function","title":"Plausible.DataMigration.Repo.delete/2","doc":"","ref":"Plausible.DataMigration.Repo.html#delete/2"},{"type":"function","title":"Plausible.DataMigration.Repo.delete!/2","doc":"","ref":"Plausible.DataMigration.Repo.html#delete!/2"},{"type":"function","title":"Plausible.DataMigration.Repo.delete_all/2","doc":"","ref":"Plausible.DataMigration.Repo.html#delete_all/2"},{"type":"function","title":"Plausible.DataMigration.Repo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","ref":"Plausible.DataMigration.Repo.html#disconnect_all/2"},{"type":"function","title":"Plausible.DataMigration.Repo.exists?/2","doc":"","ref":"Plausible.DataMigration.Repo.html#exists?/2"},{"type":"function","title":"Plausible.DataMigration.Repo.get/3","doc":"","ref":"Plausible.DataMigration.Repo.html#get/3"},{"type":"function","title":"Plausible.DataMigration.Repo.get!/3","doc":"","ref":"Plausible.DataMigration.Repo.html#get!/3"},{"type":"function","title":"Plausible.DataMigration.Repo.get_by/3","doc":"","ref":"Plausible.DataMigration.Repo.html#get_by/3"},{"type":"function","title":"Plausible.DataMigration.Repo.get_by!/3","doc":"","ref":"Plausible.DataMigration.Repo.html#get_by!/3"},{"type":"function","title":"Plausible.DataMigration.Repo.get_dynamic_repo/0","doc":"","ref":"Plausible.DataMigration.Repo.html#get_dynamic_repo/0"},{"type":"function","title":"Plausible.DataMigration.Repo.insert/2","doc":"","ref":"Plausible.DataMigration.Repo.html#insert/2"},{"type":"function","title":"Plausible.DataMigration.Repo.insert!/2","doc":"","ref":"Plausible.DataMigration.Repo.html#insert!/2"},{"type":"function","title":"Plausible.DataMigration.Repo.insert_all/3","doc":"","ref":"Plausible.DataMigration.Repo.html#insert_all/3"},{"type":"function","title":"Plausible.DataMigration.Repo.insert_or_update/2","doc":"","ref":"Plausible.DataMigration.Repo.html#insert_or_update/2"},{"type":"function","title":"Plausible.DataMigration.Repo.insert_or_update!/2","doc":"","ref":"Plausible.DataMigration.Repo.html#insert_or_update!/2"},{"type":"function","title":"Plausible.DataMigration.Repo.insert_stream/3","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","ref":"Plausible.DataMigration.Repo.html#insert_stream/3"},{"type":"function","title":"Plausible.DataMigration.Repo.load/2","doc":"","ref":"Plausible.DataMigration.Repo.html#load/2"},{"type":"function","title":"Plausible.DataMigration.Repo.one/2","doc":"","ref":"Plausible.DataMigration.Repo.html#one/2"},{"type":"function","title":"Plausible.DataMigration.Repo.one!/2","doc":"","ref":"Plausible.DataMigration.Repo.html#one!/2"},{"type":"function","title":"Plausible.DataMigration.Repo.preload/3","doc":"","ref":"Plausible.DataMigration.Repo.html#preload/3"},{"type":"function","title":"Plausible.DataMigration.Repo.prepare_query/3","doc":"","ref":"Plausible.DataMigration.Repo.html#prepare_query/3"},{"type":"function","title":"Plausible.DataMigration.Repo.put_dynamic_repo/1","doc":"","ref":"Plausible.DataMigration.Repo.html#put_dynamic_repo/1"},{"type":"function","title":"Plausible.DataMigration.Repo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","ref":"Plausible.DataMigration.Repo.html#query/3"},{"type":"function","title":"Plausible.DataMigration.Repo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","ref":"Plausible.DataMigration.Repo.html#query!/3"},{"type":"function","title":"Plausible.DataMigration.Repo.reload/2","doc":"","ref":"Plausible.DataMigration.Repo.html#reload/2"},{"type":"function","title":"Plausible.DataMigration.Repo.reload!/2","doc":"","ref":"Plausible.DataMigration.Repo.html#reload!/2"},{"type":"function","title":"Plausible.DataMigration.Repo.start/2","doc":"","ref":"Plausible.DataMigration.Repo.html#start/2"},{"type":"function","title":"Plausible.DataMigration.Repo.start_link/1","doc":"","ref":"Plausible.DataMigration.Repo.html#start_link/1"},{"type":"function","title":"Plausible.DataMigration.Repo.stop/1","doc":"","ref":"Plausible.DataMigration.Repo.html#stop/1"},{"type":"function","title":"Plausible.DataMigration.Repo.stream/2","doc":"","ref":"Plausible.DataMigration.Repo.html#stream/2"},{"type":"function","title":"Plausible.DataMigration.Repo.to_inline_sql/2","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","ref":"Plausible.DataMigration.Repo.html#to_inline_sql/2"},{"type":"function","title":"Plausible.DataMigration.Repo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","ref":"Plausible.DataMigration.Repo.html#to_sql/2"},{"type":"function","title":"Plausible.DataMigration.Repo.update/2","doc":"","ref":"Plausible.DataMigration.Repo.html#update/2"},{"type":"function","title":"Plausible.DataMigration.Repo.update!/2","doc":"","ref":"Plausible.DataMigration.Repo.html#update!/2"},{"type":"function","title":"Plausible.DataMigration.Repo.update_all/3","doc":"","ref":"Plausible.DataMigration.Repo.html#update_all/3"},{"type":"module","title":"Plausible.DataMigration.SiteImports","doc":"!!!WARNING!!!: This script is used in migrations. Please take special care\nwhen altering it.\n\nSite imports migration backfilling SiteImport entries for old imports\nand alters import end dates to match actual end date of respective import stats.","ref":"Plausible.DataMigration.SiteImports.html"},{"type":"function","title":"Plausible.DataMigration.SiteImports.run/1","doc":"","ref":"Plausible.DataMigration.SiteImports.html#run/1"},{"type":"module","title":"Plausible.DataMigration.VersionedSessions","doc":"!!!WARNING!!!: This script is used in migrations. Please take special care\nwhen altering it.\n\nSessions CollapsingMergeTree -> VersionedCollapsingMergeTree migration,\nSQL files available at:\n\npriv/data_migrations/VersionedSessions/sql","ref":"Plausible.DataMigration.VersionedSessions.html"},{"type":"function","title":"Plausible.DataMigration.VersionedSessions.confirm/3","doc":"","ref":"Plausible.DataMigration.VersionedSessions.html#confirm/3"},{"type":"function","title":"Plausible.DataMigration.VersionedSessions.run/1","doc":"","ref":"Plausible.DataMigration.VersionedSessions.html#run/1"},{"type":"function","title":"Plausible.DataMigration.VersionedSessions.run_sql/3","doc":"","ref":"Plausible.DataMigration.VersionedSessions.html#run_sql/3"},{"type":"function","title":"Plausible.DataMigration.VersionedSessions.run_sql_confirm/2","doc":"","ref":"Plausible.DataMigration.VersionedSessions.html#run_sql_confirm/2"},{"type":"module","title":"Plausible.DebugReplayInfo","doc":"Function execution context (with arguments) to Sentry reports.","ref":"Plausible.DebugReplayInfo.html"},{"type":"function","title":"Plausible.DebugReplayInfo.deserialize/1","doc":"","ref":"Plausible.DebugReplayInfo.html#deserialize/1"},{"type":"macro","title":"Plausible.DebugReplayInfo.include_sentry_replay_info/0","doc":"","ref":"Plausible.DebugReplayInfo.html#include_sentry_replay_info/0"},{"type":"module","title":"Plausible.Ecto.EventName","doc":"Custom type for event name. Accepts Strings and Integers and stores them as String. Returns\n cast error if any other type is provided. Accepting integers is important for 404 tracking.","ref":"Plausible.Ecto.EventName.html"},{"type":"function","title":"Plausible.Ecto.EventName.cast/1","doc":"","ref":"Plausible.Ecto.EventName.html#cast/1"},{"type":"function","title":"Plausible.Ecto.EventName.dump/1","doc":"","ref":"Plausible.Ecto.EventName.html#dump/1"},{"type":"function","title":"Plausible.Ecto.EventName.embed_as/1","doc":"","ref":"Plausible.Ecto.EventName.html#embed_as/1"},{"type":"function","title":"Plausible.Ecto.EventName.equal?/2","doc":"","ref":"Plausible.Ecto.EventName.html#equal?/2"},{"type":"function","title":"Plausible.Ecto.EventName.load/1","doc":"","ref":"Plausible.Ecto.EventName.html#load/1"},{"type":"function","title":"Plausible.Ecto.EventName.type/0","doc":"","ref":"Plausible.Ecto.EventName.html#type/0"},{"type":"module","title":"Plausible.Ecto.Types.CompiledRegex","doc":"Ensures that the regex is compiled on load","ref":"Plausible.Ecto.Types.CompiledRegex.html"},{"type":"function","title":"Plausible.Ecto.Types.CompiledRegex.cast/1","doc":"","ref":"Plausible.Ecto.Types.CompiledRegex.html#cast/1"},{"type":"function","title":"Plausible.Ecto.Types.CompiledRegex.dump/1","doc":"","ref":"Plausible.Ecto.Types.CompiledRegex.html#dump/1"},{"type":"function","title":"Plausible.Ecto.Types.CompiledRegex.embed_as/1","doc":"","ref":"Plausible.Ecto.Types.CompiledRegex.html#embed_as/1"},{"type":"function","title":"Plausible.Ecto.Types.CompiledRegex.equal?/2","doc":"","ref":"Plausible.Ecto.Types.CompiledRegex.html#equal?/2"},{"type":"function","title":"Plausible.Ecto.Types.CompiledRegex.load/1","doc":"","ref":"Plausible.Ecto.Types.CompiledRegex.html#load/1"},{"type":"function","title":"Plausible.Ecto.Types.CompiledRegex.type/0","doc":"","ref":"Plausible.Ecto.Types.CompiledRegex.html#type/0"},{"type":"module","title":"Plausible.Exports","doc":"Contains functions to export data for events and sessions as Zip archives.","ref":"Plausible.Exports.html"},{"type":"function","title":"Plausible.Exports.archive_filename/2","doc":"Renders export archive filename.\n\nExamples:\n\n iex> archive_filename(\"plausible.io\", _created_on = ~D[2024-12-31])\n \"plausible_io_20241231.zip\"","ref":"Plausible.Exports.html#archive_filename/2"},{"type":"function","title":"Plausible.Exports.content_disposition/1","doc":"Safely renders content disposition for an arbitrary export filename.\n\nExamples:\n\n iex> content_disposition(\"plausible_io_20241231.zip\")\n \"attachment; filename=\\\"plausible_io_20241231.zip\\\"\"\n\n iex> content_disposition(\"📊.zip\")\n \"attachment; filename=\\\"plausible-export.zip\\\"; filename*=utf-8''%F0%9F%93%8A.zip\"","ref":"Plausible.Exports.html#content_disposition/1"},{"type":"function","title":"Plausible.Exports.date_range/2","doc":"Returns the date range for the site's events data in site's timezone or `nil` if there is no data","ref":"Plausible.Exports.html#date_range/2"},{"type":"function","title":"Plausible.Exports.delete_local_export/1","doc":"Deletes local export for a site","ref":"Plausible.Exports.html#delete_local_export/1"},{"type":"function","title":"Plausible.Exports.delete_s3_export!/1","doc":"Deletes S3 export for a site. Raises if object storage is unavailable.","ref":"Plausible.Exports.html#delete_s3_export!/1"},{"type":"function","title":"Plausible.Exports.export_queries/2","doc":"Builds Ecto queries to export data from `events_v2` and `sessions_v2`\ntables into the format of `imported_*` tables for a website.","ref":"Plausible.Exports.html#export_queries/2"},{"type":"function","title":"Plausible.Exports.get_last_export_job/1","doc":"Gets last CSV export job for a site","ref":"Plausible.Exports.html#get_last_export_job/1"},{"type":"function","title":"Plausible.Exports.get_local_export/3","doc":"Gets local export for a site","ref":"Plausible.Exports.html#get_local_export/3"},{"type":"function","title":"Plausible.Exports.get_s3_export!/2","doc":"Gets S3 export for a site. Raises if object storage is unavailable.","ref":"Plausible.Exports.html#get_s3_export!/2"},{"type":"function","title":"Plausible.Exports.oban_listen/0","doc":"Subscribes to CSV export job notifications","ref":"Plausible.Exports.html#oban_listen/0"},{"type":"function","title":"Plausible.Exports.schedule_local_export/2","doc":"Schedules CSV export job to local storage","ref":"Plausible.Exports.html#schedule_local_export/2"},{"type":"function","title":"Plausible.Exports.schedule_s3_export/2","doc":"Schedules CSV export job to S3 storage","ref":"Plausible.Exports.html#schedule_s3_export/2"},{"type":"function","title":"Plausible.Exports.stream_archive/3","doc":"Creates a streamable Zip archive from the provided (named) Ecto queries.\n\nExample usage:\n\n {:ok, pool} = Ch.start_link(pool_size: 1)\n\n DBConnection.run(pool, fn conn ->\n conn\n |> stream_archive(export_queries(_site_id = 1), format: \"CSVWithNames\")\n |> Stream.into(File.stream!(\"export.zip\"))\n |> Stream.run()\n end)","ref":"Plausible.Exports.html#stream_archive/3"},{"type":"type","title":"Plausible.Exports.export/0","doc":"","ref":"Plausible.Exports.html#t:export/0"},{"type":"module","title":"Plausible.Factory","doc":"","ref":"Plausible.Factory.html"},{"type":"function","title":"Plausible.Factory.api_key_factory/0","doc":"","ref":"Plausible.Factory.html#api_key_factory/0"},{"type":"function","title":"Plausible.Factory.build/2","doc":"","ref":"Plausible.Factory.html#build/2"},{"type":"function","title":"Plausible.Factory.build_list/3","doc":"","ref":"Plausible.Factory.html#build_list/3"},{"type":"function","title":"Plausible.Factory.build_pair/2","doc":"","ref":"Plausible.Factory.html#build_pair/2"},{"type":"function","title":"Plausible.Factory.business_subscription_factory/0","doc":"","ref":"Plausible.Factory.html#business_subscription_factory/0"},{"type":"function","title":"Plausible.Factory.ch_session_factory/0","doc":"","ref":"Plausible.Factory.html#ch_session_factory/0"},{"type":"function","title":"Plausible.Factory.country_rule_factory/0","doc":"","ref":"Plausible.Factory.html#country_rule_factory/0"},{"type":"function","title":"Plausible.Factory.create/1","doc":"","ref":"Plausible.Factory.html#create/1"},{"type":"function","title":"Plausible.Factory.create/2","doc":"","ref":"Plausible.Factory.html#create/2"},{"type":"function","title":"Plausible.Factory.create_list/3","doc":"","ref":"Plausible.Factory.html#create_list/3"},{"type":"function","title":"Plausible.Factory.create_pair/2","doc":"","ref":"Plausible.Factory.html#create_pair/2"},{"type":"function","title":"Plausible.Factory.drop_notification_factory/0","doc":"","ref":"Plausible.Factory.html#drop_notification_factory/0"},{"type":"function","title":"Plausible.Factory.enterprise_plan_factory/0","doc":"","ref":"Plausible.Factory.html#enterprise_plan_factory/0"},{"type":"function","title":"Plausible.Factory.event_factory/0","doc":"","ref":"Plausible.Factory.html#event_factory/0"},{"type":"function","title":"Plausible.Factory.factory/1","doc":"Raises a helpful error if no factory is defined.","ref":"Plausible.Factory.html#factory/1"},{"type":"function","title":"Plausible.Factory.goal_factory/1","doc":"","ref":"Plausible.Factory.html#goal_factory/1"},{"type":"function","title":"Plausible.Factory.google_auth_factory/0","doc":"","ref":"Plausible.Factory.html#google_auth_factory/0"},{"type":"function","title":"Plausible.Factory.growth_subscription_factory/0","doc":"","ref":"Plausible.Factory.html#growth_subscription_factory/0"},{"type":"function","title":"Plausible.Factory.imported_browsers_factory/0","doc":"","ref":"Plausible.Factory.html#imported_browsers_factory/0"},{"type":"function","title":"Plausible.Factory.imported_custom_events_factory/0","doc":"","ref":"Plausible.Factory.html#imported_custom_events_factory/0"},{"type":"function","title":"Plausible.Factory.imported_devices_factory/0","doc":"","ref":"Plausible.Factory.html#imported_devices_factory/0"},{"type":"function","title":"Plausible.Factory.imported_entry_pages_factory/0","doc":"","ref":"Plausible.Factory.html#imported_entry_pages_factory/0"},{"type":"function","title":"Plausible.Factory.imported_exit_pages_factory/0","doc":"","ref":"Plausible.Factory.html#imported_exit_pages_factory/0"},{"type":"function","title":"Plausible.Factory.imported_locations_factory/0","doc":"","ref":"Plausible.Factory.html#imported_locations_factory/0"},{"type":"function","title":"Plausible.Factory.imported_operating_systems_factory/0","doc":"","ref":"Plausible.Factory.html#imported_operating_systems_factory/0"},{"type":"function","title":"Plausible.Factory.imported_pages_factory/0","doc":"","ref":"Plausible.Factory.html#imported_pages_factory/0"},{"type":"function","title":"Plausible.Factory.imported_sources_factory/0","doc":"","ref":"Plausible.Factory.html#imported_sources_factory/0"},{"type":"function","title":"Plausible.Factory.imported_visitors_factory/0","doc":"","ref":"Plausible.Factory.html#imported_visitors_factory/0"},{"type":"function","title":"Plausible.Factory.insert/1","doc":"","ref":"Plausible.Factory.html#insert/1"},{"type":"function","title":"Plausible.Factory.insert/2","doc":"","ref":"Plausible.Factory.html#insert/2"},{"type":"function","title":"Plausible.Factory.insert/3","doc":"","ref":"Plausible.Factory.html#insert/3"},{"type":"function","title":"Plausible.Factory.insert_list/3","doc":"","ref":"Plausible.Factory.html#insert_list/3"},{"type":"function","title":"Plausible.Factory.insert_list/4","doc":"","ref":"Plausible.Factory.html#insert_list/4"},{"type":"function","title":"Plausible.Factory.insert_pair/2","doc":"","ref":"Plausible.Factory.html#insert_pair/2"},{"type":"function","title":"Plausible.Factory.insert_pair/3","doc":"","ref":"Plausible.Factory.html#insert_pair/3"},{"type":"function","title":"Plausible.Factory.invitation_factory/0","doc":"","ref":"Plausible.Factory.html#invitation_factory/0"},{"type":"function","title":"Plausible.Factory.ip_rule_factory/0","doc":"","ref":"Plausible.Factory.html#ip_rule_factory/0"},{"type":"function","title":"Plausible.Factory.monthly_report_factory/0","doc":"","ref":"Plausible.Factory.html#monthly_report_factory/0"},{"type":"function","title":"Plausible.Factory.pageview_factory/0","doc":"","ref":"Plausible.Factory.html#pageview_factory/0"},{"type":"function","title":"Plausible.Factory.params_for/2","doc":"","ref":"Plausible.Factory.html#params_for/2"},{"type":"function","title":"Plausible.Factory.params_with_assocs/2","doc":"","ref":"Plausible.Factory.html#params_with_assocs/2"},{"type":"function","title":"Plausible.Factory.shared_link_factory/0","doc":"","ref":"Plausible.Factory.html#shared_link_factory/0"},{"type":"function","title":"Plausible.Factory.site_factory/1","doc":"","ref":"Plausible.Factory.html#site_factory/1"},{"type":"function","title":"Plausible.Factory.site_import_factory/0","doc":"","ref":"Plausible.Factory.html#site_import_factory/0"},{"type":"function","title":"Plausible.Factory.site_membership_factory/0","doc":"","ref":"Plausible.Factory.html#site_membership_factory/0"},{"type":"function","title":"Plausible.Factory.spike_notification_factory/0","doc":"","ref":"Plausible.Factory.html#spike_notification_factory/0"},{"type":"function","title":"Plausible.Factory.string_params_for/2","doc":"","ref":"Plausible.Factory.html#string_params_for/2"},{"type":"function","title":"Plausible.Factory.string_params_with_assocs/2","doc":"","ref":"Plausible.Factory.html#string_params_with_assocs/2"},{"type":"function","title":"Plausible.Factory.subscription_factory/0","doc":"","ref":"Plausible.Factory.html#subscription_factory/0"},{"type":"function","title":"Plausible.Factory.user_factory/1","doc":"","ref":"Plausible.Factory.html#user_factory/1"},{"type":"function","title":"Plausible.Factory.weekly_report_factory/0","doc":"","ref":"Plausible.Factory.html#weekly_report_factory/0"},{"type":"module","title":"Plausible.Funnel","doc":"A funnel is a marketing term used to capture and describe the journey\nthat users go through, from initial step to conversion.\nA funnel consists of several steps (here: 2..8).\n\nThis module defines the database schema for storing funnels\nand changeset helpers for enumerating the steps within.\n\nEach step references a goal (either a Custom Event or Visit)\n- see: `Plausible.Goal`.","ref":"Plausible.Funnel.html"},{"type":"function","title":"Plausible.Funnel.changeset/2","doc":"","ref":"Plausible.Funnel.html#changeset/2"},{"type":"macro","title":"Plausible.Funnel.max_steps/0","doc":"","ref":"Plausible.Funnel.html#max_steps/0"},{"type":"macro","title":"Plausible.Funnel.min_steps/0","doc":"","ref":"Plausible.Funnel.html#min_steps/0"},{"type":"function","title":"Plausible.Funnel.put_steps/2","doc":"","ref":"Plausible.Funnel.html#put_steps/2"},{"type":"type","title":"Plausible.Funnel.t/0","doc":"","ref":"Plausible.Funnel.html#t:t/0"},{"type":"module","title":"Plausible.Funnel.Const","doc":"Compile-time convenience constants for funnel characteristics.","ref":"Plausible.Funnel.Const.html"},{"type":"macro","title":"Plausible.Funnel.Const.max_steps/0","doc":"","ref":"Plausible.Funnel.Const.html#max_steps/0"},{"type":"macro","title":"Plausible.Funnel.Const.min_steps/0","doc":"","ref":"Plausible.Funnel.Const.html#min_steps/0"},{"type":"module","title":"Plausible.Funnel.Step","doc":"This module defines the database schema for a single Funnel step.\nSee: `Plausible.Funnel` for more information.","ref":"Plausible.Funnel.Step.html"},{"type":"function","title":"Plausible.Funnel.Step.changeset/2","doc":"","ref":"Plausible.Funnel.Step.html#changeset/2"},{"type":"type","title":"Plausible.Funnel.Step.t/0","doc":"","ref":"Plausible.Funnel.Step.html#t:t/0"},{"type":"module","title":"Plausible.Funnels","doc":"This module implements contextual Funnel interface, allowing listing,\ncreating and deleting funnel definitions.\n\nFor brief explanation of what a Funnel is, please see `Plausible.Funnel` schema.\nSee `Plausible.Stats.Funnel` for the evaluation logic.","ref":"Plausible.Funnels.html"},{"type":"function","title":"Plausible.Funnels.create/3","doc":"","ref":"Plausible.Funnels.html#create/3"},{"type":"function","title":"Plausible.Funnels.create_changeset/3","doc":"","ref":"Plausible.Funnels.html#create_changeset/3"},{"type":"function","title":"Plausible.Funnels.delete/2","doc":"","ref":"Plausible.Funnels.html#delete/2"},{"type":"function","title":"Plausible.Funnels.edit_changeset/3","doc":"","ref":"Plausible.Funnels.html#edit_changeset/3"},{"type":"function","title":"Plausible.Funnels.ephemeral_definition/3","doc":"","ref":"Plausible.Funnels.html#ephemeral_definition/3"},{"type":"function","title":"Plausible.Funnels.get/2","doc":"","ref":"Plausible.Funnels.html#get/2"},{"type":"function","title":"Plausible.Funnels.list/1","doc":"","ref":"Plausible.Funnels.html#list/1"},{"type":"function","title":"Plausible.Funnels.update/3","doc":"","ref":"Plausible.Funnels.html#update/3"},{"type":"function","title":"Plausible.Funnels.with_goals_query/1","doc":"","ref":"Plausible.Funnels.html#with_goals_query/1"},{"type":"module","title":"Plausible.Geo","doc":"This module provides an API for fetching IP geolocation.","ref":"Plausible.Geo.html"},{"type":"function","title":"Plausible.Geo.await_loader/0","doc":"Waits for the database to start after calling `load_db/1` with the async option.","ref":"Plausible.Geo.html#await_loader/0"},{"type":"function","title":"Plausible.Geo.database_type/0","doc":"Returns geodatabase type.\n\nUsed for deciding whether to show the DB-IP disclaimer or not.","ref":"Plausible.Geo.html#database_type/0"},{"type":"function","title":"Examples - Plausible.Geo.database_type/0","doc":"In the case of a DB-IP database:\n\n iex> database_type()\n \"DBIP-City-Lite\"\n\n In the case of a MaxMind database:\n\n iex> database_type()\n \"GeoLite2-City\"","ref":"Plausible.Geo.html#database_type/0-examples"},{"type":"function","title":"Plausible.Geo.load_db/1","doc":"Starts the geodatabase loading process. Two modes are supported: local file\nand MaxMind license key.","ref":"Plausible.Geo.html#load_db/1"},{"type":"function","title":"Options - Plausible.Geo.load_db/1","doc":"* `:path` - the path to the .mmdb database local file. When present,\n `:license_key` and `:edition` are not required.\n\n * `:license_key` - the [license key](https://support.maxmind.com/hc/en-us/articles/4407111582235-Generate-a-License-Key)\n from MaxMind to authenticate requests to MaxMind.\n\n * `:edition` - the name of the MaxMind database to be downloaded from MaxMind\n servers. Defaults to `GeoLite2-City`.\n\n * `:cache_dir` - if set, the downloaded .mmdb files are cached there across\n restarts.\n\n * `:async` - when used, configures the database loading to run\n asynchronously.","ref":"Plausible.Geo.html#load_db/1-options"},{"type":"function","title":"Examples - Plausible.Geo.load_db/1","doc":"Loading from a local file:\n\n iex> load_db(path: \"/etc/plausible/dbip-city.mmdb\")\n :ok\n\n Downloading a MaxMind DB (this license key is no longer active):\n\n iex> load_db(license_key: \"LNpsJCCKPis6XvBP\", edition: \"GeoLite2-City\", async: true)\n :ok","ref":"Plausible.Geo.html#load_db/1-examples"},{"type":"function","title":"Plausible.Geo.lookup/1","doc":"Looks up geo info about an IP address.","ref":"Plausible.Geo.html#lookup/1"},{"type":"function","title":"Examples - Plausible.Geo.lookup/1","doc":"iex> lookup(\"8.7.6.5\")\n %{\n \"city\" => %{\n \"geoname_id\" => 5349755,\n \"names\" => %{\n \"de\" => \"Fontana\",\n \"en\" => \"Fontana\",\n \"ja\" => \"フォンタナ\",\n \"ru\" => \"Фонтана\"\n }\n },\n \"continent\" => %{\n \"code\" => \"NA\",\n \"geoname_id\" => 6255149,\n \"names\" => %{\n \"de\" => \"Nordamerika\",\n \"en\" => \"North America\",\n \"es\" => \"Norteamérica\",\n \"fr\" => \"Amérique du Nord\",\n \"ja\" => \"北アメリカ\",\n \"pt-BR\" => \"América do Norte\",\n \"ru\" => \"Северная Америка\",\n \"zh-CN\" => \"北美洲\"\n }\n },\n \"country\" => %{\n \"geoname_id\" => 6252001,\n \"iso_code\" => \"US\",\n \"names\" => %{\n \"de\" => \"Vereinigte Staaten\",\n \"en\" => \"United States\",\n \"es\" => \"Estados Unidos\",\n \"fr\" => \"États Unis\",\n \"ja\" => \"アメリカ\",\n \"pt-BR\" => \"EUA\",\n \"ru\" => \"США\",\n \"zh-CN\" => \"美国\"\n }\n },\n \"location\" => %{\n \"accuracy_radius\" => 50,\n \"latitude\" => 34.1211,\n \"longitude\" => -117.4362,\n \"metro_code\" => 803,\n \"time_zone\" => \"America/Los_Angeles\"\n },\n \"postal\" => %{\"code\" => \"92336\"},\n \"registered_country\" => %{\n \"geoname_id\" => 6252001,\n \"iso_code\" => \"US\",\n \"names\" => %{\n \"de\" => \"Vereinigte Staaten\",\n \"en\" => \"United States\",\n \"es\" => \"Estados Unidos\",\n \"fr\" => \"États Unis\",\n \"ja\" => \"アメリカ\",\n \"pt-BR\" => \"EUA\",\n \"ru\" => \"США\",\n \"zh-CN\" => \"美国\"\n }\n },\n \"subdivisions\" => [\n %{\n \"geoname_id\" => 5332921,\n \"iso_code\" => \"CA\",\n \"names\" => %{\n \"de\" => \"Kalifornien\",\n \"en\" => \"California\",\n \"es\" => \"California\",\n \"fr\" => \"Californie\",\n \"ja\" => \"カリフォルニア州\",\n \"pt-BR\" => \"Califórnia\",\n \"ru\" => \"Калифорния\",\n \"zh-CN\" => \"加州\"\n }\n }\n ]\n }","ref":"Plausible.Geo.html#lookup/1-examples"},{"type":"module","title":"Plausible.Goal","doc":"","ref":"Plausible.Goal.html"},{"type":"function","title":"Plausible.Goal.changeset/2","doc":"","ref":"Plausible.Goal.html#changeset/2"},{"type":"function","title":"Plausible.Goal.display_name/1","doc":"","ref":"Plausible.Goal.html#display_name/1"},{"type":"function","title":"Plausible.Goal.max_event_name_length/0","doc":"","ref":"Plausible.Goal.html#max_event_name_length/0"},{"type":"function","title":"Plausible.Goal.type/1","doc":"","ref":"Plausible.Goal.html#type/1"},{"type":"type","title":"Plausible.Goal.t/0","doc":"","ref":"Plausible.Goal.html#t:t/0"},{"type":"module","title":"Plausible.Goal.Revenue","doc":"Currency specific functions for revenue goals","ref":"Plausible.Goal.Revenue.html"},{"type":"function","title":"Plausible.Goal.Revenue.currency_option/1","doc":"","ref":"Plausible.Goal.Revenue.html#currency_option/1"},{"type":"function","title":"Plausible.Goal.Revenue.currency_options/0","doc":"","ref":"Plausible.Goal.Revenue.html#currency_options/0"},{"type":"function","title":"Plausible.Goal.Revenue.display/1","doc":"","ref":"Plausible.Goal.Revenue.html#display/1"},{"type":"function","title":"Plausible.Goal.Revenue.revenue?/1","doc":"","ref":"Plausible.Goal.Revenue.html#revenue?/1"},{"type":"function","title":"Plausible.Goal.Revenue.valid_currencies/0","doc":"","ref":"Plausible.Goal.Revenue.html#valid_currencies/0"},{"type":"module","title":"Plausible.Goals","doc":"","ref":"Plausible.Goals.html"},{"type":"function","title":"Plausible.Goals.batch_create_event_goals/2","doc":"","ref":"Plausible.Goals.html#batch_create_event_goals/2"},{"type":"function","title":"Plausible.Goals.count/1","doc":"","ref":"Plausible.Goals.html#count/1"},{"type":"function","title":"Plausible.Goals.create/3","doc":"Creates a Goal for a site.\n\nIf the created goal is a revenue goal, it sets site.updated_at to be\nrefreshed by the sites cache, as revenue goals are used during ingestion.","ref":"Plausible.Goals.html#create/3"},{"type":"function","title":"Plausible.Goals.create_404/1","doc":"","ref":"Plausible.Goals.html#create_404/1"},{"type":"function","title":"Plausible.Goals.create_file_downloads/1","doc":"","ref":"Plausible.Goals.html#create_file_downloads/1"},{"type":"function","title":"Plausible.Goals.create_outbound_links/1","doc":"","ref":"Plausible.Goals.html#create_outbound_links/1"},{"type":"function","title":"Plausible.Goals.delete/2","doc":"If a goal belongs to funnel(s), we need to inspect their number of steps.\n\nIf it exceeds the minimum allowed (defined via `Plausible.Funnel.min_steps/0`),\nthe funnel will be reduced (i.e. a step associated with the goal to be deleted\nis removed), so that the minimum number of steps is preserved. This is done\nimplicitly, by postgres, as per on_delete: :delete_all.\n\nOtherwise, for associated funnel(s) consisting of minimum number steps only,\nfunnel record(s) are removed completely along with the targeted goal.","ref":"Plausible.Goals.html#delete/2"},{"type":"function","title":"Plausible.Goals.delete_404/1","doc":"","ref":"Plausible.Goals.html#delete_404/1"},{"type":"function","title":"Plausible.Goals.delete_file_downloads/1","doc":"","ref":"Plausible.Goals.html#delete_file_downloads/1"},{"type":"function","title":"Plausible.Goals.delete_outbound_links/1","doc":"","ref":"Plausible.Goals.html#delete_outbound_links/1"},{"type":"function","title":"Plausible.Goals.find_or_create/2","doc":"","ref":"Plausible.Goals.html#find_or_create/2"},{"type":"function","title":"Plausible.Goals.for_site/2","doc":"","ref":"Plausible.Goals.html#for_site/2"},{"type":"function","title":"Plausible.Goals.for_site_query/2","doc":"","ref":"Plausible.Goals.html#for_site_query/2"},{"type":"function","title":"Plausible.Goals.get/2","doc":"","ref":"Plausible.Goals.html#get/2"},{"type":"function","title":"Plausible.Goals.list_revenue_goals/1","doc":"","ref":"Plausible.Goals.html#list_revenue_goals/1"},{"type":"function","title":"Plausible.Goals.update/2","doc":"","ref":"Plausible.Goals.html#update/2"},{"type":"module","title":"Plausible.Google.API","doc":"API to Google services.","ref":"Plausible.Google.API.html"},{"type":"function","title":"Plausible.Google.API.fetch_access_token!/1","doc":"","ref":"Plausible.Google.API.html#fetch_access_token!/1"},{"type":"function","title":"Plausible.Google.API.fetch_stats/4","doc":"","ref":"Plausible.Google.API.html#fetch_stats/4"},{"type":"function","title":"Plausible.Google.API.fetch_verified_properties/1","doc":"","ref":"Plausible.Google.API.html#fetch_verified_properties/1"},{"type":"function","title":"Plausible.Google.API.get_analytics_end_date/2","doc":"","ref":"Plausible.Google.API.html#get_analytics_end_date/2"},{"type":"function","title":"Plausible.Google.API.get_analytics_start_date/2","doc":"","ref":"Plausible.Google.API.html#get_analytics_start_date/2"},{"type":"function","title":"Plausible.Google.API.get_property/2","doc":"","ref":"Plausible.Google.API.html#get_property/2"},{"type":"function","title":"Plausible.Google.API.import_authorize_url/1","doc":"","ref":"Plausible.Google.API.html#import_authorize_url/1"},{"type":"function","title":"Plausible.Google.API.list_properties/1","doc":"","ref":"Plausible.Google.API.html#list_properties/1"},{"type":"function","title":"Plausible.Google.API.maybe_refresh_token/1","doc":"","ref":"Plausible.Google.API.html#maybe_refresh_token/1"},{"type":"function","title":"Plausible.Google.API.property?/1","doc":"","ref":"Plausible.Google.API.html#property?/1"},{"type":"function","title":"Plausible.Google.API.search_console_authorize_url/1","doc":"","ref":"Plausible.Google.API.html#search_console_authorize_url/1"},{"type":"module","title":"Plausible.Google.API.Mock","doc":"Mock of API to Google services.","ref":"Plausible.Google.API.Mock.html"},{"type":"function","title":"Plausible.Google.API.Mock.fetch_stats/4","doc":"","ref":"Plausible.Google.API.Mock.html#fetch_stats/4"},{"type":"module","title":"Plausible.Google.GA4.API","doc":"API for Google Analytics 4.","ref":"Plausible.Google.GA4.API.html"},{"type":"function","title":"Plausible.Google.GA4.API.fetch_and_persist/2","doc":"","ref":"Plausible.Google.GA4.API.html#fetch_and_persist/2"},{"type":"function","title":"Plausible.Google.GA4.API.get_analytics_end_date/2","doc":"","ref":"Plausible.Google.GA4.API.html#get_analytics_end_date/2"},{"type":"function","title":"Plausible.Google.GA4.API.get_analytics_start_date/2","doc":"","ref":"Plausible.Google.GA4.API.html#get_analytics_start_date/2"},{"type":"function","title":"Plausible.Google.GA4.API.get_property/2","doc":"","ref":"Plausible.Google.GA4.API.html#get_property/2"},{"type":"function","title":"Plausible.Google.GA4.API.import_analytics/4","doc":"","ref":"Plausible.Google.GA4.API.html#import_analytics/4"},{"type":"function","title":"Plausible.Google.GA4.API.list_properties/1","doc":"","ref":"Plausible.Google.GA4.API.html#list_properties/1"},{"type":"type","title":"Plausible.Google.GA4.API.import_auth/0","doc":"","ref":"Plausible.Google.GA4.API.html#t:import_auth/0"},{"type":"module","title":"Plausible.Google.GA4.HTTP","doc":"HTTP client implementation for Google Analytics 4 API.","ref":"Plausible.Google.GA4.HTTP.html"},{"type":"function","title":"Plausible.Google.GA4.HTTP.get_analytics_end_date/2","doc":"","ref":"Plausible.Google.GA4.HTTP.html#get_analytics_end_date/2"},{"type":"function","title":"Plausible.Google.GA4.HTTP.get_analytics_start_date/2","doc":"","ref":"Plausible.Google.GA4.HTTP.html#get_analytics_start_date/2"},{"type":"function","title":"Plausible.Google.GA4.HTTP.get_property/2","doc":"","ref":"Plausible.Google.GA4.HTTP.html#get_property/2"},{"type":"function","title":"Plausible.Google.GA4.HTTP.get_report/1","doc":"","ref":"Plausible.Google.GA4.HTTP.html#get_report/1"},{"type":"function","title":"Plausible.Google.GA4.HTTP.list_accounts_for_user/1","doc":"","ref":"Plausible.Google.GA4.HTTP.html#list_accounts_for_user/1"},{"type":"module","title":"Plausible.Google.GA4.ReportRequest","doc":"Report request struct for Google Analytics 4 API","ref":"Plausible.Google.GA4.ReportRequest.html"},{"type":"function","title":"Plausible.Google.GA4.ReportRequest.full_report/0","doc":"","ref":"Plausible.Google.GA4.ReportRequest.html#full_report/0"},{"type":"type","title":"Plausible.Google.GA4.ReportRequest.t/0","doc":"","ref":"Plausible.Google.GA4.ReportRequest.html#t:t/0"},{"type":"module","title":"Plausible.Google.HTTP","doc":"","ref":"Plausible.Google.HTTP.html"},{"type":"function","title":"Plausible.Google.HTTP.fetch_access_token!/1","doc":"","ref":"Plausible.Google.HTTP.html#fetch_access_token!/1"},{"type":"function","title":"Plausible.Google.HTTP.list_sites/1","doc":"","ref":"Plausible.Google.HTTP.html#list_sites/1"},{"type":"function","title":"Plausible.Google.HTTP.list_stats/5","doc":"","ref":"Plausible.Google.HTTP.html#list_stats/5"},{"type":"function","title":"Plausible.Google.HTTP.refresh_auth_token/1","doc":"","ref":"Plausible.Google.HTTP.html#refresh_auth_token/1"},{"type":"module","title":"Plausible.HTTPClient","doc":"HTTP Client built on top of Finch.\n\nBy default, request parameters are json-encoded.\n\nIf a raw binary value is supplied, no encoding is performed.\nIf x-www-form-urlencoded content-type is set in headers,\nURL encoding is invoked.","ref":"Plausible.HTTPClient.html"},{"type":"function","title":"Plausible.HTTPClient.get/3","doc":"Make a GET request","ref":"Plausible.HTTPClient.html#get/3"},{"type":"function","title":"Plausible.HTTPClient.impl/0","doc":"","ref":"Plausible.HTTPClient.html#impl/0"},{"type":"function","title":"Plausible.HTTPClient.post/4","doc":"Make a POST request","ref":"Plausible.HTTPClient.html#post/4"},{"type":"behaviour","title":"Plausible.HTTPClient.Interface","doc":"","ref":"Plausible.HTTPClient.Interface.html"},{"type":"callback","title":"Plausible.HTTPClient.Interface.get/1","doc":"","ref":"Plausible.HTTPClient.Interface.html#c:get/1"},{"type":"callback","title":"Plausible.HTTPClient.Interface.get/2","doc":"","ref":"Plausible.HTTPClient.Interface.html#c:get/2"},{"type":"callback","title":"Plausible.HTTPClient.Interface.get/3","doc":"","ref":"Plausible.HTTPClient.Interface.html#c:get/3"},{"type":"callback","title":"Plausible.HTTPClient.Interface.post/3","doc":"","ref":"Plausible.HTTPClient.Interface.html#c:post/3"},{"type":"callback","title":"Plausible.HTTPClient.Interface.post/4","doc":"","ref":"Plausible.HTTPClient.Interface.html#c:post/4"},{"type":"type","title":"Plausible.HTTPClient.Interface.finch_request_opts/0","doc":"","ref":"Plausible.HTTPClient.Interface.html#t:finch_request_opts/0"},{"type":"type","title":"Plausible.HTTPClient.Interface.headers/0","doc":"","ref":"Plausible.HTTPClient.Interface.html#t:headers/0"},{"type":"type","title":"Plausible.HTTPClient.Interface.params/0","doc":"","ref":"Plausible.HTTPClient.Interface.html#t:params/0"},{"type":"type","title":"Plausible.HTTPClient.Interface.response/0","doc":"","ref":"Plausible.HTTPClient.Interface.html#t:response/0"},{"type":"type","title":"Plausible.HTTPClient.Interface.url/0","doc":"","ref":"Plausible.HTTPClient.Interface.html#t:url/0"},{"type":"module","title":"Plausible.HTTPClient.Non200Error","doc":"","ref":"Plausible.HTTPClient.Non200Error.html"},{"type":"function","title":"Plausible.HTTPClient.Non200Error.new/1","doc":"","ref":"Plausible.HTTPClient.Non200Error.html#new/1"},{"type":"type","title":"Plausible.HTTPClient.Non200Error.t/0","doc":"","ref":"Plausible.HTTPClient.Non200Error.html#t:t/0"},{"type":"module","title":"Plausible.HelpScout","doc":"HelpScout callback API logic.","ref":"Plausible.HelpScout.html"},{"type":"function","title":"Plausible.HelpScout.get_details_for_customer/1","doc":"","ref":"Plausible.HelpScout.html#get_details_for_customer/1"},{"type":"function","title":"Plausible.HelpScout.get_details_for_emails/2","doc":"","ref":"Plausible.HelpScout.html#get_details_for_emails/2"},{"type":"function","title":"Plausible.HelpScout.search_users/2","doc":"","ref":"Plausible.HelpScout.html#search_users/2"},{"type":"function","title":"Plausible.HelpScout.signature_errors/0","doc":"","ref":"Plausible.HelpScout.html#signature_errors/0"},{"type":"function","title":"Plausible.HelpScout.validate_signature/1","doc":"Validates signature against secret key configured for the\nHelpScout application.\n\nNOTE: HelpScout signature generation procedure at\nhttps://developer.helpscout.com/apps/guides/signature-validation/\nfails to mention that it's implicitly dependent on request params\norder getting preserved. PHP arrays are ordered maps, so they provide\nthis guarantee. Here, on the other hand, we have to determine the original\norder of the keys directly from the query string and serialize\nparams to JSON using wrapper struct, informing Jason to put the values\nin the serialized object in this particular order matching query string.","ref":"Plausible.HelpScout.html#validate_signature/1"},{"type":"type","title":"Plausible.HelpScout.signature_error/0","doc":"","ref":"Plausible.HelpScout.html#t:signature_error/0"},{"type":"module","title":"Plausible.HelpScout.Vault","doc":"Provides a vault that will be used to encrypt/decrypt the stored HelpScout API access tokens.","ref":"Plausible.HelpScout.Vault.html"},{"type":"function","title":"Plausible.HelpScout.Vault.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.HelpScout.Vault.html#child_spec/1"},{"type":"function","title":"Plausible.HelpScout.Vault.start_link/1","doc":"","ref":"Plausible.HelpScout.Vault.html#start_link/1"},{"type":"module","title":"Plausible.Helpers.JSON","doc":"Common helpers for JSON handling","ref":"Plausible.Helpers.JSON.html"},{"type":"function","title":"Plausible.Helpers.JSON.decode_or_fallback/1","doc":"","ref":"Plausible.Helpers.JSON.html#decode_or_fallback/1"},{"type":"module","title":"Plausible.ImportDeletionRepo","doc":"A dedicated repo for import related mutations","ref":"Plausible.ImportDeletionRepo.html"},{"type":"function","title":"Plausible.ImportDeletionRepo.aggregate/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#aggregate/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.aggregate/4","doc":"","ref":"Plausible.ImportDeletionRepo.html#aggregate/4"},{"type":"function","title":"Plausible.ImportDeletionRepo.all/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#all/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.alter_update_all/3","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","ref":"Plausible.ImportDeletionRepo.html#alter_update_all/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.checked_out?/0","doc":"","ref":"Plausible.ImportDeletionRepo.html#checked_out?/0"},{"type":"function","title":"Plausible.ImportDeletionRepo.checkout/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#checkout/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.child_spec/1","doc":"","ref":"Plausible.ImportDeletionRepo.html#child_spec/1"},{"type":"function","title":"Plausible.ImportDeletionRepo.config/0","doc":"","ref":"Plausible.ImportDeletionRepo.html#config/0"},{"type":"function","title":"Plausible.ImportDeletionRepo.default_options/1","doc":"","ref":"Plausible.ImportDeletionRepo.html#default_options/1"},{"type":"function","title":"Plausible.ImportDeletionRepo.delete/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#delete/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.delete!/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#delete!/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.delete_all/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#delete_all/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","ref":"Plausible.ImportDeletionRepo.html#disconnect_all/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.exists?/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#exists?/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.get/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#get/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.get!/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#get!/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.get_by/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#get_by/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.get_by!/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#get_by!/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.get_dynamic_repo/0","doc":"","ref":"Plausible.ImportDeletionRepo.html#get_dynamic_repo/0"},{"type":"function","title":"Plausible.ImportDeletionRepo.insert/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#insert/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.insert!/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#insert!/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.insert_all/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#insert_all/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.insert_or_update/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#insert_or_update/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.insert_or_update!/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#insert_or_update!/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.insert_stream/3","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","ref":"Plausible.ImportDeletionRepo.html#insert_stream/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.load/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#load/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.one/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#one/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.one!/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#one!/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.preload/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#preload/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.prepare_query/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#prepare_query/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.put_dynamic_repo/1","doc":"","ref":"Plausible.ImportDeletionRepo.html#put_dynamic_repo/1"},{"type":"function","title":"Plausible.ImportDeletionRepo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","ref":"Plausible.ImportDeletionRepo.html#query/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","ref":"Plausible.ImportDeletionRepo.html#query!/3"},{"type":"function","title":"Plausible.ImportDeletionRepo.reload/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#reload/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.reload!/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#reload!/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.start_link/1","doc":"","ref":"Plausible.ImportDeletionRepo.html#start_link/1"},{"type":"function","title":"Plausible.ImportDeletionRepo.stop/1","doc":"","ref":"Plausible.ImportDeletionRepo.html#stop/1"},{"type":"function","title":"Plausible.ImportDeletionRepo.stream/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#stream/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.to_inline_sql/2","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","ref":"Plausible.ImportDeletionRepo.html#to_inline_sql/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","ref":"Plausible.ImportDeletionRepo.html#to_sql/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.update/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#update/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.update!/2","doc":"","ref":"Plausible.ImportDeletionRepo.html#update!/2"},{"type":"function","title":"Plausible.ImportDeletionRepo.update_all/3","doc":"","ref":"Plausible.ImportDeletionRepo.html#update_all/3"},{"type":"module","title":"Plausible.Imported","doc":"Context for managing site statistics imports.\n\nFor list of currently supported import sources see `Plausible.Imported.ImportSources`.\n\nFor more information on implementing importers, see `Plausible.Imported.Importer`.","ref":"Plausible.Imported.html"},{"type":"function","title":"Plausible.Imported.clamp_dates/3","doc":"","ref":"Plausible.Imported.html#clamp_dates/3"},{"type":"function","title":"Plausible.Imported.clamp_dates/4","doc":"","ref":"Plausible.Imported.html#clamp_dates/4"},{"type":"function","title":"Plausible.Imported.delete_imports_for_site/1","doc":"","ref":"Plausible.Imported.html#delete_imports_for_site/1"},{"type":"function","title":"Plausible.Imported.get_cutoff_date/1","doc":"","ref":"Plausible.Imported.html#get_cutoff_date/1"},{"type":"function","title":"Plausible.Imported.get_import/2","doc":"","ref":"Plausible.Imported.html#get_import/2"},{"type":"function","title":"Plausible.Imported.get_imports_date_range/1","doc":"","ref":"Plausible.Imported.html#get_imports_date_range/1"},{"type":"function","title":"Plausible.Imported.get_legacy_import/1","doc":"","ref":"Plausible.Imported.html#get_legacy_import/1"},{"type":"function","title":"Plausible.Imported.get_occupied_date_ranges/1","doc":"","ref":"Plausible.Imported.html#get_occupied_date_ranges/1"},{"type":"function","title":"Plausible.Imported.goals_with_path/0","doc":"","ref":"Plausible.Imported.html#goals_with_path/0"},{"type":"function","title":"Plausible.Imported.goals_with_url/0","doc":"","ref":"Plausible.Imported.html#goals_with_url/0"},{"type":"function","title":"Plausible.Imported.imported_custom_props/0","doc":"","ref":"Plausible.Imported.html#imported_custom_props/0"},{"type":"function","title":"Plausible.Imported.list_all_imports/2","doc":"","ref":"Plausible.Imported.html#list_all_imports/2"},{"type":"function","title":"Plausible.Imported.list_complete_import_ids/1","doc":"","ref":"Plausible.Imported.html#list_complete_import_ids/1"},{"type":"function","title":"Plausible.Imported.listen/0","doc":"","ref":"Plausible.Imported.html#listen/0"},{"type":"function","title":"Plausible.Imported.load_import_data/1","doc":"","ref":"Plausible.Imported.html#load_import_data/1"},{"type":"function","title":"Plausible.Imported.max_complete_imports/0","doc":"","ref":"Plausible.Imported.html#max_complete_imports/0"},{"type":"function","title":"Plausible.Imported.other_imports_in_progress?/1","doc":"","ref":"Plausible.Imported.html#other_imports_in_progress?/1"},{"type":"function","title":"Plausible.Imported.schemas/0","doc":"","ref":"Plausible.Imported.html#schemas/0"},{"type":"function","title":"Plausible.Imported.tables/0","doc":"","ref":"Plausible.Imported.html#tables/0"},{"type":"module","title":"Plausible.Imported.Buffer","doc":"This GenServer inserts records into Clickhouse `imported_*` tables. Multiple buffers are\nautomatically created for each table. Records are flushed when the table buffer reaches the\nmaximum size, defined by `max_buffer_size/0`.","ref":"Plausible.Imported.Buffer.html"},{"type":"function","title":"Plausible.Imported.Buffer.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.Imported.Buffer.html#child_spec/1"},{"type":"function","title":"Plausible.Imported.Buffer.flush/2","doc":"Flushes all table buffers to Clickhouse.","ref":"Plausible.Imported.Buffer.html#flush/2"},{"type":"function","title":"Plausible.Imported.Buffer.handle_continue/2","doc":"","ref":"Plausible.Imported.Buffer.html#handle_continue/2"},{"type":"function","title":"Plausible.Imported.Buffer.init/1","doc":"","ref":"Plausible.Imported.Buffer.html#init/1"},{"type":"function","title":"Plausible.Imported.Buffer.insert_many/3","doc":"Puts the given records into the table buffer.","ref":"Plausible.Imported.Buffer.html#insert_many/3"},{"type":"function","title":"Plausible.Imported.Buffer.size/2","doc":"Returns the total count of items in the given table buffer.","ref":"Plausible.Imported.Buffer.html#size/2"},{"type":"function","title":"Plausible.Imported.Buffer.start_link/1","doc":"","ref":"Plausible.Imported.Buffer.html#start_link/1"},{"type":"function","title":"Plausible.Imported.Buffer.stop/1","doc":"","ref":"Plausible.Imported.Buffer.html#stop/1"},{"type":"module","title":"Plausible.Imported.CSVImporter","doc":"CSV importer from either S3 for which it uses ClickHouse [s3 table function](https://clickhouse.com/docs/en/sql-reference/table-functions/s3)\nor from local storage for which it uses [input function.](https://clickhouse.com/docs/en/sql-reference/table-functions/input)","ref":"Plausible.Imported.CSVImporter.html"},{"type":"function","title":"Plausible.Imported.CSVImporter.date_range/1","doc":"Extracts min/max date range from a list of uploads.\n\nExamples:\n\n iex> date_range([\n ...> %{\"filename\" => \"imported_devices_20190101_20210101.csv\"},\n ...> \"pages_20200101_20220101.csv\"\n ...> ])\n Date.range(~D[2019-01-01], ~D[2022-01-01])\n\n iex> date_range([])\n nil","ref":"Plausible.Imported.CSVImporter.html#date_range/1"},{"type":"function","title":"Plausible.Imported.CSVImporter.extract_table/1","doc":"Extracts the table name from the provided filename.\n\nRaises if the filename doesn't conform to the expected format.\n\nExamples:\n\n iex> extract_table(\"my_data.csv\")\n ** (ArgumentError) invalid filename\n\n iex> extract_table(\"imported_devices_00010101_20250101.csv\")\n \"imported_devices\"\n\n iex> extract_table(\"devices_00010101_20250101.csv\")\n \"imported_devices\"","ref":"Plausible.Imported.CSVImporter.html#extract_table/1"},{"type":"function","title":"Plausible.Imported.CSVImporter.local_dir/1","doc":"Returns local directory for CSV imports storage.\n\nBuilds upon `$DATA_DIR`, `$PERSISTENT_CACHE_DIR` or `$DEFAULT_DATA_DIR` (if set) and falls back to /tmp.\n\n`$DEFAULT_DATA_DIR` is set to `/var/lib/plausible` in container images.\n\nExamples:\n\n iex> local_dir = local_dir(_site_id = 37)\n iex> String.ends_with?(local_dir, \"/plausible-imports/37\")\n true","ref":"Plausible.Imported.CSVImporter.html#local_dir/1"},{"type":"function","title":"Plausible.Imported.CSVImporter.new_import/3","doc":"","ref":"Plausible.Imported.CSVImporter.html#new_import/3"},{"type":"function","title":"Plausible.Imported.CSVImporter.parse_filename!/1","doc":"Extracts table name and min/max dates from the filename.\n\nExamples:\n\n iex> parse_filename!(\"my_data.csv\")\n ** (ArgumentError) invalid filename\n\n iex> parse_filename!(\"imported_devices_00010101_20250101.csv\")\n {\"imported_devices\", ~D[0001-01-01], ~D[2025-01-01]}\n\n iex> parse_filename!(\"devices_00010101_20250101.csv\")\n {\"imported_devices\", ~D[0001-01-01], ~D[2025-01-01]}","ref":"Plausible.Imported.CSVImporter.html#parse_filename!/1"},{"type":"function","title":"Plausible.Imported.CSVImporter.valid_filename?/1","doc":"Checks if the provided filename conforms to the expected format.\n\nExamples:\n\n iex> valid_filename?(\"my_data.csv\")\n false\n\n iex> valid_filename?(\"imported_devices_00010101_20250101.csv\")\n true\n\n iex> valid_filename?(\"devices_00010101_20250101.csv\")\n true","ref":"Plausible.Imported.CSVImporter.html#valid_filename?/1"},{"type":"module","title":"Plausible.Imported.GoogleAnalytics4","doc":"Import implementation for Google Analytics 4.","ref":"Plausible.Imported.GoogleAnalytics4.html"},{"type":"function","title":"Plausible.Imported.GoogleAnalytics4.from_report/4","doc":"","ref":"Plausible.Imported.GoogleAnalytics4.html#from_report/4"},{"type":"function","title":"Plausible.Imported.GoogleAnalytics4.import_data/2","doc":"Imports stats from a Google Analytics 4 property to a Plausible site.\n\nThis function fetches Google Analytics 4 reports which are then passed in batches\nto Clickhouse by the `Plausible.Imported.Buffer` process.","ref":"Plausible.Imported.GoogleAnalytics4.html#import_data/2"},{"type":"function","title":"Plausible.Imported.GoogleAnalytics4.new_import/3","doc":"","ref":"Plausible.Imported.GoogleAnalytics4.html#new_import/3"},{"type":"module","title":"Plausible.Imported.ImportSources","doc":"Definitions of import sources.","ref":"Plausible.Imported.ImportSources.html"},{"type":"function","title":"Plausible.Imported.ImportSources.by_name/1","doc":"","ref":"Plausible.Imported.ImportSources.html#by_name/1"},{"type":"function","title":"Plausible.Imported.ImportSources.names/0","doc":"","ref":"Plausible.Imported.ImportSources.html#names/0"},{"type":"behaviour","title":"Plausible.Imported.Importer","doc":"Behaviour that should be implemented for each import source.\n\nAll imports are executed as background jobs run via `Plausible.Workers.ImportAnalytics`\nOban worker. Each import source must define a module conforming `Importer` behaviour.\n\nThe callbacks that need to be implemented:\n\n* `name/0` - Returns import source name as an atom. Example: `:universal_analytics`.\n* `label/0` - Descriptive, display friendly name of the source.\n Example: \"Google Analytics\".\n* `email_template/0` - Name of the email template to use for notifications in\n `PlausibleWeb.Email` (`import_success` and `import_failure`). The template\n should have content customized for a particular source.\n* `parse_args/1` - Receives Oban job arguments coming from `new_import/3`. Whatever\n options were passed to `new_import/3` will be present in the input map with string\n keys and values serialized to primitives. If, for instance `start_date: ~D[2024-01-03]`\n is passed as an option, `parse_args/1` receives `%{..., \"start_date\" => \"2024-01-03\"}`.\n The expectation is parsing the map values producing a keyword list of options to\n pass to `import_data/2`.\n* `import_data/2` - Receives site import struct and options produced by `parse_args/1`.\n This is where all the import processing is done. The way the import is implemented\n is entirely arbitrary except the requirement that the process as a whole must\n by synchronous. The callback is expected to return either `:ok` or `{:ok, %{...}}`\n on successful import or `{:error, ...}` on failure. The map in success tuple is\n used for updating site import struct and is passed to `on_success/2` callback.\n Please note that error tuple should be only returned on errors that can't be\n recovered from. For transient errors, the import should throw an exception or\n simply crash. The error tuple has an alternative `{error, reason, opts}` form,\n where `opts` allow to skip purging imported data so far via `skip_purge?` flag\n and skip marking the import as failed and notifying the user via `skip_mark_failed?`\n flag. Both flags are booleans.\n* `before_start/2` - Optional callback run right before scheduling import job. It's\n expected to either return `{:ok, site_import}` for the import to proceed\n or `{:error, ...}` tuple, which will be returned from `new_import/3` call.\n The `site_import` can be altered or replaced at this stage. The second argument\n are opts passed to `new_import/3`.\n* `on_success/2` - Optional callback run once site import is completed. Receives map\n returned from `import_data/2`. Expected to always return `:ok`.\n* `on_failure/1` - Optional callback run when import job fails permanently.\n\nAll sources must be added to the list in `Plausible.Imported.ImportSources`.\n\nIn order to schedule a new import job using a given source, respective importer's\n`new_import/3` function must be called. It accepts site, user who is doing the import\nand any options necessary to carry out the import.\n\nThere's an expectation that `start_date` and `end_date` are provided either as options\npassed to `new_import/3` or data in map returned from `import_data/2`. If these parameters\nare not provided, the import will eventually crash. These parameters define time range\nof imported data which is in turn used for efficient querying.\n\nLogic running inside `import_data/2` is expected to populated all `imported_*` tables\nin ClickHouse with `import_id` column set to site import's ID.\n\nManaging any configuration or authentication prior to running import is outside of\nscope of importer logic and is expected to be implemented separately.","ref":"Plausible.Imported.Importer.html"},{"type":"behaviour","title":"Running import fully synchronously - Plausible.Imported.Importer","doc":"In case it's necessary to run the whole import job fully synchronously, the\n`Plausible.Workers.ImportAnalytics` worker sends an `Oban.Notifier` message\non completion, failure or transient failure of the import.\n\nA basic usage scenario looks like this:\n\n```elixir\n{:ok, job} = Plausible.Imported.NoopImporter.new_import(\n site,\n user,\n start_date: ~D[2005-01-01],\n end_date: Date.utc_today(),\n # this option is necessary to setup the calling process as listener\n listen?: true\n)\n\nimport_id = job.args[:import_id]\n\nreceive do\n {:notification, :analytics_imports_jobs, %{\"event\" => \"complete\", \"import_id\" => ^import_id}} ->\n IO.puts(\"Job completed\")\n\n {:notification, :analytics_imports_jobs, %{\"event\" => \"transient_fail\", \"import_id\" => ^import_id}} ->\n IO.puts(\"Job failed transiently\")\n\n {:notification, :analytics_imports_jobs, %{\"event\" => \"fail\", \"import_id\" => ^import_id}} ->\n IO.puts(\"Job failed permanently\")\nafter\n 15_000 ->\n IO.puts(\"Job didn't finish in 15 seconds\")\nend\n```\n\nIn a more realistic scenario, job scheduling will be done inside a GenServer process\nlike LiveView, where notifications can be listened for via `handle_info/2`.","ref":"Plausible.Imported.Importer.html#module-running-import-fully-synchronously"},{"type":"callback","title":"Plausible.Imported.Importer.before_start/2","doc":"","ref":"Plausible.Imported.Importer.html#c:before_start/2"},{"type":"callback","title":"Plausible.Imported.Importer.email_template/0","doc":"","ref":"Plausible.Imported.Importer.html#c:email_template/0"},{"type":"callback","title":"Plausible.Imported.Importer.import_data/2","doc":"","ref":"Plausible.Imported.Importer.html#c:import_data/2"},{"type":"callback","title":"Plausible.Imported.Importer.label/0","doc":"","ref":"Plausible.Imported.Importer.html#c:label/0"},{"type":"function","title":"Plausible.Imported.Importer.listen/0","doc":"Allows to explicitly start listening for importer job notifications.\n\nListener must explicitly filter out a subset of imports that apply to the given context.","ref":"Plausible.Imported.Importer.html#listen/0"},{"type":"callback","title":"Plausible.Imported.Importer.name/0","doc":"","ref":"Plausible.Imported.Importer.html#c:name/0"},{"type":"callback","title":"Plausible.Imported.Importer.on_failure/1","doc":"","ref":"Plausible.Imported.Importer.html#c:on_failure/1"},{"type":"callback","title":"Plausible.Imported.Importer.on_success/2","doc":"","ref":"Plausible.Imported.Importer.html#c:on_success/2"},{"type":"callback","title":"Plausible.Imported.Importer.parse_args/1","doc":"","ref":"Plausible.Imported.Importer.html#c:parse_args/1"},{"type":"module","title":"Plausible.Imported.NoopImporter","doc":"Stub import implementation.","ref":"Plausible.Imported.NoopImporter.html"},{"type":"function","title":"Plausible.Imported.NoopImporter.new_import/3","doc":"","ref":"Plausible.Imported.NoopImporter.html#new_import/3"},{"type":"module","title":"Plausible.Imported.SiteImport","doc":"Site import schema.","ref":"Plausible.Imported.SiteImport.html"},{"type":"function","title":"Plausible.Imported.SiteImport.complete_changeset/2","doc":"","ref":"Plausible.Imported.SiteImport.html#complete_changeset/2"},{"type":"macro","title":"Plausible.Imported.SiteImport.completed/0","doc":"","ref":"Plausible.Imported.SiteImport.html#completed/0"},{"type":"function","title":"Plausible.Imported.SiteImport.create_changeset/3","doc":"","ref":"Plausible.Imported.SiteImport.html#create_changeset/3"},{"type":"function","title":"Plausible.Imported.SiteImport.fail_changeset/1","doc":"","ref":"Plausible.Imported.SiteImport.html#fail_changeset/1"},{"type":"macro","title":"Plausible.Imported.SiteImport.failed/0","doc":"","ref":"Plausible.Imported.SiteImport.html#failed/0"},{"type":"macro","title":"Plausible.Imported.SiteImport.importing/0","doc":"","ref":"Plausible.Imported.SiteImport.html#importing/0"},{"type":"function","title":"Plausible.Imported.SiteImport.label/1","doc":"","ref":"Plausible.Imported.SiteImport.html#label/1"},{"type":"macro","title":"Plausible.Imported.SiteImport.pending/0","doc":"","ref":"Plausible.Imported.SiteImport.html#pending/0"},{"type":"function","title":"Plausible.Imported.SiteImport.start_changeset/1","doc":"","ref":"Plausible.Imported.SiteImport.html#start_changeset/1"},{"type":"type","title":"Plausible.Imported.SiteImport.t/0","doc":"","ref":"Plausible.Imported.SiteImport.html#t:t/0"},{"type":"module","title":"Plausible.Imported.UniversalAnalytics","doc":"Import implementation for Universal Analytics.\n\nNOTE: As importing from UA is no longer supported, this module\nis only used to support rendering existing imports.","ref":"Plausible.Imported.UniversalAnalytics.html"},{"type":"function","title":"Plausible.Imported.UniversalAnalytics.new_import/3","doc":"","ref":"Plausible.Imported.UniversalAnalytics.html#new_import/3"},{"type":"module","title":"Plausible.IngestRepo","doc":"Write-centric Clickhouse access interface","ref":"Plausible.IngestRepo.html"},{"type":"function","title":"Plausible.IngestRepo.aggregate/3","doc":"","ref":"Plausible.IngestRepo.html#aggregate/3"},{"type":"function","title":"Plausible.IngestRepo.aggregate/4","doc":"","ref":"Plausible.IngestRepo.html#aggregate/4"},{"type":"function","title":"Plausible.IngestRepo.all/2","doc":"","ref":"Plausible.IngestRepo.html#all/2"},{"type":"function","title":"Plausible.IngestRepo.alter_update_all/3","doc":"Similar to `Ecto.Repo.update_all/3` but uses [`ALTER TABLE ... UPDATE`](https://clickhouse.com/docs/en/sql-reference/statements/alter/update) instead.\n\nFor more information and performance implications please see:\n\n - https://clickhouse.com/blog/handling-updates-and-deletes-in-clickhouse\n - https://clickhouse.com/docs/en/guides/developer/mutations","ref":"Plausible.IngestRepo.html#alter_update_all/3"},{"type":"function","title":"Plausible.IngestRepo.checked_out?/0","doc":"","ref":"Plausible.IngestRepo.html#checked_out?/0"},{"type":"function","title":"Plausible.IngestRepo.checkout/2","doc":"","ref":"Plausible.IngestRepo.html#checkout/2"},{"type":"function","title":"Plausible.IngestRepo.child_spec/1","doc":"","ref":"Plausible.IngestRepo.html#child_spec/1"},{"type":"function","title":"Plausible.IngestRepo.clustered_table?/1","doc":"","ref":"Plausible.IngestRepo.html#clustered_table?/1"},{"type":"function","title":"Plausible.IngestRepo.config/0","doc":"","ref":"Plausible.IngestRepo.html#config/0"},{"type":"function","title":"Plausible.IngestRepo.default_options/1","doc":"","ref":"Plausible.IngestRepo.html#default_options/1"},{"type":"function","title":"Plausible.IngestRepo.delete/2","doc":"","ref":"Plausible.IngestRepo.html#delete/2"},{"type":"function","title":"Plausible.IngestRepo.delete!/2","doc":"","ref":"Plausible.IngestRepo.html#delete!/2"},{"type":"function","title":"Plausible.IngestRepo.delete_all/2","doc":"","ref":"Plausible.IngestRepo.html#delete_all/2"},{"type":"function","title":"Plausible.IngestRepo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","ref":"Plausible.IngestRepo.html#disconnect_all/2"},{"type":"function","title":"Plausible.IngestRepo.exists?/2","doc":"","ref":"Plausible.IngestRepo.html#exists?/2"},{"type":"function","title":"Plausible.IngestRepo.get/3","doc":"","ref":"Plausible.IngestRepo.html#get/3"},{"type":"function","title":"Plausible.IngestRepo.get!/3","doc":"","ref":"Plausible.IngestRepo.html#get!/3"},{"type":"function","title":"Plausible.IngestRepo.get_by/3","doc":"","ref":"Plausible.IngestRepo.html#get_by/3"},{"type":"function","title":"Plausible.IngestRepo.get_by!/3","doc":"","ref":"Plausible.IngestRepo.html#get_by!/3"},{"type":"function","title":"Plausible.IngestRepo.get_dynamic_repo/0","doc":"","ref":"Plausible.IngestRepo.html#get_dynamic_repo/0"},{"type":"function","title":"Plausible.IngestRepo.insert/2","doc":"","ref":"Plausible.IngestRepo.html#insert/2"},{"type":"function","title":"Plausible.IngestRepo.insert!/2","doc":"","ref":"Plausible.IngestRepo.html#insert!/2"},{"type":"function","title":"Plausible.IngestRepo.insert_all/3","doc":"","ref":"Plausible.IngestRepo.html#insert_all/3"},{"type":"function","title":"Plausible.IngestRepo.insert_or_update/2","doc":"","ref":"Plausible.IngestRepo.html#insert_or_update/2"},{"type":"function","title":"Plausible.IngestRepo.insert_or_update!/2","doc":"","ref":"Plausible.IngestRepo.html#insert_or_update!/2"},{"type":"function","title":"Plausible.IngestRepo.insert_stream/3","doc":"Similar to `insert_all/2` but with the following differences:\n\n - accepts rows as streams or lists\n - sends rows as a chunked request\n - doesn't autogenerate ids or does any other preprocessing\n\nExample:\n\n Repo.query!(\"create table ecto_ch_demo(a UInt64, b String) engine Null\")\n\n defmodule Demo do\n use Ecto.Schema\n\n @primary_key false\n schema \"ecto_ch_demo\" do\n field :a, Ch, type: \"UInt64\"\n field :b, :string\n end\n end\n\n rows = Stream.map(1..100_000, fn i -> %{a: i, b: to_string(i)} end)\n {100_000, nil} = Repo.insert_stream(Demo, rows)\n\n # schemaless\n {100_000, nil} = Repo.insert_stream(\"ecto_ch_demo\", rows, types: [a: Ch.Types.u64(), b: :string])","ref":"Plausible.IngestRepo.html#insert_stream/3"},{"type":"function","title":"Plausible.IngestRepo.load/2","doc":"","ref":"Plausible.IngestRepo.html#load/2"},{"type":"function","title":"Plausible.IngestRepo.one/2","doc":"","ref":"Plausible.IngestRepo.html#one/2"},{"type":"function","title":"Plausible.IngestRepo.one!/2","doc":"","ref":"Plausible.IngestRepo.html#one!/2"},{"type":"function","title":"Plausible.IngestRepo.preload/3","doc":"","ref":"Plausible.IngestRepo.html#preload/3"},{"type":"function","title":"Plausible.IngestRepo.prepare_query/3","doc":"","ref":"Plausible.IngestRepo.html#prepare_query/3"},{"type":"function","title":"Plausible.IngestRepo.put_dynamic_repo/1","doc":"","ref":"Plausible.IngestRepo.html#put_dynamic_repo/1"},{"type":"function","title":"Plausible.IngestRepo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","ref":"Plausible.IngestRepo.html#query/3"},{"type":"function","title":"Plausible.IngestRepo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","ref":"Plausible.IngestRepo.html#query!/3"},{"type":"function","title":"Plausible.IngestRepo.reload/2","doc":"","ref":"Plausible.IngestRepo.html#reload/2"},{"type":"function","title":"Plausible.IngestRepo.reload!/2","doc":"","ref":"Plausible.IngestRepo.html#reload!/2"},{"type":"function","title":"Plausible.IngestRepo.start_link/1","doc":"","ref":"Plausible.IngestRepo.html#start_link/1"},{"type":"function","title":"Plausible.IngestRepo.stop/1","doc":"","ref":"Plausible.IngestRepo.html#stop/1"},{"type":"function","title":"Plausible.IngestRepo.stream/2","doc":"","ref":"Plausible.IngestRepo.html#stream/2"},{"type":"function","title":"Plausible.IngestRepo.to_inline_sql/2","doc":"Similar to `to_sql/2` but inlines the parameters into the SQL query.\n\nSee `Ecto.Adapters.ClickHouse.to_inline_sql/2` for more information.","ref":"Plausible.IngestRepo.html#to_inline_sql/2"},{"type":"function","title":"Plausible.IngestRepo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","ref":"Plausible.IngestRepo.html#to_sql/2"},{"type":"function","title":"Plausible.IngestRepo.update/2","doc":"","ref":"Plausible.IngestRepo.html#update/2"},{"type":"function","title":"Plausible.IngestRepo.update!/2","doc":"","ref":"Plausible.IngestRepo.html#update!/2"},{"type":"function","title":"Plausible.IngestRepo.update_all/3","doc":"","ref":"Plausible.IngestRepo.html#update_all/3"},{"type":"module","title":"Plausible.Ingestion.Counters","doc":"This is instrumentation necessary for keeping track of per-domain\ninternal metrics. Due to metric labels cardinality (domain x metric_name),\nthese statistics are not suitable for prometheus/grafana exposure,\nhence an internal storage is used.\n\nThe module installs `Counters.TelemetryHandler` and periodically\nflushes the internal counter aggregates via `Counters.Buffer` interface.\n\nThe underlying database schema is running `SummingMergeTree` engine.\nTo take advantage of automatic roll-ups it provides, upon dispatching the\nbuffered records to Clickhouse this module transforms each `event_timebucket`\naggregate into a 1-minute resolution.\n\nClickhouse connection is set to insert counters asynchronously every time\na pool checkout is made. Those properties are reverted once the insert is done\n(or naturally, if the connection crashes).","ref":"Plausible.Ingestion.Counters.html"},{"type":"function","title":"Plausible.Ingestion.Counters.child_spec/1","doc":"","ref":"Plausible.Ingestion.Counters.html#child_spec/1"},{"type":"function","title":"Plausible.Ingestion.Counters.enabled?/0","doc":"","ref":"Plausible.Ingestion.Counters.html#enabled?/0"},{"type":"function","title":"Plausible.Ingestion.Counters.stop/1","doc":"","ref":"Plausible.Ingestion.Counters.html#stop/1"},{"type":"module","title":"Plausible.Ingestion.Counters.Buffer","doc":"A buffer aggregating counters for internal metrics, within 10 seconds time buckets.\n\nSee `Plausible.Ingestion.Counters` for integration.\n\nFlushing is by default possible only once the 10s bucket is complete\n(its window has moved). This is to avoid race conditions \nwhen clearing up the buffer on dequeue - because there is no atomic \"get and delete\",\nand items are buffered concurrently, there is a gap between get and delete\nin which items written may disappear otherwise.\n\n`aggregate_bucket_fn` and `flush_boundary_fn` control that semantics and\nare configurable only for test purposes.","ref":"Plausible.Ingestion.Counters.Buffer.html"},{"type":"function","title":"Plausible.Ingestion.Counters.Buffer.aggregate/4","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#aggregate/4"},{"type":"function","title":"Plausible.Ingestion.Counters.Buffer.bucket_10s/1","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#bucket_10s/1"},{"type":"function","title":"Plausible.Ingestion.Counters.Buffer.flush/2","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#flush/2"},{"type":"function","title":"Plausible.Ingestion.Counters.Buffer.new/2","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#new/2"},{"type":"function","title":"Plausible.Ingestion.Counters.Buffer.previous_10s/1","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#previous_10s/1"},{"type":"type","title":"Plausible.Ingestion.Counters.Buffer.bucket_fn_opt/0","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#t:bucket_fn_opt/0"},{"type":"type","title":"Plausible.Ingestion.Counters.Buffer.t/0","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#t:t/0"},{"type":"type","title":"Plausible.Ingestion.Counters.Buffer.unix_timestamp/0","doc":"","ref":"Plausible.Ingestion.Counters.Buffer.html#t:unix_timestamp/0"},{"type":"module","title":"Plausible.Ingestion.Counters.Record","doc":"Clickhouse schema for storing ingest counter metrics","ref":"Plausible.Ingestion.Counters.Record.html"},{"type":"type","title":"Plausible.Ingestion.Counters.Record.t/0","doc":"","ref":"Plausible.Ingestion.Counters.Record.html#t:t/0"},{"type":"module","title":"Plausible.Ingestion.Counters.TelemetryHandler","doc":"Subscribes to telemetry events emitted by `Plausible.Ingestion.Event`.\nEvery time a request derived event is either dispatched to clickhouse or dropped,\na telemetry event is emitted respectively. That event is captured here,\nits metadata is extracted and sent for internal stats aggregation via\n`Counters.Buffer` interface.","ref":"Plausible.Ingestion.Counters.TelemetryHandler.html"},{"type":"function","title":"Plausible.Ingestion.Counters.TelemetryHandler.handle_event/4","doc":"","ref":"Plausible.Ingestion.Counters.TelemetryHandler.html#handle_event/4"},{"type":"function","title":"Plausible.Ingestion.Counters.TelemetryHandler.install/1","doc":"","ref":"Plausible.Ingestion.Counters.TelemetryHandler.html#install/1"},{"type":"module","title":"Plausible.Ingestion.Event","doc":"This module exposes the `build_and_buffer/1` function capable of\nturning %Plausible.Ingestion.Request{} into a series of events that in turn\nare uniformly either buffered in batches (to Clickhouse) or dropped\n(e.g. due to spam blocklist) from the processing pipeline.","ref":"Plausible.Ingestion.Event.html"},{"type":"function","title":"Plausible.Ingestion.Event.build_and_buffer/1","doc":"","ref":"Plausible.Ingestion.Event.html#build_and_buffer/1"},{"type":"function","title":"Plausible.Ingestion.Event.emit_telemetry_buffered/1","doc":"","ref":"Plausible.Ingestion.Event.html#emit_telemetry_buffered/1"},{"type":"function","title":"Plausible.Ingestion.Event.emit_telemetry_dropped/2","doc":"","ref":"Plausible.Ingestion.Event.html#emit_telemetry_dropped/2"},{"type":"function","title":"Plausible.Ingestion.Event.telemetry_event_buffered/0","doc":"","ref":"Plausible.Ingestion.Event.html#telemetry_event_buffered/0"},{"type":"function","title":"Plausible.Ingestion.Event.telemetry_event_dropped/0","doc":"","ref":"Plausible.Ingestion.Event.html#telemetry_event_dropped/0"},{"type":"function","title":"Plausible.Ingestion.Event.telemetry_pipeline_step_duration/0","doc":"","ref":"Plausible.Ingestion.Event.html#telemetry_pipeline_step_duration/0"},{"type":"type","title":"Plausible.Ingestion.Event.drop_reason/0","doc":"","ref":"Plausible.Ingestion.Event.html#t:drop_reason/0"},{"type":"type","title":"Plausible.Ingestion.Event.t/0","doc":"","ref":"Plausible.Ingestion.Event.html#t:t/0"},{"type":"module","title":"Plausible.Ingestion.Event.Revenue","doc":"Revenue specific functions for the ingestion scope","ref":"Plausible.Ingestion.Event.Revenue.html"},{"type":"function","title":"Plausible.Ingestion.Event.Revenue.get_revenue_attrs/1","doc":"","ref":"Plausible.Ingestion.Event.Revenue.html#get_revenue_attrs/1"},{"type":"module","title":"Plausible.Ingestion.Request","doc":"The %Plausible.Ingestion.Request{} struct stores all needed fields\nto create an event downstream. Pre-eliminary validation is made\nto detect user errors early.","ref":"Plausible.Ingestion.Request.html"},{"type":"function","title":"Plausible.Ingestion.Request.build/2","doc":"Builds and initially validates %Plausible.Ingestion.Request{} struct from %Plug.Conn{}.","ref":"Plausible.Ingestion.Request.html#build/2"},{"type":"function","title":"Plausible.Ingestion.Request.sanitize_hostname/1","doc":"Removes the \"www\" part of a hostname.","ref":"Plausible.Ingestion.Request.html#sanitize_hostname/1"},{"type":"type","title":"Plausible.Ingestion.Request.t/0","doc":"","ref":"Plausible.Ingestion.Request.html#t:t/0"},{"type":"module","title":"Plausible.Ingestion.Request.Revenue","doc":"Revenue specific functions for the ingestion scope","ref":"Plausible.Ingestion.Request.Revenue.html"},{"type":"function","title":"Plausible.Ingestion.Request.Revenue.put_revenue_source/2","doc":"","ref":"Plausible.Ingestion.Request.Revenue.html#put_revenue_source/2"},{"type":"module","title":"Plausible.License","doc":"This module ensures that you cannot run Plausible Analytics Enterprise Edition without a valid license key.\n The software contained within the ee/ and assets/js/dashboard/ee directories are Copyright © Plausible Insights OÜ.\n We have made this code available solely for informational and transparency purposes. No rights are granted to use,\n distribute, or exploit this software in any form.\n\n Any attempt to disable or modify the behavior of this module will be considered a violation of copyright.\n If you wish to use the Plausible Analytics Enterprise Edition for your own requirements, please contact us\n at hello@plausible.io to discuss obtaining a license.","ref":"Plausible.License.html"},{"type":"function","title":"Plausible.License.ensure_valid_license/0","doc":"","ref":"Plausible.License.html#ensure_valid_license/0"},{"type":"module","title":"Plausible.LiveViewTest","doc":"Temporary fix for `Phoenix.LiveViewTest.render_component/2` failing CI with warnings.\n\nThis module can be removed once Plausible switches to `phoenix_live_view ~> 1.0.0`","ref":"Plausible.LiveViewTest.html"},{"type":"macro","title":"Plausible.LiveViewTest.render_component/3","doc":"Same as `Phoenix.LiveViewTest.render_component/2` but with backported fixes from\nhttps://github.com/phoenixframework/phoenix_live_view/commit/489e8de024e03976e9ae38138eec517fbd456d27","ref":"Plausible.LiveViewTest.html#render_component/3"},{"type":"module","title":"Plausible.Mailer","doc":"","ref":"Plausible.Mailer.html"},{"type":"function","title":"Plausible.Mailer.deliver/1","doc":"","ref":"Plausible.Mailer.html#deliver/1"},{"type":"function","title":"Plausible.Mailer.deliver_later/2","doc":"","ref":"Plausible.Mailer.html#deliver_later/2"},{"type":"function","title":"Plausible.Mailer.deliver_later!/2","doc":"","ref":"Plausible.Mailer.html#deliver_later!/2"},{"type":"function","title":"Plausible.Mailer.deliver_now/2","doc":"","ref":"Plausible.Mailer.html#deliver_now/2"},{"type":"function","title":"Plausible.Mailer.deliver_now!/2","doc":"","ref":"Plausible.Mailer.html#deliver_now!/2"},{"type":"function","title":"Plausible.Mailer.send/1","doc":"","ref":"Plausible.Mailer.html#send/1"},{"type":"type","title":"Plausible.Mailer.result/0","doc":"","ref":"Plausible.Mailer.html#t:result/0"},{"type":"module","title":"Plausible.MigrationUtils","doc":"Base module for to use in Clickhouse migrations","ref":"Plausible.MigrationUtils.html"},{"type":"function","title":"Plausible.MigrationUtils.dictionary_connection_params/0","doc":"","ref":"Plausible.MigrationUtils.html#dictionary_connection_params/0"},{"type":"function","title":"Plausible.MigrationUtils.on_cluster_statement/1","doc":"","ref":"Plausible.MigrationUtils.html#on_cluster_statement/1"},{"type":"module","title":"Plausible.OpenTelemetry.Sampler","doc":"[Custom OpenTelemetry sampler](https://hexdocs.pm/opentelemetry/readme.html#samplers)\nimplementation that ignores particular traces to reduce noise. Ingestion\nHTTP requests and queries to Oban tables are ignored, for example.\n\nFor non-ignored traces, implements trace ID ratio-based sampling following the method\nfrom [built-in sampler](https://github.com/open-telemetry/opentelemetry-erlang/blob/main/apps/opentelemetry/src/otel_sampler_trace_id_ratio_based.erl).","ref":"Plausible.OpenTelemetry.Sampler.html"},{"type":"module","title":"Plausible.PaddleApi.Mock","doc":"","ref":"Plausible.PaddleApi.Mock.html"},{"type":"function","title":"Plausible.PaddleApi.Mock.fetch_prices/2","doc":"","ref":"Plausible.PaddleApi.Mock.html#fetch_prices/2"},{"type":"function","title":"Plausible.PaddleApi.Mock.get_invoices/1","doc":"","ref":"Plausible.PaddleApi.Mock.html#get_invoices/1"},{"type":"function","title":"Plausible.PaddleApi.Mock.get_subscription/1","doc":"","ref":"Plausible.PaddleApi.Mock.html#get_subscription/1"},{"type":"function","title":"Plausible.PaddleApi.Mock.update_subscription/2","doc":"","ref":"Plausible.PaddleApi.Mock.html#update_subscription/2"},{"type":"function","title":"Plausible.PaddleApi.Mock.update_subscription_preview/2","doc":"","ref":"Plausible.PaddleApi.Mock.html#update_subscription_preview/2"},{"type":"module","title":"Plausible.Pagination","doc":"Cursor-based pagination.","ref":"Plausible.Pagination.html"},{"type":"function","title":"Plausible.Pagination.paginate/4","doc":"","ref":"Plausible.Pagination.html#paginate/4"},{"type":"module","title":"Plausible.Plugins.API.Capabilities","doc":"Context module for querying API capabilities","ref":"Plausible.Plugins.API.Capabilities.html"},{"type":"function","title":"Plausible.Plugins.API.Capabilities.get/1","doc":"","ref":"Plausible.Plugins.API.Capabilities.html#get/1"},{"type":"module","title":"Plausible.Plugins.API.CustomProps","doc":"Plugins API context module for Custom Props.\nAll high level Custom Props operations should be implemented here.","ref":"Plausible.Plugins.API.CustomProps.html"},{"type":"function","title":"Plausible.Plugins.API.CustomProps.disable/2","doc":"","ref":"Plausible.Plugins.API.CustomProps.html#disable/2"},{"type":"function","title":"Plausible.Plugins.API.CustomProps.enable/2","doc":"","ref":"Plausible.Plugins.API.CustomProps.html#enable/2"},{"type":"module","title":"Plausible.Plugins.API.Funnels","doc":"Plugins API context module for Funnels.\nAll high level Funnel operations should be implemented here.","ref":"Plausible.Plugins.API.Funnels.html"},{"type":"function","title":"Plausible.Plugins.API.Funnels.create/2","doc":"","ref":"Plausible.Plugins.API.Funnels.html#create/2"},{"type":"function","title":"Plausible.Plugins.API.Funnels.get/2","doc":"","ref":"Plausible.Plugins.API.Funnels.html#get/2"},{"type":"function","title":"Plausible.Plugins.API.Funnels.get_funnels/2","doc":"","ref":"Plausible.Plugins.API.Funnels.html#get_funnels/2"},{"type":"type","title":"Plausible.Plugins.API.Funnels.create_request/0","doc":"","ref":"Plausible.Plugins.API.Funnels.html#t:create_request/0"},{"type":"module","title":"Plausible.Plugins.API.Goals","doc":"Plugins API context module for Goals.\nAll high level Goal operations should be implemented here.","ref":"Plausible.Plugins.API.Goals.html"},{"type":"function","title":"Plausible.Plugins.API.Goals.create/2","doc":"","ref":"Plausible.Plugins.API.Goals.html#create/2"},{"type":"function","title":"Plausible.Plugins.API.Goals.delete/2","doc":"","ref":"Plausible.Plugins.API.Goals.html#delete/2"},{"type":"function","title":"Plausible.Plugins.API.Goals.get/2","doc":"","ref":"Plausible.Plugins.API.Goals.html#get/2"},{"type":"function","title":"Plausible.Plugins.API.Goals.get_goals/2","doc":"","ref":"Plausible.Plugins.API.Goals.html#get_goals/2"},{"type":"type","title":"Plausible.Plugins.API.Goals.create_request/0","doc":"","ref":"Plausible.Plugins.API.Goals.html#t:create_request/0"},{"type":"module","title":"Plausible.Plugins.API.SharedLinks","doc":"Plugins API context module for Shared Links.\nAll high level Shared Links operations should be implemented here.","ref":"Plausible.Plugins.API.SharedLinks.html"},{"type":"function","title":"Plausible.Plugins.API.SharedLinks.get/2","doc":"","ref":"Plausible.Plugins.API.SharedLinks.html#get/2"},{"type":"function","title":"Plausible.Plugins.API.SharedLinks.get_or_create/3","doc":"","ref":"Plausible.Plugins.API.SharedLinks.html#get_or_create/3"},{"type":"function","title":"Plausible.Plugins.API.SharedLinks.get_shared_links/2","doc":"","ref":"Plausible.Plugins.API.SharedLinks.html#get_shared_links/2"},{"type":"module","title":"Plausible.Plugins.API.Token","doc":"Ecto schema for Plugins API Tokens.\nTokens are stored hashed and require a description.\n\nTokens are considered secret, although the Plugins API\nby nature will expose very little, if any, destructive/insecure operations.\n\nThe raw token version is meant to be presented to the user upon creation.\nIt is prefixed with a plain text identifier allowing source scanning\nfor leaked secrets.","ref":"Plausible.Plugins.API.Token.html"},{"type":"function","title":"Plausible.Plugins.API.Token.generate/1","doc":"","ref":"Plausible.Plugins.API.Token.html#generate/1"},{"type":"function","title":"Plausible.Plugins.API.Token.hash/1","doc":"","ref":"Plausible.Plugins.API.Token.html#hash/1"},{"type":"function","title":"Plausible.Plugins.API.Token.insert_changeset/3","doc":"","ref":"Plausible.Plugins.API.Token.html#insert_changeset/3"},{"type":"function","title":"Plausible.Plugins.API.Token.last_used_humanize/1","doc":"","ref":"Plausible.Plugins.API.Token.html#last_used_humanize/1"},{"type":"function","title":"Plausible.Plugins.API.Token.prefix/0","doc":"Raw tokens are prefixed so that tools like\nhttps://docs.github.com/en/code-security/secret-scanning/about-secret-scanning\ncan scan repositories for accidental secret commits.","ref":"Plausible.Plugins.API.Token.html#prefix/0"},{"type":"type","title":"Plausible.Plugins.API.Token.t/0","doc":"","ref":"Plausible.Plugins.API.Token.html#t:t/0"},{"type":"module","title":"Plausible.Plugins.API.Tokens","doc":"Context module for Plugins API Tokens.\nExposes high-level operation for token-based authentication flows.","ref":"Plausible.Plugins.API.Tokens.html"},{"type":"function","title":"Plausible.Plugins.API.Tokens.any?/1","doc":"","ref":"Plausible.Plugins.API.Tokens.html#any?/1"},{"type":"function","title":"Plausible.Plugins.API.Tokens.create/3","doc":"","ref":"Plausible.Plugins.API.Tokens.html#create/3"},{"type":"function","title":"Plausible.Plugins.API.Tokens.delete/2","doc":"","ref":"Plausible.Plugins.API.Tokens.html#delete/2"},{"type":"function","title":"Plausible.Plugins.API.Tokens.find/1","doc":"","ref":"Plausible.Plugins.API.Tokens.html#find/1"},{"type":"function","title":"Plausible.Plugins.API.Tokens.list/1","doc":"","ref":"Plausible.Plugins.API.Tokens.html#list/1"},{"type":"function","title":"Plausible.Plugins.API.Tokens.update_last_seen/2","doc":"","ref":"Plausible.Plugins.API.Tokens.html#update_last_seen/2"},{"type":"module","title":"Plausible.PromEx","doc":"","ref":"Plausible.PromEx.html"},{"type":"function","title":"Plausible.PromEx.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.PromEx.html#child_spec/1"},{"type":"module","title":"Plausible.PromEx.Plugins.PlausibleMetrics","doc":"Custom PromEx plugin for instrumenting code within Plausible app.","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html"},{"type":"function","title":"Plausible.PromEx.Plugins.PlausibleMetrics.execute_cache_metrics/0","doc":"Fire telemetry events for various caches","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html#execute_cache_metrics/0"},{"type":"function","title":"Plausible.PromEx.Plugins.PlausibleMetrics.execute_write_buffer_metrics/0","doc":"Add telemetry events for Session and Event write buffers","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html#execute_write_buffer_metrics/0"},{"type":"function","title":"Plausible.PromEx.Plugins.PlausibleMetrics.measure_duration/3","doc":"","ref":"Plausible.PromEx.Plugins.PlausibleMetrics.html#measure_duration/3"},{"type":"module","title":"Plausible.Props","doc":"Context module for handling custom event props.","ref":"Plausible.Props.html"},{"type":"function","title":"Plausible.Props.allow/2","doc":"Allows a prop key or a list of props keys to be included in ClickHouse\nqueries. Allowing prop keys does not affect ingestion, as we don't want any\ndata to be dropped or lost.","ref":"Plausible.Props.html#allow/2"},{"type":"function","title":"Plausible.Props.allow_changeset/2","doc":"","ref":"Plausible.Props.html#allow_changeset/2"},{"type":"function","title":"Plausible.Props.allow_existing_props/1","doc":"Allows the 300 most frequent props keys for a specific site over\nthe past 6 months.","ref":"Plausible.Props.html#allow_existing_props/1"},{"type":"function","title":"Plausible.Props.allowed_for/2","doc":"Returns the custom props allowed in queries for the given site. There are\ntwo factors deciding whether a custom property is allowed for a site.\n\n#","ref":"Plausible.Props.html#allowed_for/2"},{"type":"function","title":"1. Subscription plan including the props feature. - Plausible.Props.allowed_for/2","doc":"Internally used keys (i.e. `[\"url\", \"path\", \"search_query\"]`) are always allowed,\neven for plans that don't include props. For any other props, access to the\nCustom Properties feature is required.\n\n#","ref":"Plausible.Props.html#allowed_for/2-1-subscription-plan-including-the-props-feature"},{"type":"function","title":"2. The site having an `allowed_event_props` list configured. - Plausible.Props.allowed_for/2","doc":"For customers with a configured `allowed_event_props` list, this function\nreturns that list (+ internally used keys). That helps to filter out garbage\nprops which people might not want to see in their dashboards.\n\nWith the `bypass_setup?` boolean option you can override the requirement of\nthe site having set up props in the `allowed_event_props` list. For example,\nthis is currently used for fetching allowed properties in Stats API queries\nin order to ensure the props feature access.\n\nSince `allowed_event_props` was added after the props feature had already\nbeen used for a while, there are sites with `allowed_event_props = nil`. For\nthose sites, all custom properties that exist in the database are allowed to\nbe queried.","ref":"Plausible.Props.html#allowed_for/2-2-the-site-having-an-allowed_event_props-list-configured"},{"type":"function","title":"Plausible.Props.configured?/1","doc":"Returns whether the site has configured custom props or not.","ref":"Plausible.Props.html#configured?/1"},{"type":"function","title":"Plausible.Props.disallow/2","doc":"Removes previously allowed prop key(s) from the allow list. This means this\nprop key won't be included in ClickHouse queries. This doesn't drop any\nClickHouse data, nor affects ingestion.","ref":"Plausible.Props.html#disallow/2"},{"type":"function","title":"Plausible.Props.ensure_prop_key_accessible/2","doc":"","ref":"Plausible.Props.html#ensure_prop_key_accessible/2"},{"type":"function","title":"Plausible.Props.internal_keys/0","doc":"Lists prop keys used internally.\n\nThese props should be allowed by default, and should not be displayed in the\nprops settings page. For example, `url` is a special prop key used for file\ndownloads and outbound links. It doesn't make sense to remove this prop key\nfrom the allow list, or to suggest users to add this prop key.","ref":"Plausible.Props.html#internal_keys/0"},{"type":"function","title":"Plausible.Props.max_prop_key_length/0","doc":"","ref":"Plausible.Props.html#max_prop_key_length/0"},{"type":"function","title":"Plausible.Props.max_prop_value_length/0","doc":"","ref":"Plausible.Props.html#max_prop_value_length/0"},{"type":"function","title":"Plausible.Props.max_props/0","doc":"","ref":"Plausible.Props.html#max_props/0"},{"type":"function","title":"Plausible.Props.suggest_keys_to_allow/2","doc":"Queries the events table to fetch the 300 most frequent prop keys\nfor a specific site over the past 6 months, excluding keys that are already\nallowed.","ref":"Plausible.Props.html#suggest_keys_to_allow/2"},{"type":"type","title":"Plausible.Props.prop/0","doc":"","ref":"Plausible.Props.html#t:prop/0"},{"type":"module","title":"Plausible.Purge","doc":"Deletes data from a site.\n\nStats are stored on Clickhouse, and unlike other databases data deletion is\ndone asynchronously.\n\nAll import tables have MergeTree's deduplication mechanism _disabled_ by setting\n`replicated_deduplication_window` from default 100 to 0. When enabled, every insert\ninto a given table is compared against hashes of 100 previous inserts (as complete\nparts, not concrete rows) and ignored when match is found. The prupose of that\nmechanism is making inserts of exact same batches idempotent when retrying them\nshortly after - for instance due to timeout, when the client can't easily tell if\nprevious insert succeeded or not. Deduplication, however, only considers inserts,\nnot mutations. Deletions do not affect stored hashes, so further inserts of parts\nthat were deleted will still be treated as duplicates. That's why this feature\nis disabled for import tables.\n\nAlthough deletions are asynchronous, the parts to delete are \"remembered\", so there's\nno risk of overlapping deletion causing problems with import following right after it.\n\nIMPORTANT: Deletion requires revision if/when import tables get moved to sharded CH\ncluster setup. Mutation queries, which have to be run with `ON CLUSTER` in such setup,\ndispatch independent queries across shards and those queries can start at different\ntimes. This in turn means risk of deletions corrupting data of follow-up inserts\nin some edge cases. Ideally, imported entries should be unique for a given import\n- an extra `import_id` column can be introduced, holding identifier. Last processed\nimport identifier should be stored with other site data and should be used for scoping\nimported stats queries. No longer used imports can then be safely removed fully\nasynchronously.\n\n- [Clickhouse `ALTER TABLE ... DELETE` Statement](https://clickhouse.com/docs/en/sql-reference/statements/alter/delete)\n- [Synchronicity of `ALTER` Queries](https://clickhouse.com/docs/en/sql-reference/statements/alter/#synchronicity-of-alter-queries)","ref":"Plausible.Purge.html"},{"type":"function","title":"Plausible.Purge.delete_imported_stats!/1","doc":"Deletes imported stats from and clears the `stats_start_date` field.\n\nThe `stats_start_date` is expected to get repopulated the next time\n`Plausible.Sites.stats_start_date/1` is called.\n\nIf the input argument is a site, all imported stats are deleted. If it's a site import,\nonly imported stats for that import are deleted.","ref":"Plausible.Purge.html#delete_imported_stats!/1"},{"type":"function","title":"Plausible.Purge.delete_imported_stats!/2","doc":"","ref":"Plausible.Purge.html#delete_imported_stats!/2"},{"type":"function","title":"Plausible.Purge.delete_native_stats!/1","doc":"Move stats pointers so that no historical stats are available.","ref":"Plausible.Purge.html#delete_native_stats!/1"},{"type":"function","title":"Plausible.Purge.reset!/1","doc":"","ref":"Plausible.Purge.html#reset!/1"},{"type":"module","title":"Plausible.RateLimit","doc":"Thin wrapper around `:ets.update_counter/4` and a\nclean-up process to act as a rate limiter.","ref":"Plausible.RateLimit.html"},{"type":"function","title":"Plausible.RateLimit.check_rate/5","doc":"Checks the rate-limit for a key.","ref":"Plausible.RateLimit.html#check_rate/5"},{"type":"function","title":"Plausible.RateLimit.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.RateLimit.html#child_spec/1"},{"type":"function","title":"Plausible.RateLimit.start_link/1","doc":"Starts the process that creates and cleans the ETS table.\n\nAccepts the following options:\n - `GenServer.options()`\n - `:table` for the ETS table name, defaults to `Elixir.Plausible.RateLimit`\n - `:clean_period` for how often to perform garbage collection","ref":"Plausible.RateLimit.html#start_link/1"},{"type":"module","title":"Plausible.Release","doc":"","ref":"Plausible.Release.html"},{"type":"function","title":"Plausible.Release.configure_ref_inspector/0","doc":"","ref":"Plausible.Release.html#configure_ref_inspector/0"},{"type":"function","title":"Plausible.Release.configure_ua_inspector/0","doc":"","ref":"Plausible.Release.html#configure_ua_inspector/0"},{"type":"function","title":"Plausible.Release.createdb/0","doc":"","ref":"Plausible.Release.html#createdb/0"},{"type":"function","title":"Plausible.Release.dump_plans/0","doc":"","ref":"Plausible.Release.html#dump_plans/0"},{"type":"function","title":"Plausible.Release.migrate/0","doc":"","ref":"Plausible.Release.html#migrate/0"},{"type":"function","title":"Plausible.Release.pending_migrations/0","doc":"","ref":"Plausible.Release.html#pending_migrations/0"},{"type":"function","title":"Plausible.Release.rollback/0","doc":"","ref":"Plausible.Release.html#rollback/0"},{"type":"function","title":"Plausible.Release.seed/0","doc":"","ref":"Plausible.Release.html#seed/0"},{"type":"function","title":"Plausible.Release.should_be_first_launch?/0","doc":"","ref":"Plausible.Release.html#should_be_first_launch?/0"},{"type":"module","title":"Plausible.Repo","doc":"","ref":"Plausible.Repo.html"},{"type":"function","title":"Plausible.Repo.aggregate/3","doc":"","ref":"Plausible.Repo.html#aggregate/3"},{"type":"function","title":"Plausible.Repo.aggregate/4","doc":"","ref":"Plausible.Repo.html#aggregate/4"},{"type":"function","title":"Plausible.Repo.all/2","doc":"","ref":"Plausible.Repo.html#all/2"},{"type":"function","title":"Plausible.Repo.checked_out?/0","doc":"","ref":"Plausible.Repo.html#checked_out?/0"},{"type":"function","title":"Plausible.Repo.checkout/2","doc":"","ref":"Plausible.Repo.html#checkout/2"},{"type":"function","title":"Plausible.Repo.child_spec/1","doc":"","ref":"Plausible.Repo.html#child_spec/1"},{"type":"function","title":"Plausible.Repo.config/0","doc":"","ref":"Plausible.Repo.html#config/0"},{"type":"function","title":"Plausible.Repo.default_options/1","doc":"","ref":"Plausible.Repo.html#default_options/1"},{"type":"function","title":"Plausible.Repo.delete/2","doc":"","ref":"Plausible.Repo.html#delete/2"},{"type":"function","title":"Plausible.Repo.delete!/2","doc":"","ref":"Plausible.Repo.html#delete!/2"},{"type":"function","title":"Plausible.Repo.delete_all/2","doc":"","ref":"Plausible.Repo.html#delete_all/2"},{"type":"function","title":"Plausible.Repo.disconnect_all/2","doc":"A convenience function for SQL-based repositories that forces all connections in the\npool to disconnect within the given interval.\n\nSee `Ecto.Adapters.SQL.disconnect_all/3` for more information.","ref":"Plausible.Repo.html#disconnect_all/2"},{"type":"function","title":"Plausible.Repo.exists?/2","doc":"","ref":"Plausible.Repo.html#exists?/2"},{"type":"function","title":"Plausible.Repo.explain/3","doc":"A convenience function for SQL-based repositories that executes an EXPLAIN statement or similar\ndepending on the adapter to obtain statistics for the given query.\n\nSee `Ecto.Adapters.SQL.explain/4` for more information.","ref":"Plausible.Repo.html#explain/3"},{"type":"function","title":"Plausible.Repo.get/3","doc":"","ref":"Plausible.Repo.html#get/3"},{"type":"function","title":"Plausible.Repo.get!/3","doc":"","ref":"Plausible.Repo.html#get!/3"},{"type":"function","title":"Plausible.Repo.get_by/3","doc":"","ref":"Plausible.Repo.html#get_by/3"},{"type":"function","title":"Plausible.Repo.get_by!/3","doc":"","ref":"Plausible.Repo.html#get_by!/3"},{"type":"function","title":"Plausible.Repo.get_dynamic_repo/0","doc":"","ref":"Plausible.Repo.html#get_dynamic_repo/0"},{"type":"function","title":"Plausible.Repo.in_transaction?/0","doc":"","ref":"Plausible.Repo.html#in_transaction?/0"},{"type":"function","title":"Plausible.Repo.insert/2","doc":"","ref":"Plausible.Repo.html#insert/2"},{"type":"function","title":"Plausible.Repo.insert!/2","doc":"","ref":"Plausible.Repo.html#insert!/2"},{"type":"function","title":"Plausible.Repo.insert_all/3","doc":"","ref":"Plausible.Repo.html#insert_all/3"},{"type":"function","title":"Plausible.Repo.insert_or_update/2","doc":"","ref":"Plausible.Repo.html#insert_or_update/2"},{"type":"function","title":"Plausible.Repo.insert_or_update!/2","doc":"","ref":"Plausible.Repo.html#insert_or_update!/2"},{"type":"function","title":"Plausible.Repo.load/2","doc":"","ref":"Plausible.Repo.html#load/2"},{"type":"function","title":"Plausible.Repo.one/2","doc":"","ref":"Plausible.Repo.html#one/2"},{"type":"function","title":"Plausible.Repo.one!/2","doc":"","ref":"Plausible.Repo.html#one!/2"},{"type":"function","title":"Plausible.Repo.paginate/2","doc":"","ref":"Plausible.Repo.html#paginate/2"},{"type":"function","title":"Plausible.Repo.preload/3","doc":"","ref":"Plausible.Repo.html#preload/3"},{"type":"function","title":"Plausible.Repo.prepare_query/3","doc":"","ref":"Plausible.Repo.html#prepare_query/3"},{"type":"function","title":"Plausible.Repo.put_dynamic_repo/1","doc":"","ref":"Plausible.Repo.html#put_dynamic_repo/1"},{"type":"function","title":"Plausible.Repo.query/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query/4` for more information.","ref":"Plausible.Repo.html#query/3"},{"type":"function","title":"Plausible.Repo.query!/3","doc":"A convenience function for SQL-based repositories that executes the given query.\n\nSee `Ecto.Adapters.SQL.query!/4` for more information.","ref":"Plausible.Repo.html#query!/3"},{"type":"function","title":"Plausible.Repo.query_many/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query.\n\nSee `Ecto.Adapters.SQL.query_many/4` for more information.","ref":"Plausible.Repo.html#query_many/3"},{"type":"function","title":"Plausible.Repo.query_many!/3","doc":"A convenience function for SQL-based repositories that executes the given multi-result query.\n\nSee `Ecto.Adapters.SQL.query_many!/4` for more information.","ref":"Plausible.Repo.html#query_many!/3"},{"type":"function","title":"Plausible.Repo.reload/2","doc":"","ref":"Plausible.Repo.html#reload/2"},{"type":"function","title":"Plausible.Repo.reload!/2","doc":"","ref":"Plausible.Repo.html#reload!/2"},{"type":"function","title":"Plausible.Repo.rollback/1","doc":"","ref":"Plausible.Repo.html#rollback/1"},{"type":"function","title":"Plausible.Repo.scrivener_defaults/0","doc":"","ref":"Plausible.Repo.html#scrivener_defaults/0"},{"type":"function","title":"Plausible.Repo.start_link/1","doc":"","ref":"Plausible.Repo.html#start_link/1"},{"type":"function","title":"Plausible.Repo.stop/1","doc":"","ref":"Plausible.Repo.html#stop/1"},{"type":"function","title":"Plausible.Repo.stream/2","doc":"","ref":"Plausible.Repo.html#stream/2"},{"type":"function","title":"Plausible.Repo.to_sql/2","doc":"A convenience function for SQL-based repositories that translates the given query to SQL.\n\nSee `Ecto.Adapters.SQL.to_sql/3` for more information.","ref":"Plausible.Repo.html#to_sql/2"},{"type":"function","title":"Plausible.Repo.transaction/2","doc":"","ref":"Plausible.Repo.html#transaction/2"},{"type":"function","title":"Plausible.Repo.update/2","doc":"","ref":"Plausible.Repo.html#update/2"},{"type":"function","title":"Plausible.Repo.update!/2","doc":"","ref":"Plausible.Repo.html#update!/2"},{"type":"function","title":"Plausible.Repo.update_all/3","doc":"","ref":"Plausible.Repo.html#update_all/3"},{"type":"module","title":"Plausible.RequestLogger","doc":"Custom request logger which:\n- Includes query parameters on the same line\n- Includes request duration on the same line","ref":"Plausible.RequestLogger.html"},{"type":"function","title":"Plausible.RequestLogger.log_request/4","doc":"","ref":"Plausible.RequestLogger.html#log_request/4"},{"type":"module","title":"Plausible.S3","doc":"Helper functions for S3 exports/imports.","ref":"Plausible.S3.html"},{"type":"function","title":"Plausible.S3.download_url/2","doc":"Returns a presigned URL to download the exported Zip archive from S3.\nThe URL expires in 300 seconds, which should be enough for a redirect.\n\nIn the current implementation the bucket always goes into the path component.","ref":"Plausible.S3.html#download_url/2"},{"type":"function","title":"Plausible.S3.export_upload_multipart/4","doc":"Chunks and uploads Zip archive to the provided S3 destination.\n\nIn the current implementation the bucket always goes into the path component.","ref":"Plausible.S3.html#export_upload_multipart/4"},{"type":"function","title":"Plausible.S3.exports_bucket/0","doc":"Returns the pre-configured S3 bucket for CSV exports.\n\n config :plausible, Plausible.S3,\n exports_bucket: System.fetch_env!(\"S3_EXPORTS_BUCKET\")\n\nExample:\n\n iex> exports_bucket()\n \"test-exports\"","ref":"Plausible.S3.html#exports_bucket/0"},{"type":"function","title":"Plausible.S3.import_clickhouse_credentials/0","doc":"Returns `access_key_id` and `secret_access_key` to be used by ClickHouse during imports from S3.\n\nExample:\n\n iex> import_clickhouse_credentials()\n %{access_key_id: \"minioadmin\", secret_access_key: \"minioadmin\"}","ref":"Plausible.S3.html#import_clickhouse_credentials/0"},{"type":"function","title":"Plausible.S3.import_presign_upload/2","doc":"Presigns an upload for an imported file.\n\nIn the current implementation the bucket always goes into the path component.\n\nExample:\n\n iex> upload = import_presign_upload(_site_id = 123, _filename = \"imported_browsers.csv\")\n iex> true = String.ends_with?(upload.s3_url, \"/test-imports/123/imported_browsers.csv\")\n iex> true = String.contains?(upload.presigned_url, \"/test-imports/123/imported_browsers.csv?X-Amz-Algorithm=AWS4-HMAC-SHA256&\")","ref":"Plausible.S3.html#import_presign_upload/2"},{"type":"function","title":"Plausible.S3.imports_bucket/0","doc":"Returns the pre-configured S3 bucket for CSV imports.\n\n config :plausible, Plausible.S3,\n imports_bucket: System.fetch_env!(\"S3_IMPORTS_BUCKET\")\n\nExample:\n\n iex> imports_bucket()\n \"test-imports\"","ref":"Plausible.S3.html#imports_bucket/0"},{"type":"module","title":"Plausible.Sentry.Client","doc":"","ref":"Plausible.Sentry.Client.html"},{"type":"function","title":"Plausible.Sentry.Client.post/3","doc":"","ref":"Plausible.Sentry.Client.html#post/3"},{"type":"module","title":"Plausible.SentryFilter","doc":"Sentry callbacks for filtering and grouping events","ref":"Plausible.SentryFilter.html"},{"type":"function","title":"Plausible.SentryFilter.before_send/1","doc":"","ref":"Plausible.SentryFilter.html#before_send/1"},{"type":"module","title":"Plausible.Session.CacheStore","doc":"","ref":"Plausible.Session.CacheStore.html"},{"type":"function","title":"Plausible.Session.CacheStore.lock_telemetry_event/0","doc":"","ref":"Plausible.Session.CacheStore.html#lock_telemetry_event/0"},{"type":"function","title":"Plausible.Session.CacheStore.on_event/4","doc":"","ref":"Plausible.Session.CacheStore.html#on_event/4"},{"type":"module","title":"Plausible.Session.Salts","doc":"","ref":"Plausible.Session.Salts.html"},{"type":"function","title":"Plausible.Session.Salts.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"Plausible.Session.Salts.html#child_spec/1"},{"type":"function","title":"Plausible.Session.Salts.fetch/0","doc":"","ref":"Plausible.Session.Salts.html#fetch/0"},{"type":"function","title":"Plausible.Session.Salts.rotate/0","doc":"","ref":"Plausible.Session.Salts.html#rotate/0"},{"type":"function","title":"Plausible.Session.Salts.start_link/1","doc":"","ref":"Plausible.Session.Salts.html#start_link/1"},{"type":"module","title":"Plausible.Shield.CountryRule","doc":"Schema for Country Block List","ref":"Plausible.Shield.CountryRule.html"},{"type":"function","title":"Plausible.Shield.CountryRule.changeset/2","doc":"","ref":"Plausible.Shield.CountryRule.html#changeset/2"},{"type":"type","title":"Plausible.Shield.CountryRule.t/0","doc":"","ref":"Plausible.Shield.CountryRule.html#t:t/0"},{"type":"module","title":"Plausible.Shield.CountryRuleCache","doc":"Allows retrieving Country Rules by domain and country code","ref":"Plausible.Shield.CountryRuleCache.html"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.child_spec/1","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#child_spec/1"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.get/2","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#get/2"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.get_or_store/3","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#get_or_store/3"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.hit_rate/1","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#hit_rate/1"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.merge_items/2","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#merge_items/2"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.ready?/1","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","ref":"Plausible.Shield.CountryRuleCache.html#ready?/1"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.refresh_all/1","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#refresh_all/1"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.refresh_updated_recently/1","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#refresh_updated_recently/1"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.size/1","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#size/1"},{"type":"function","title":"Plausible.Shield.CountryRuleCache.telemetry_event_refresh/2","doc":"","ref":"Plausible.Shield.CountryRuleCache.html#telemetry_event_refresh/2"},{"type":"module","title":"Plausible.Shield.HostnameRule","doc":"Schema for Hostnames allow list","ref":"Plausible.Shield.HostnameRule.html"},{"type":"function","title":"Plausible.Shield.HostnameRule.changeset/2","doc":"","ref":"Plausible.Shield.HostnameRule.html#changeset/2"},{"type":"type","title":"Plausible.Shield.HostnameRule.t/0","doc":"","ref":"Plausible.Shield.HostnameRule.html#t:t/0"},{"type":"module","title":"Plausible.Shield.HostnameRuleCache","doc":"Allows retrieving Hostname Rules by domain","ref":"Plausible.Shield.HostnameRuleCache.html"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.child_spec/1","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#child_spec/1"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.get/2","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#get/2"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.get_or_store/3","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#get_or_store/3"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.hit_rate/1","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#hit_rate/1"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.merge_items/2","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#merge_items/2"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.ready?/1","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","ref":"Plausible.Shield.HostnameRuleCache.html#ready?/1"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.refresh_all/1","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#refresh_all/1"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.refresh_updated_recently/1","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#refresh_updated_recently/1"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.size/1","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#size/1"},{"type":"function","title":"Plausible.Shield.HostnameRuleCache.telemetry_event_refresh/2","doc":"","ref":"Plausible.Shield.HostnameRuleCache.html#telemetry_event_refresh/2"},{"type":"module","title":"Plausible.Shield.IPRule","doc":"Schema for IP block list","ref":"Plausible.Shield.IPRule.html"},{"type":"function","title":"Plausible.Shield.IPRule.changeset/2","doc":"","ref":"Plausible.Shield.IPRule.html#changeset/2"},{"type":"type","title":"Plausible.Shield.IPRule.t/0","doc":"","ref":"Plausible.Shield.IPRule.html#t:t/0"},{"type":"module","title":"Plausible.Shield.IPRuleCache","doc":"Allows retrieving IP Rules by domain and IP","ref":"Plausible.Shield.IPRuleCache.html"},{"type":"function","title":"Plausible.Shield.IPRuleCache.child_spec/1","doc":"","ref":"Plausible.Shield.IPRuleCache.html#child_spec/1"},{"type":"function","title":"Plausible.Shield.IPRuleCache.get/2","doc":"","ref":"Plausible.Shield.IPRuleCache.html#get/2"},{"type":"function","title":"Plausible.Shield.IPRuleCache.get_or_store/3","doc":"","ref":"Plausible.Shield.IPRuleCache.html#get_or_store/3"},{"type":"function","title":"Plausible.Shield.IPRuleCache.hit_rate/1","doc":"","ref":"Plausible.Shield.IPRuleCache.html#hit_rate/1"},{"type":"function","title":"Plausible.Shield.IPRuleCache.merge_items/2","doc":"","ref":"Plausible.Shield.IPRuleCache.html#merge_items/2"},{"type":"function","title":"Plausible.Shield.IPRuleCache.ready?/1","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","ref":"Plausible.Shield.IPRuleCache.html#ready?/1"},{"type":"function","title":"Plausible.Shield.IPRuleCache.refresh_all/1","doc":"","ref":"Plausible.Shield.IPRuleCache.html#refresh_all/1"},{"type":"function","title":"Plausible.Shield.IPRuleCache.refresh_updated_recently/1","doc":"","ref":"Plausible.Shield.IPRuleCache.html#refresh_updated_recently/1"},{"type":"function","title":"Plausible.Shield.IPRuleCache.size/1","doc":"","ref":"Plausible.Shield.IPRuleCache.html#size/1"},{"type":"function","title":"Plausible.Shield.IPRuleCache.telemetry_event_refresh/2","doc":"","ref":"Plausible.Shield.IPRuleCache.html#telemetry_event_refresh/2"},{"type":"module","title":"Plausible.Shield.PageRule","doc":"Schema for Pages block list","ref":"Plausible.Shield.PageRule.html"},{"type":"function","title":"Plausible.Shield.PageRule.changeset/2","doc":"","ref":"Plausible.Shield.PageRule.html#changeset/2"},{"type":"type","title":"Plausible.Shield.PageRule.t/0","doc":"","ref":"Plausible.Shield.PageRule.html#t:t/0"},{"type":"module","title":"Plausible.Shield.PageRuleCache","doc":"Allows retrieving Page Rules by domain","ref":"Plausible.Shield.PageRuleCache.html"},{"type":"function","title":"Plausible.Shield.PageRuleCache.child_spec/1","doc":"","ref":"Plausible.Shield.PageRuleCache.html#child_spec/1"},{"type":"function","title":"Plausible.Shield.PageRuleCache.get/2","doc":"","ref":"Plausible.Shield.PageRuleCache.html#get/2"},{"type":"function","title":"Plausible.Shield.PageRuleCache.get_or_store/3","doc":"","ref":"Plausible.Shield.PageRuleCache.html#get_or_store/3"},{"type":"function","title":"Plausible.Shield.PageRuleCache.hit_rate/1","doc":"","ref":"Plausible.Shield.PageRuleCache.html#hit_rate/1"},{"type":"function","title":"Plausible.Shield.PageRuleCache.merge_items/2","doc":"","ref":"Plausible.Shield.PageRuleCache.html#merge_items/2"},{"type":"function","title":"Plausible.Shield.PageRuleCache.ready?/1","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","ref":"Plausible.Shield.PageRuleCache.html#ready?/1"},{"type":"function","title":"Plausible.Shield.PageRuleCache.refresh_all/1","doc":"","ref":"Plausible.Shield.PageRuleCache.html#refresh_all/1"},{"type":"function","title":"Plausible.Shield.PageRuleCache.refresh_updated_recently/1","doc":"","ref":"Plausible.Shield.PageRuleCache.html#refresh_updated_recently/1"},{"type":"function","title":"Plausible.Shield.PageRuleCache.size/1","doc":"","ref":"Plausible.Shield.PageRuleCache.html#size/1"},{"type":"function","title":"Plausible.Shield.PageRuleCache.telemetry_event_refresh/2","doc":"","ref":"Plausible.Shield.PageRuleCache.html#telemetry_event_refresh/2"},{"type":"module","title":"Plausible.Shields","doc":"Contextual interface for shields.","ref":"Plausible.Shields.html"},{"type":"function","title":"Plausible.Shields.add_country_rule/3","doc":"","ref":"Plausible.Shields.html#add_country_rule/3"},{"type":"function","title":"Plausible.Shields.add_hostname_rule/3","doc":"","ref":"Plausible.Shields.html#add_hostname_rule/3"},{"type":"function","title":"Plausible.Shields.add_ip_rule/3","doc":"","ref":"Plausible.Shields.html#add_ip_rule/3"},{"type":"function","title":"Plausible.Shields.add_page_rule/3","doc":"","ref":"Plausible.Shields.html#add_page_rule/3"},{"type":"function","title":"Plausible.Shields.allowed_hostname_patterns/1","doc":"","ref":"Plausible.Shields.html#allowed_hostname_patterns/1"},{"type":"function","title":"Plausible.Shields.count_country_rules/1","doc":"","ref":"Plausible.Shields.html#count_country_rules/1"},{"type":"function","title":"Plausible.Shields.count_hostname_rules/1","doc":"","ref":"Plausible.Shields.html#count_hostname_rules/1"},{"type":"function","title":"Plausible.Shields.count_ip_rules/1","doc":"","ref":"Plausible.Shields.html#count_ip_rules/1"},{"type":"function","title":"Plausible.Shields.count_page_rules/1","doc":"","ref":"Plausible.Shields.html#count_page_rules/1"},{"type":"function","title":"Plausible.Shields.country_blocked?/2","doc":"","ref":"Plausible.Shields.html#country_blocked?/2"},{"type":"function","title":"Plausible.Shields.hostname_allowed?/2","doc":"","ref":"Plausible.Shields.html#hostname_allowed?/2"},{"type":"function","title":"Plausible.Shields.ip_blocked?/2","doc":"","ref":"Plausible.Shields.html#ip_blocked?/2"},{"type":"function","title":"Plausible.Shields.list_country_rules/1","doc":"","ref":"Plausible.Shields.html#list_country_rules/1"},{"type":"function","title":"Plausible.Shields.list_hostname_rules/1","doc":"","ref":"Plausible.Shields.html#list_hostname_rules/1"},{"type":"function","title":"Plausible.Shields.list_ip_rules/1","doc":"","ref":"Plausible.Shields.html#list_ip_rules/1"},{"type":"function","title":"Plausible.Shields.list_page_rules/1","doc":"","ref":"Plausible.Shields.html#list_page_rules/1"},{"type":"function","title":"Plausible.Shields.maximum_country_rules/0","doc":"","ref":"Plausible.Shields.html#maximum_country_rules/0"},{"type":"function","title":"Plausible.Shields.maximum_hostname_rules/0","doc":"","ref":"Plausible.Shields.html#maximum_hostname_rules/0"},{"type":"function","title":"Plausible.Shields.maximum_ip_rules/0","doc":"","ref":"Plausible.Shields.html#maximum_ip_rules/0"},{"type":"function","title":"Plausible.Shields.maximum_page_rules/0","doc":"","ref":"Plausible.Shields.html#maximum_page_rules/0"},{"type":"function","title":"Plausible.Shields.page_blocked?/2","doc":"","ref":"Plausible.Shields.html#page_blocked?/2"},{"type":"function","title":"Plausible.Shields.remove_country_rule/2","doc":"","ref":"Plausible.Shields.html#remove_country_rule/2"},{"type":"function","title":"Plausible.Shields.remove_hostname_rule/2","doc":"","ref":"Plausible.Shields.html#remove_hostname_rule/2"},{"type":"function","title":"Plausible.Shields.remove_ip_rule/2","doc":"","ref":"Plausible.Shields.html#remove_ip_rule/2"},{"type":"function","title":"Plausible.Shields.remove_page_rule/2","doc":"","ref":"Plausible.Shields.html#remove_page_rule/2"},{"type":"module","title":"Plausible.Site","doc":"Site schema","ref":"Plausible.Site.html"},{"type":"function","title":"Plausible.Site.changeset/2","doc":"","ref":"Plausible.Site.html#changeset/2"},{"type":"function","title":"Plausible.Site.crm_changeset/2","doc":"","ref":"Plausible.Site.html#crm_changeset/2"},{"type":"function","title":"Plausible.Site.make_private/1","doc":"","ref":"Plausible.Site.html#make_private/1"},{"type":"function","title":"Plausible.Site.make_public/1","doc":"","ref":"Plausible.Site.html#make_public/1"},{"type":"function","title":"Plausible.Site.new/1","doc":"","ref":"Plausible.Site.html#new/1"},{"type":"function","title":"Plausible.Site.set_native_stats_start_at/2","doc":"","ref":"Plausible.Site.html#set_native_stats_start_at/2"},{"type":"function","title":"Plausible.Site.set_stats_start_date/2","doc":"","ref":"Plausible.Site.html#set_stats_start_date/2"},{"type":"function","title":"Plausible.Site.tz_offset/2","doc":"","ref":"Plausible.Site.html#tz_offset/2"},{"type":"function","title":"Plausible.Site.update_changeset/3","doc":"","ref":"Plausible.Site.html#update_changeset/3"},{"type":"type","title":"Plausible.Site.t/0","doc":"","ref":"Plausible.Site.html#t:t/0"},{"type":"module","title":"Plausible.Site.Cache","doc":"The cache allows lookups by both `domain` and `domain_changed_from`\nfields - this is to allow traffic from sites whose domains changed within a certain\ngrace period (see: `Plausible.Site.Transfer`).\n\nTo differentiate cached Site structs from those retrieved directly from the\ndatabase, a virtual schema field `from_cache?: true` is set.\nThis indicates the `Plausible.Site` struct is incomplete in comparison to its\ndatabase counterpart -- to spare bandwidth and query execution time,\nonly selected database columns are retrieved and cached.\n\nThe `@cached_schema_fields` attribute defines the list of DB columns\nqueried on each cache refresh.\n\nAlso see tests for more comprehensive examples.","ref":"Plausible.Site.Cache.html"},{"type":"function","title":"Plausible.Site.Cache.child_spec/1","doc":"","ref":"Plausible.Site.Cache.html#child_spec/1"},{"type":"function","title":"Plausible.Site.Cache.get/2","doc":"","ref":"Plausible.Site.Cache.html#get/2"},{"type":"function","title":"Plausible.Site.Cache.get_or_store/3","doc":"","ref":"Plausible.Site.Cache.html#get_or_store/3"},{"type":"function","title":"Plausible.Site.Cache.get_site_id/2","doc":"","ref":"Plausible.Site.Cache.html#get_site_id/2"},{"type":"function","title":"Plausible.Site.Cache.hit_rate/1","doc":"","ref":"Plausible.Site.Cache.html#hit_rate/1"},{"type":"function","title":"Plausible.Site.Cache.merge_items/2","doc":"","ref":"Plausible.Site.Cache.html#merge_items/2"},{"type":"function","title":"Plausible.Site.Cache.ready?/1","doc":"Ensures the cache has non-zero size unless no items exist.\nUseful for orchestrating app startup to prevent the service\ngoing up asynchronously with an empty cache.","ref":"Plausible.Site.Cache.html#ready?/1"},{"type":"function","title":"Plausible.Site.Cache.refresh_all/1","doc":"","ref":"Plausible.Site.Cache.html#refresh_all/1"},{"type":"function","title":"Plausible.Site.Cache.refresh_updated_recently/1","doc":"","ref":"Plausible.Site.Cache.html#refresh_updated_recently/1"},{"type":"function","title":"Plausible.Site.Cache.size/1","doc":"","ref":"Plausible.Site.Cache.html#size/1"},{"type":"function","title":"Plausible.Site.Cache.telemetry_event_refresh/2","doc":"","ref":"Plausible.Site.Cache.html#telemetry_event_refresh/2"},{"type":"function","title":"Plausible.Site.Cache.touch_site!/2","doc":"","ref":"Plausible.Site.Cache.html#touch_site!/2"},{"type":"module","title":"Plausible.Site.Domain","doc":"Basic interface for domain changes.\n\nOnce `Plausible.DataMigration.NumericIDs` schema migration is ready, \ndomain change operation will be enabled, accessible to the users.\n\nWe will set a transition period of 72 hours\nduring which, both old and new domains, will be accepted as traffic\nidentifiers to the same site. \n\nA periodic worker will call the `expire/0` function to end it where applicable.\nSee: `Plausible.Workers.ExpireDomainChangeTransitions`.\n\nThe underlying changeset for domain change (see: `Plausible.Site`) relies\non database trigger installed via `Plausible.Repo.Migrations.AllowDomainChange`\nPostgres migration. The trigger checks if either `domain` or `domain_changed_from`\nexist to ensure unicity.","ref":"Plausible.Site.Domain.html"},{"type":"function","title":"Plausible.Site.Domain.change/3","doc":"","ref":"Plausible.Site.Domain.html#change/3"},{"type":"function","title":"Plausible.Site.Domain.expire_change_transitions/1","doc":"","ref":"Plausible.Site.Domain.html#expire_change_transitions/1"},{"type":"module","title":"Plausible.Site.GateKeeper","doc":"Thin wrapper around `Plausible.RateLimit` for gate keeping domain-specific events\nduring the ingestion phase. When the site is allowed, gate keeping\ncheck returns `:allow`, otherwise a `:deny` tagged tuple is returned\nwith one of the following policy markers:\n * `:not_found` (indicates site not found in cache)\n * `:block` (indicates disabled sites)\n * `:throttle` (indicates rate limiting)\n\nRate Limiting buckets are configured per site (externally via the CRM).\nSee: `Plausible.Site`\n\nTo look up each site's configuration, the RateLimiter fetches\na Site by domain using `Plausible.Cache` interface.\n\nThe module defines two policies outside the regular bucket inspection:\n * when the site is not found in cache: not_found\n * when the underlying rate limiting mechanism returns\n an internal error: :allow","ref":"Plausible.Site.GateKeeper.html"},{"type":"function","title":"Plausible.Site.GateKeeper.check/2","doc":"","ref":"Plausible.Site.GateKeeper.html#check/2"},{"type":"function","title":"Plausible.Site.GateKeeper.key/1","doc":"","ref":"Plausible.Site.GateKeeper.html#key/1"},{"type":"type","title":"Plausible.Site.GateKeeper.policy/0","doc":"","ref":"Plausible.Site.GateKeeper.html#t:policy/0"},{"type":"type","title":"Plausible.Site.GateKeeper.t/0","doc":"","ref":"Plausible.Site.GateKeeper.html#t:t/0"},{"type":"module","title":"Plausible.Site.GoogleAuth","doc":"","ref":"Plausible.Site.GoogleAuth.html"},{"type":"function","title":"Plausible.Site.GoogleAuth.changeset/2","doc":"","ref":"Plausible.Site.GoogleAuth.html#changeset/2"},{"type":"function","title":"Plausible.Site.GoogleAuth.set_property/2","doc":"","ref":"Plausible.Site.GoogleAuth.html#set_property/2"},{"type":"module","title":"Plausible.Site.ImportedData","doc":"Embedded schema for analytics imports\n\nNOTE: needed by `SiteImports` data migration script","ref":"Plausible.Site.ImportedData.html"},{"type":"type","title":"Plausible.Site.ImportedData.t/0","doc":"","ref":"Plausible.Site.ImportedData.html#t:t/0"},{"type":"module","title":"Plausible.Site.InstallationMeta","doc":"Embedded schema for installation meta-data","ref":"Plausible.Site.InstallationMeta.html"},{"type":"type","title":"Plausible.Site.InstallationMeta.t/0","doc":"","ref":"Plausible.Site.InstallationMeta.html#t:t/0"},{"type":"module","title":"Plausible.Site.Membership","doc":"","ref":"Plausible.Site.Membership.html"},{"type":"function","title":"Plausible.Site.Membership.new/2","doc":"","ref":"Plausible.Site.Membership.html#new/2"},{"type":"function","title":"Plausible.Site.Membership.set_role/2","doc":"","ref":"Plausible.Site.Membership.html#set_role/2"},{"type":"type","title":"Plausible.Site.Membership.role/0","doc":"","ref":"Plausible.Site.Membership.html#t:role/0"},{"type":"type","title":"Plausible.Site.Membership.t/0","doc":"","ref":"Plausible.Site.Membership.html#t:t/0"},{"type":"module","title":"Plausible.Site.Memberships","doc":"API for site memberships and invitations","ref":"Plausible.Site.Memberships.html"},{"type":"function","title":"Plausible.Site.Memberships.accept_invitation/2","doc":"","ref":"Plausible.Site.Memberships.html#accept_invitation/2"},{"type":"function","title":"Plausible.Site.Memberships.all_pending_ownerships/1","doc":"","ref":"Plausible.Site.Memberships.html#all_pending_ownerships/1"},{"type":"function","title":"Plausible.Site.Memberships.any?/1","doc":"","ref":"Plausible.Site.Memberships.html#any?/1"},{"type":"function","title":"Plausible.Site.Memberships.any_or_pending?/1","doc":"","ref":"Plausible.Site.Memberships.html#any_or_pending?/1"},{"type":"function","title":"Plausible.Site.Memberships.bulk_create_invitation/5","doc":"","ref":"Plausible.Site.Memberships.html#bulk_create_invitation/5"},{"type":"function","title":"Plausible.Site.Memberships.bulk_transfer_ownership_direct/2","doc":"","ref":"Plausible.Site.Memberships.html#bulk_transfer_ownership_direct/2"},{"type":"function","title":"Plausible.Site.Memberships.create_invitation/4","doc":"","ref":"Plausible.Site.Memberships.html#create_invitation/4"},{"type":"function","title":"Plausible.Site.Memberships.pending?/1","doc":"","ref":"Plausible.Site.Memberships.html#pending?/1"},{"type":"function","title":"Plausible.Site.Memberships.pending_ownerships?/1","doc":"","ref":"Plausible.Site.Memberships.html#pending_ownerships?/1"},{"type":"function","title":"Plausible.Site.Memberships.reject_invitation/2","doc":"","ref":"Plausible.Site.Memberships.html#reject_invitation/2"},{"type":"function","title":"Plausible.Site.Memberships.remove_invitation/2","doc":"","ref":"Plausible.Site.Memberships.html#remove_invitation/2"},{"type":"function","title":"Plausible.Site.Memberships.transfer_ownership/2","doc":"","ref":"Plausible.Site.Memberships.html#transfer_ownership/2"},{"type":"module","title":"Plausible.Site.Memberships.AcceptInvitation","doc":"Service for accepting invitations, including ownership transfers.\n\nAccepting invitation accounts for the fact that it's possible\nthat accepting user has an existing membership for the site and\nacts permissively to not unnecessarily disrupt the flow while\nalso maintaining integrity of site memberships. This also applies\nto cases where users update their email address between issuing\nthe invitation and accepting it.","ref":"Plausible.Site.Memberships.AcceptInvitation.html"},{"type":"function","title":"Plausible.Site.Memberships.AcceptInvitation.accept_invitation/2","doc":"","ref":"Plausible.Site.Memberships.AcceptInvitation.html#accept_invitation/2"},{"type":"function","title":"Plausible.Site.Memberships.AcceptInvitation.transfer_ownership/2","doc":"","ref":"Plausible.Site.Memberships.AcceptInvitation.html#transfer_ownership/2"},{"type":"module","title":"Plausible.Site.Memberships.CreateInvitation","doc":"Service for inviting new or existing users to a sites, including ownershhip\ntransfers.","ref":"Plausible.Site.Memberships.CreateInvitation.html"},{"type":"function","title":"Plausible.Site.Memberships.CreateInvitation.bulk_create_invitation/5","doc":"","ref":"Plausible.Site.Memberships.CreateInvitation.html#bulk_create_invitation/5"},{"type":"function","title":"Plausible.Site.Memberships.CreateInvitation.bulk_transfer_ownership_direct/2","doc":"","ref":"Plausible.Site.Memberships.CreateInvitation.html#bulk_transfer_ownership_direct/2"},{"type":"function","title":"Plausible.Site.Memberships.CreateInvitation.create_invitation/4","doc":"Invites a new team member to the given site. Returns a\n%Plausible.Auth.Invitation{} struct and sends the invitee an email to accept\nthis invitation.\n\nThe inviter must have enough permissions to invite the new team member,\notherwise this function returns `{:error, :forbidden}`.\n\nIf the new team member role is `:owner`, this function handles the invitation\nas an ownership transfer and requires the inviter to be the owner of the site.","ref":"Plausible.Site.Memberships.CreateInvitation.html#create_invitation/4"},{"type":"type","title":"Plausible.Site.Memberships.CreateInvitation.invite_error/0","doc":"","ref":"Plausible.Site.Memberships.CreateInvitation.html#t:invite_error/0"},{"type":"module","title":"Plausible.Site.Memberships.RejectInvitation","doc":"Service for rejecting invitations.","ref":"Plausible.Site.Memberships.RejectInvitation.html"},{"type":"function","title":"Plausible.Site.Memberships.RejectInvitation.reject_invitation/2","doc":"","ref":"Plausible.Site.Memberships.RejectInvitation.html#reject_invitation/2"},{"type":"module","title":"Plausible.Site.Memberships.RemoveInvitation","doc":"Service for removing invitations.","ref":"Plausible.Site.Memberships.RemoveInvitation.html"},{"type":"function","title":"Plausible.Site.Memberships.RemoveInvitation.remove_invitation/2","doc":"","ref":"Plausible.Site.Memberships.RemoveInvitation.html#remove_invitation/2"},{"type":"module","title":"Plausible.Site.MonthlyReport","doc":"","ref":"Plausible.Site.MonthlyReport.html"},{"type":"function","title":"Plausible.Site.MonthlyReport.add_recipient/2","doc":"","ref":"Plausible.Site.MonthlyReport.html#add_recipient/2"},{"type":"function","title":"Plausible.Site.MonthlyReport.changeset/2","doc":"","ref":"Plausible.Site.MonthlyReport.html#changeset/2"},{"type":"function","title":"Plausible.Site.MonthlyReport.remove_recipient/2","doc":"","ref":"Plausible.Site.MonthlyReport.html#remove_recipient/2"},{"type":"module","title":"Plausible.Site.Removal","doc":"A site deletion service stub.","ref":"Plausible.Site.Removal.html"},{"type":"function","title":"Plausible.Site.Removal.run/1","doc":"","ref":"Plausible.Site.Removal.html#run/1"},{"type":"module","title":"Plausible.Site.SharedLink","doc":"","ref":"Plausible.Site.SharedLink.html"},{"type":"function","title":"Plausible.Site.SharedLink.changeset/2","doc":"","ref":"Plausible.Site.SharedLink.html#changeset/2"},{"type":"type","title":"Plausible.Site.SharedLink.t/0","doc":"","ref":"Plausible.Site.SharedLink.html#t:t/0"},{"type":"module","title":"Plausible.Site.TrafficChangeNotification","doc":"Configuration schema for site-specific traffic change notifications.","ref":"Plausible.Site.TrafficChangeNotification.html"},{"type":"function","title":"Plausible.Site.TrafficChangeNotification.add_recipient/2","doc":"","ref":"Plausible.Site.TrafficChangeNotification.html#add_recipient/2"},{"type":"function","title":"Plausible.Site.TrafficChangeNotification.changeset/2","doc":"","ref":"Plausible.Site.TrafficChangeNotification.html#changeset/2"},{"type":"function","title":"Plausible.Site.TrafficChangeNotification.remove_recipient/2","doc":"","ref":"Plausible.Site.TrafficChangeNotification.html#remove_recipient/2"},{"type":"function","title":"Plausible.Site.TrafficChangeNotification.was_sent/1","doc":"","ref":"Plausible.Site.TrafficChangeNotification.html#was_sent/1"},{"type":"module","title":"Plausible.Site.UserPreference","doc":"User-specific site preferences schema","ref":"Plausible.Site.UserPreference.html"},{"type":"function","title":"Plausible.Site.UserPreference.changeset/3","doc":"","ref":"Plausible.Site.UserPreference.html#changeset/3"},{"type":"macro","title":"Plausible.Site.UserPreference.options/0","doc":"","ref":"Plausible.Site.UserPreference.html#options/0"},{"type":"type","title":"Plausible.Site.UserPreference.t/0","doc":"","ref":"Plausible.Site.UserPreference.html#t:t/0"},{"type":"module","title":"Plausible.Site.WeeklyReport","doc":"","ref":"Plausible.Site.WeeklyReport.html"},{"type":"function","title":"Plausible.Site.WeeklyReport.add_recipient/2","doc":"","ref":"Plausible.Site.WeeklyReport.html#add_recipient/2"},{"type":"function","title":"Plausible.Site.WeeklyReport.changeset/2","doc":"","ref":"Plausible.Site.WeeklyReport.html#changeset/2"},{"type":"function","title":"Plausible.Site.WeeklyReport.remove_recipient/2","doc":"","ref":"Plausible.Site.WeeklyReport.html#remove_recipient/2"},{"type":"module","title":"Plausible.SiteAdmin","doc":"","ref":"Plausible.SiteAdmin.html"},{"type":"function","title":"Plausible.SiteAdmin.before_update/2","doc":"","ref":"Plausible.SiteAdmin.html#before_update/2"},{"type":"function","title":"Plausible.SiteAdmin.create_changeset/2","doc":"","ref":"Plausible.SiteAdmin.html#create_changeset/2"},{"type":"function","title":"Plausible.SiteAdmin.custom_index_query/3","doc":"","ref":"Plausible.SiteAdmin.html#custom_index_query/3"},{"type":"function","title":"Plausible.SiteAdmin.form_fields/1","doc":"","ref":"Plausible.SiteAdmin.html#form_fields/1"},{"type":"function","title":"Plausible.SiteAdmin.get_struct_fields/1","doc":"","ref":"Plausible.SiteAdmin.html#get_struct_fields/1"},{"type":"function","title":"Plausible.SiteAdmin.index/1","doc":"","ref":"Plausible.SiteAdmin.html#index/1"},{"type":"function","title":"Plausible.SiteAdmin.list_actions/1","doc":"","ref":"Plausible.SiteAdmin.html#list_actions/1"},{"type":"function","title":"Plausible.SiteAdmin.ordering/1","doc":"","ref":"Plausible.SiteAdmin.html#ordering/1"},{"type":"function","title":"Plausible.SiteAdmin.search_fields/1","doc":"","ref":"Plausible.SiteAdmin.html#search_fields/1"},{"type":"function","title":"Plausible.SiteAdmin.update_changeset/2","doc":"","ref":"Plausible.SiteAdmin.html#update_changeset/2"},{"type":"module","title":"Plausible.Sites","doc":"Sites context functions.","ref":"Plausible.Sites.html"},{"type":"function","title":"Plausible.Sites.clear_stats_start_date!/1","doc":"","ref":"Plausible.Sites.html#clear_stats_start_date!/1"},{"type":"function","title":"Plausible.Sites.create/2","doc":"","ref":"Plausible.Sites.html#create/2"},{"type":"function","title":"Plausible.Sites.create_shared_link/3","doc":"","ref":"Plausible.Sites.html#create_shared_link/3"},{"type":"function","title":"Plausible.Sites.for_user_query/1","doc":"","ref":"Plausible.Sites.html#for_user_query/1"},{"type":"function","title":"Plausible.Sites.get_by_domain/1","doc":"","ref":"Plausible.Sites.html#get_by_domain/1"},{"type":"function","title":"Plausible.Sites.get_by_domain!/1","doc":"","ref":"Plausible.Sites.html#get_by_domain!/1"},{"type":"function","title":"Plausible.Sites.get_for_user/3","doc":"","ref":"Plausible.Sites.html#get_for_user/3"},{"type":"function","title":"Plausible.Sites.get_for_user!/3","doc":"","ref":"Plausible.Sites.html#get_for_user!/3"},{"type":"function","title":"Plausible.Sites.has_admin_access?/2","doc":"","ref":"Plausible.Sites.html#has_admin_access?/2"},{"type":"function","title":"Plausible.Sites.has_goals?/1","doc":"","ref":"Plausible.Sites.html#has_goals?/1"},{"type":"function","title":"Plausible.Sites.has_stats?/1","doc":"","ref":"Plausible.Sites.html#has_stats?/1"},{"type":"function","title":"Plausible.Sites.is_member?/2","doc":"","ref":"Plausible.Sites.html#is_member?/2"},{"type":"function","title":"Plausible.Sites.list/3","doc":"","ref":"Plausible.Sites.html#list/3"},{"type":"function","title":"Plausible.Sites.list_with_invitations/3","doc":"","ref":"Plausible.Sites.html#list_with_invitations/3"},{"type":"function","title":"Plausible.Sites.locked?/1","doc":"","ref":"Plausible.Sites.html#locked?/1"},{"type":"function","title":"Plausible.Sites.native_stats_start_date/1","doc":"","ref":"Plausible.Sites.html#native_stats_start_date/1"},{"type":"function","title":"Plausible.Sites.owned_site_ids/1","doc":"","ref":"Plausible.Sites.html#owned_site_ids/1"},{"type":"function","title":"Plausible.Sites.owned_sites_count/1","doc":"","ref":"Plausible.Sites.html#owned_sites_count/1"},{"type":"function","title":"Plausible.Sites.owned_sites_domains/1","doc":"","ref":"Plausible.Sites.html#owned_sites_domains/1"},{"type":"function","title":"Plausible.Sites.owned_sites_locked?/1","doc":"","ref":"Plausible.Sites.html#owned_sites_locked?/1"},{"type":"function","title":"Plausible.Sites.role/2","doc":"","ref":"Plausible.Sites.html#role/2"},{"type":"function","title":"Plausible.Sites.set_option/4","doc":"","ref":"Plausible.Sites.html#set_option/4"},{"type":"function","title":"Plausible.Sites.shared_link_url/2","doc":"","ref":"Plausible.Sites.html#shared_link_url/2"},{"type":"function","title":"Plausible.Sites.stats_start_date/1","doc":"Returns the date of the first event of the given site, or `nil` if the site\ndoes not have stats yet.\n\nIf this is the first time the function is called for the site, it queries\nimported stats and Clickhouse, choosing the earliest start date and saves\nit in the sites table.","ref":"Plausible.Sites.html#stats_start_date/1"},{"type":"function","title":"Plausible.Sites.toggle_pin/2","doc":"","ref":"Plausible.Sites.html#toggle_pin/2"},{"type":"function","title":"Plausible.Sites.update_installation_meta!/2","doc":"","ref":"Plausible.Sites.html#update_installation_meta!/2"},{"type":"type","title":"Plausible.Sites.list_opt/0","doc":"","ref":"Plausible.Sites.html#t:list_opt/0"},{"type":"module","title":"Plausible.Stats","doc":"","ref":"Plausible.Stats.html"},{"type":"function","title":"Plausible.Stats.aggregate/3","doc":"","ref":"Plausible.Stats.html#aggregate/3"},{"type":"function","title":"Plausible.Stats.breakdown/4","doc":"","ref":"Plausible.Stats.html#breakdown/4"},{"type":"function","title":"Plausible.Stats.current_visitors/2","doc":"","ref":"Plausible.Stats.html#current_visitors/2"},{"type":"function","title":"Plausible.Stats.filter_suggestions/4","doc":"","ref":"Plausible.Stats.html#filter_suggestions/4"},{"type":"function","title":"Plausible.Stats.funnel/3","doc":"","ref":"Plausible.Stats.html#funnel/3"},{"type":"function","title":"Plausible.Stats.query/2","doc":"","ref":"Plausible.Stats.html#query/2"},{"type":"function","title":"Plausible.Stats.timeseries/3","doc":"","ref":"Plausible.Stats.html#timeseries/3"},{"type":"module","title":"Plausible.Stats.Aggregate","doc":"Builds aggregate results for v1 of our stats API and dashboards.\n\nAvoid adding new logic here - update QueryBuilder etc instead.","ref":"Plausible.Stats.Aggregate.html"},{"type":"function","title":"Plausible.Stats.Aggregate.aggregate/3","doc":"","ref":"Plausible.Stats.Aggregate.html#aggregate/3"},{"type":"module","title":"Plausible.Stats.Base","doc":"","ref":"Plausible.Stats.Base.html"},{"type":"function","title":"Plausible.Stats.Base.base_event_query/2","doc":"","ref":"Plausible.Stats.Base.html#base_event_query/2"},{"type":"function","title":"Plausible.Stats.Base.query_sessions/2","doc":"","ref":"Plausible.Stats.Base.html#query_sessions/2"},{"type":"module","title":"Plausible.Stats.Breakdown","doc":"Builds breakdown results for v1 of our stats API and dashboards.\n\nAvoid adding new logic here - update QueryBuilder etc instead.","ref":"Plausible.Stats.Breakdown.html"},{"type":"function","title":"Plausible.Stats.Breakdown.breakdown/5","doc":"","ref":"Plausible.Stats.Breakdown.html#breakdown/5"},{"type":"function","title":"Plausible.Stats.Breakdown.transform_dimensions/1","doc":"","ref":"Plausible.Stats.Breakdown.html#transform_dimensions/1"},{"type":"module","title":"Plausible.Stats.Clickhouse","doc":"","ref":"Plausible.Stats.Clickhouse.html"},{"type":"function","title":"Plausible.Stats.Clickhouse.current_visitors/1","doc":"","ref":"Plausible.Stats.Clickhouse.html#current_visitors/1"},{"type":"function","title":"Plausible.Stats.Clickhouse.current_visitors_12h/1","doc":"","ref":"Plausible.Stats.Clickhouse.html#current_visitors_12h/1"},{"type":"function","title":"Plausible.Stats.Clickhouse.empty_24h_visitors_hourly_intervals/2","doc":"","ref":"Plausible.Stats.Clickhouse.html#empty_24h_visitors_hourly_intervals/2"},{"type":"function","title":"Plausible.Stats.Clickhouse.has_pageviews?/1","doc":"","ref":"Plausible.Stats.Clickhouse.html#has_pageviews?/1"},{"type":"function","title":"Plausible.Stats.Clickhouse.imported_pageview_count/1","doc":"","ref":"Plausible.Stats.Clickhouse.html#imported_pageview_count/1"},{"type":"function","title":"Plausible.Stats.Clickhouse.imported_pageview_counts/1","doc":"","ref":"Plausible.Stats.Clickhouse.html#imported_pageview_counts/1"},{"type":"function","title":"Plausible.Stats.Clickhouse.last_24h_visitors_hourly_intervals/2","doc":"","ref":"Plausible.Stats.Clickhouse.html#last_24h_visitors_hourly_intervals/2"},{"type":"function","title":"Plausible.Stats.Clickhouse.pageview_start_date_local/1","doc":"","ref":"Plausible.Stats.Clickhouse.html#pageview_start_date_local/1"},{"type":"function","title":"Plausible.Stats.Clickhouse.top_sources_for_spike/4","doc":"","ref":"Plausible.Stats.Clickhouse.html#top_sources_for_spike/4"},{"type":"function","title":"Plausible.Stats.Clickhouse.usage_breakdown/2","doc":"","ref":"Plausible.Stats.Clickhouse.html#usage_breakdown/2"},{"type":"module","title":"Plausible.Stats.Compare","doc":"","ref":"Plausible.Stats.Compare.html"},{"type":"function","title":"Plausible.Stats.Compare.calculate_change/3","doc":"","ref":"Plausible.Stats.Compare.html#calculate_change/3"},{"type":"function","title":"Plausible.Stats.Compare.percent_change/2","doc":"","ref":"Plausible.Stats.Compare.html#percent_change/2"},{"type":"module","title":"Plausible.Stats.Comparisons","doc":"This module provides functions for comparing query periods.\n\nIt allows you to compare a given period with a previous period or with the\nsame period from the previous year. For example, you can compare this month's\nmain graph with last month or with the same month from last year.","ref":"Plausible.Stats.Comparisons.html"},{"type":"function","title":"Plausible.Stats.Comparisons.compare/4","doc":"Generates a comparison query based on the source query and comparison mode.\n\nCurrently only historical periods are supported for comparisons (not `realtime`\nand `30m` periods).\n\nThe mode parameter specifies the type of comparison and can be one of the\nfollowing:\n\n * `\"previous_period\"` - shifts back the query by the same number of days the\n source query has.\n\n * `\"year_over_year\"` - shifts back the query by 1 year.\n\n * `\"custom\"` - compares the query using a custom date range. See options for\n more details.\n\nThe comparison query returned by the function has its end date restricted to\nthe current day. This can be overridden by the `now` option, described below.","ref":"Plausible.Stats.Comparisons.html#compare/4"},{"type":"function","title":"Options - Plausible.Stats.Comparisons.compare/4","doc":"* `:now` - a `NaiveDateTime` struct with the current date and time. This is\n optional and used for testing purposes.\n\n * `:from` - a ISO-8601 date string used when mode is `\"custom\"`.\n\n * `:to` - a ISO-8601 date string used when mode is `\"custom\"`. Must be\n after `from`.\n\n * `:match_day_of_week?` - determines whether the comparison query should be\n adjusted to match the day of the week of the source query. When this option\n is set to true, the comparison query is shifted to start on the same day of\n the week as the source query, rather than on the exact same date. For\n example, if the source query starts on Sunday, January 1st, 2023 and the\n `year_over_year` comparison query is configured to `match_day_of_week?`,\n it will be shifted to start on Sunday, January 2nd, 2022 instead of\n January 1st. Defaults to false.","ref":"Plausible.Stats.Comparisons.html#compare/4-options"},{"type":"type","title":"Plausible.Stats.Comparisons.mode/0","doc":"","ref":"Plausible.Stats.Comparisons.html#t:mode/0"},{"type":"module","title":"Plausible.Stats.CurrentVisitors","doc":"","ref":"Plausible.Stats.CurrentVisitors.html"},{"type":"function","title":"Plausible.Stats.CurrentVisitors.current_visitors/2","doc":"","ref":"Plausible.Stats.CurrentVisitors.html#current_visitors/2"},{"type":"module","title":"Plausible.Stats.CustomProps","doc":"Module for querying user defined 'custom properties'.","ref":"Plausible.Stats.CustomProps.html"},{"type":"function","title":"Plausible.Stats.CustomProps.fetch_prop_names/2","doc":"","ref":"Plausible.Stats.CustomProps.html#fetch_prop_names/2"},{"type":"function","title":"Plausible.Stats.CustomProps.maybe_allowed_props_only/2","doc":"","ref":"Plausible.Stats.CustomProps.html#maybe_allowed_props_only/2"},{"type":"module","title":"Plausible.Stats.DateTimeRange","doc":"Defines a struct similar `Date.Range`, but with `DateTime` instead of `Date`.\n\nThe structs should be created with the `new!/2` function.","ref":"Plausible.Stats.DateTimeRange.html"},{"type":"function","title":"Plausible.Stats.DateTimeRange.new!/2","doc":"","ref":"Plausible.Stats.DateTimeRange.html#new!/2"},{"type":"function","title":"Plausible.Stats.DateTimeRange.new!/3","doc":"Creates a `DateTimeRange` struct from the given `%Date{}` structs.\n\nThe first datetime will become the first date at 00:00:00, and the last datetime\nwill become the last date at 23:59:59. Both dates will be turned into `%DateTime{}`\nstructs in the given timezone.","ref":"Plausible.Stats.DateTimeRange.html#new!/3"},{"type":"function","title":"Plausible.Stats.DateTimeRange.to_date_range/1","doc":"","ref":"Plausible.Stats.DateTimeRange.html#to_date_range/1"},{"type":"type","title":"Plausible.Stats.DateTimeRange.t/0","doc":"","ref":"Plausible.Stats.DateTimeRange.html#t:t/0"},{"type":"module","title":"Plausible.Stats.EmailReport","doc":"This module exposes a `get/2` function that returns a map\nof stats needed for email reports. These stats include:\n\n* Total pageviews\n* Unique visitors\n* Bounce rate\n* A list of Top 5 sources (excluding \"Direct / None\")\n* A list of Top 5 pages\n\nwhere total pageviews, unique visitors, and bounce rate\nalso include the change compared to previous period.","ref":"Plausible.Stats.EmailReport.html"},{"type":"function","title":"Plausible.Stats.EmailReport.get/2","doc":"","ref":"Plausible.Stats.EmailReport.html#get/2"},{"type":"module","title":"Plausible.Stats.FilterSuggestions","doc":"","ref":"Plausible.Stats.FilterSuggestions.html"},{"type":"function","title":"Plausible.Stats.FilterSuggestions.filter_suggestions/4","doc":"","ref":"Plausible.Stats.FilterSuggestions.html#filter_suggestions/4"},{"type":"module","title":"Plausible.Stats.Filters","doc":"A module for parsing filters used in stat queries.","ref":"Plausible.Stats.Filters.html"},{"type":"function","title":"Plausible.Stats.Filters.event_props/0","doc":"","ref":"Plausible.Stats.Filters.html#event_props/0"},{"type":"function","title":"Plausible.Stats.Filters.event_table_visit_props/0","doc":"","ref":"Plausible.Stats.Filters.html#event_table_visit_props/0"},{"type":"function","title":"Plausible.Stats.Filters.parse/1","doc":"Parses different filter formats.\n\nDepending on the format and type of the `filters` argument, returns:\n\n * a decoded list, when `filters` is encoded JSON\n * a parsed filter list, when `filters` is a filter expression string\n * the same list, when `filters` is a map\n\nReturns an empty list when argument type is unexpected (e.g. `nil`).\n\n#","ref":"Plausible.Stats.Filters.html#parse/1"},{"type":"function","title":"Examples: - Plausible.Stats.Filters.parse/1","doc":"iex> Filters.parse(\"{\\\"page\\\":\\\"/blog/**\\\"}\")\n [[:matches, \"event:page\", [\"/blog/**\"]]]\n\n iex> Filters.parse(\"visit:browser!=Chrome\")\n [[:is_not, \"visit:browser\", [\"Chrome\"]]]\n\n iex> Filters.parse(nil)\n []","ref":"Plausible.Stats.Filters.html#parse/1-examples"},{"type":"function","title":"Plausible.Stats.Filters.visit_props/0","doc":"","ref":"Plausible.Stats.Filters.html#visit_props/0"},{"type":"function","title":"Plausible.Stats.Filters.without_prefix/1","doc":"","ref":"Plausible.Stats.Filters.html#without_prefix/1"},{"type":"module","title":"Plausible.Stats.Filters.Utils","doc":"Contains utility functions shared between `LegacyDashboardFilterParser`\nand `StatsAPIFilterParser`.","ref":"Plausible.Stats.Filters.Utils.html"},{"type":"function","title":"Plausible.Stats.Filters.Utils.list_expression?/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#list_expression?/1"},{"type":"function","title":"Plausible.Stats.Filters.Utils.page_regex/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#page_regex/1"},{"type":"function","title":"Plausible.Stats.Filters.Utils.parse_member_list/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#parse_member_list/1"},{"type":"function","title":"Plausible.Stats.Filters.Utils.remove_escape_chars/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#remove_escape_chars/1"},{"type":"function","title":"Plausible.Stats.Filters.Utils.split_goals/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#split_goals/1"},{"type":"function","title":"Plausible.Stats.Filters.Utils.split_goals_query_expressions/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#split_goals_query_expressions/1"},{"type":"function","title":"Plausible.Stats.Filters.Utils.wildcard_expression?/1","doc":"","ref":"Plausible.Stats.Filters.Utils.html#wildcard_expression?/1"},{"type":"module","title":"Plausible.Stats.Funnel","doc":"Module responsible for funnel evaluation, i.e. building and executing\nClickHouse funnel query based on `Plausible.Funnel` definition.","ref":"Plausible.Stats.Funnel.html"},{"type":"function","title":"Plausible.Stats.Funnel.funnel/3","doc":"","ref":"Plausible.Stats.Funnel.html#funnel/3"},{"type":"module","title":"Plausible.Stats.Goal.Revenue","doc":"Revenue specific functions for the stats scope","ref":"Plausible.Stats.Goal.Revenue.html"},{"type":"function","title":"Plausible.Stats.Goal.Revenue.cast_revenue_metrics_to_money/2","doc":"","ref":"Plausible.Stats.Goal.Revenue.html#cast_revenue_metrics_to_money/2"},{"type":"function","title":"Plausible.Stats.Goal.Revenue.get_revenue_tracking_currency/3","doc":"Returns the common currency for the goal filters in a query. If there are no\ngoal filters, multiple currencies or the site owner does not have access to\nrevenue goals, `nil` is returned and revenue metrics are dropped.\n\nAggregating revenue data works only for same currency goals. If the query is\nfiltered by goals with different currencies, for example, one USD and other\nEUR, revenue metrics are dropped.","ref":"Plausible.Stats.Goal.Revenue.html#get_revenue_tracking_currency/3"},{"type":"function","title":"Plausible.Stats.Goal.Revenue.revenue_metrics/0","doc":"","ref":"Plausible.Stats.Goal.Revenue.html#revenue_metrics/0"},{"type":"module","title":"Plausible.Stats.Imported","doc":"","ref":"Plausible.Stats.Imported.html"},{"type":"function","title":"Plausible.Stats.Imported.goals_with_path/0","doc":"","ref":"Plausible.Stats.Imported.html#goals_with_path/0"},{"type":"function","title":"Plausible.Stats.Imported.goals_with_url/0","doc":"","ref":"Plausible.Stats.Imported.html#goals_with_url/0"},{"type":"function","title":"Plausible.Stats.Imported.merge_imported/4","doc":"","ref":"Plausible.Stats.Imported.html#merge_imported/4"},{"type":"function","title":"Plausible.Stats.Imported.merge_imported_city_suggestions/3","doc":"","ref":"Plausible.Stats.Imported.html#merge_imported_city_suggestions/3"},{"type":"function","title":"Plausible.Stats.Imported.merge_imported_country_suggestions/3","doc":"","ref":"Plausible.Stats.Imported.html#merge_imported_country_suggestions/3"},{"type":"function","title":"Plausible.Stats.Imported.merge_imported_filter_suggestions/5","doc":"","ref":"Plausible.Stats.Imported.html#merge_imported_filter_suggestions/5"},{"type":"function","title":"Plausible.Stats.Imported.merge_imported_region_suggestions/3","doc":"","ref":"Plausible.Stats.Imported.html#merge_imported_region_suggestions/3"},{"type":"function","title":"Plausible.Stats.Imported.schema_supports_query?/1","doc":"Returns a boolean indicating whether the combination of filters and\nbreakdown property is possible to query from the imported tables.\n\nUsually, when no filters are used, the imported schema supports the\nquery. There is one exception though - breakdown by a custom property.\nWe are currently importing only two custom properties - `url` and `path.\nBoth these properties can only be used with their special goal filter\n(see `@goals_with_url` and `@goals_with_path`).","ref":"Plausible.Stats.Imported.html#schema_supports_query?/1"},{"type":"function","title":"Plausible.Stats.Imported.total_imported_visitors/2","doc":"","ref":"Plausible.Stats.Imported.html#total_imported_visitors/2"},{"type":"module","title":"Plausible.Stats.Imported.Base","doc":"A module for building the base of an imported stats query","ref":"Plausible.Stats.Imported.Base.html"},{"type":"function","title":"Plausible.Stats.Imported.Base.decide_tables/1","doc":"","ref":"Plausible.Stats.Imported.Base.html#decide_tables/1"},{"type":"function","title":"Plausible.Stats.Imported.Base.property_to_table_mappings/0","doc":"","ref":"Plausible.Stats.Imported.Base.html#property_to_table_mappings/0"},{"type":"function","title":"Plausible.Stats.Imported.Base.query_imported/2","doc":"","ref":"Plausible.Stats.Imported.Base.html#query_imported/2"},{"type":"function","title":"Plausible.Stats.Imported.Base.query_imported/3","doc":"","ref":"Plausible.Stats.Imported.Base.html#query_imported/3"},{"type":"function","title":"Plausible.Stats.Imported.Base.special_goals_for/1","doc":"","ref":"Plausible.Stats.Imported.Base.html#special_goals_for/1"},{"type":"module","title":"Plausible.Stats.Imported.SQL.Builder","doc":"This module is responsible for generating SQL/Ecto expressions\nfor dimensions, filters and metrics used in import table queries","ref":"Plausible.Stats.Imported.SQL.Builder.html"},{"type":"function","title":"Plausible.Stats.Imported.SQL.Builder.group_imported_by/2","doc":"","ref":"Plausible.Stats.Imported.SQL.Builder.html#group_imported_by/2"},{"type":"function","title":"Plausible.Stats.Imported.SQL.Builder.naive_dimension_join/3","doc":"","ref":"Plausible.Stats.Imported.SQL.Builder.html#naive_dimension_join/3"},{"type":"function","title":"Plausible.Stats.Imported.SQL.Builder.select_imported_metrics/2","doc":"","ref":"Plausible.Stats.Imported.SQL.Builder.html#select_imported_metrics/2"},{"type":"function","title":"Plausible.Stats.Imported.SQL.Builder.select_joined_dimensions/2","doc":"","ref":"Plausible.Stats.Imported.SQL.Builder.html#select_joined_dimensions/2"},{"type":"function","title":"Plausible.Stats.Imported.SQL.Builder.select_joined_metrics/2","doc":"","ref":"Plausible.Stats.Imported.SQL.Builder.html#select_joined_metrics/2"},{"type":"module","title":"Plausible.Stats.Interval","doc":"Collection of functions to work with intervals.\n\nThe interval of a query defines the granularity of the data. You can think of\nit as a `GROUP BY` clause. Possible values are `minute`, `hour`, `day`,\n`week`, and `month`.","ref":"Plausible.Stats.Interval.html"},{"type":"function","title":"Plausible.Stats.Interval.default_for_date_range/1","doc":"Returns the suggested interval for the given `DateTimeRange` struct.","ref":"Plausible.Stats.Interval.html#default_for_date_range/1"},{"type":"function","title":"Plausible.Stats.Interval.default_for_period/1","doc":"Returns the suggested interval for the given time period.","ref":"Plausible.Stats.Interval.html#default_for_period/1"},{"type":"function","title":"Plausible.Stats.Interval.list/0","doc":"","ref":"Plausible.Stats.Interval.html#list/0"},{"type":"function","title":"Plausible.Stats.Interval.valid?/1","doc":"","ref":"Plausible.Stats.Interval.html#valid?/1"},{"type":"function","title":"Plausible.Stats.Interval.valid_by_period/1","doc":"","ref":"Plausible.Stats.Interval.html#valid_by_period/1"},{"type":"function","title":"Plausible.Stats.Interval.valid_for_period?/3","doc":"Returns whether the given interval is valid for a time period.\n\nIntervals longer than periods are not supported, e.g. current month stats with\na month interval, or today stats with a week interval.\n\nThere are two dynamic states:\n* `custom` period is only applicable with `month` or `week` intervals,\n if the `opts[:from]` and `opts[:to]` range difference exceeds 12 months\n* `all` period's interval options depend on particular site's `stats_start_date`\n - daily interval is excluded if the all-time range exceeds 12 months","ref":"Plausible.Stats.Interval.html#valid_for_period?/3"},{"type":"type","title":"Plausible.Stats.Interval.opt/0","doc":"","ref":"Plausible.Stats.Interval.html#t:opt/0"},{"type":"type","title":"Plausible.Stats.Interval.opts/0","doc":"","ref":"Plausible.Stats.Interval.html#t:opts/0"},{"type":"type","title":"Plausible.Stats.Interval.t/0","doc":"","ref":"Plausible.Stats.Interval.html#t:t/0"},{"type":"module","title":"Plausible.Stats.JSONSchema","doc":"Module for validating query parameters against JSON schema.\n\nNote that `internal` queries expose some metrics, filter types and other features not\navailable on the public API.","ref":"Plausible.Stats.JSONSchema.html"},{"type":"function","title":"Plausible.Stats.JSONSchema.raw_public_schema/0","doc":"","ref":"Plausible.Stats.JSONSchema.html#raw_public_schema/0"},{"type":"function","title":"Plausible.Stats.JSONSchema.validate/2","doc":"","ref":"Plausible.Stats.JSONSchema.html#validate/2"},{"type":"module","title":"Plausible.Stats.Legacy.Dimensions","doc":"Deprecated module. See QueryParser for list of valid dimensions","ref":"Plausible.Stats.Legacy.Dimensions.html"},{"type":"function","title":"Plausible.Stats.Legacy.Dimensions.valid?/1","doc":"","ref":"Plausible.Stats.Legacy.Dimensions.html#valid?/1"},{"type":"module","title":"Plausible.Stats.Legacy.QueryBuilder","doc":"Module used to parse URL search params to a valid Query, used to power the API for the dashboard.\n@deprecated","ref":"Plausible.Stats.Legacy.QueryBuilder.html"},{"type":"function","title":"Plausible.Stats.Legacy.QueryBuilder.from/3","doc":"","ref":"Plausible.Stats.Legacy.QueryBuilder.html#from/3"},{"type":"function","title":"Plausible.Stats.Legacy.QueryBuilder.parse_order_by/1","doc":"#","ref":"Plausible.Stats.Legacy.QueryBuilder.html#parse_order_by/1"},{"type":"function","title":"Examples: - Plausible.Stats.Legacy.QueryBuilder.parse_order_by/1","doc":"iex> QueryBuilder.parse_order_by(nil)\n []\n\n iex> QueryBuilder.parse_order_by(\"\")\n []\n\n iex> QueryBuilder.parse_order_by(\"0\")\n []\n\n iex> QueryBuilder.parse_order_by(\"[}\")\n []\n\n iex> QueryBuilder.parse_order_by(~s({\"any\":\"object\"}))\n []\n\n iex> QueryBuilder.parse_order_by(~s([[\"visitors\",\"invalid\"]]))\n []\n\n iex> QueryBuilder.parse_order_by(~s([[\"visitors\",\"desc\"]]))\n [{:visitors, :desc}]\n\n iex> QueryBuilder.parse_order_by(~s([[\"visitors\",\"asc\"],[\"visit:source\",\"desc\"]]))\n [{:visitors, :asc}, {\"visit:source\", :desc}]","ref":"Plausible.Stats.Legacy.QueryBuilder.html#parse_order_by/1-examples"},{"type":"module","title":"Plausible.Stats.Metrics","doc":"A module listing all available metrics in Plausible.\n\nUseful for an explicit string to atom conversion.","ref":"Plausible.Stats.Metrics.html"},{"type":"function","title":"Plausible.Stats.Metrics.from_string/1","doc":"","ref":"Plausible.Stats.Metrics.html#from_string/1"},{"type":"function","title":"Plausible.Stats.Metrics.from_string!/1","doc":"","ref":"Plausible.Stats.Metrics.html#from_string!/1"},{"type":"function","title":"Plausible.Stats.Metrics.metric?/1","doc":"","ref":"Plausible.Stats.Metrics.html#metric?/1"},{"type":"module","title":"Plausible.Stats.Query","doc":"","ref":"Plausible.Stats.Query.html"},{"type":"function","title":"Plausible.Stats.Query.add_filter/2","doc":"","ref":"Plausible.Stats.Query.html#add_filter/2"},{"type":"function","title":"Plausible.Stats.Query.build/4","doc":"","ref":"Plausible.Stats.Query.html#build/4"},{"type":"function","title":"Plausible.Stats.Query.ensure_include_imported/2","doc":"","ref":"Plausible.Stats.Query.html#ensure_include_imported/2"},{"type":"function","title":"Plausible.Stats.Query.from/3","doc":"Builds query from old-style params. New code should prefer Query.build","ref":"Plausible.Stats.Query.html#from/3"},{"type":"function","title":"Plausible.Stats.Query.get_filter/2","doc":"","ref":"Plausible.Stats.Query.html#get_filter/2"},{"type":"function","title":"Plausible.Stats.Query.get_filter_by_prefix/2","doc":"","ref":"Plausible.Stats.Query.html#get_filter_by_prefix/2"},{"type":"function","title":"Plausible.Stats.Query.has_event_filters?/1","doc":"","ref":"Plausible.Stats.Query.html#has_event_filters?/1"},{"type":"function","title":"Plausible.Stats.Query.put_experimental_reduced_joins/3","doc":"","ref":"Plausible.Stats.Query.html#put_experimental_reduced_joins/3"},{"type":"function","title":"Plausible.Stats.Query.put_imported_opts/3","doc":"","ref":"Plausible.Stats.Query.html#put_imported_opts/3"},{"type":"function","title":"Plausible.Stats.Query.remove_filters/2","doc":"","ref":"Plausible.Stats.Query.html#remove_filters/2"},{"type":"function","title":"Plausible.Stats.Query.set/2","doc":"","ref":"Plausible.Stats.Query.html#set/2"},{"type":"function","title":"Plausible.Stats.Query.trace/2","doc":"","ref":"Plausible.Stats.Query.html#trace/2"},{"type":"type","title":"Plausible.Stats.Query.t/0","doc":"","ref":"Plausible.Stats.Query.html#t:t/0"},{"type":"module","title":"Plausible.Stats.QueryOptimizer","doc":"Methods to manipulate Query for business logic reasons before building an ecto query.","ref":"Plausible.Stats.QueryOptimizer.html"},{"type":"function","title":"Plausible.Stats.QueryOptimizer.optimize/1","doc":"This module manipulates an existing query, updating it according to business logic.\n\n For example, it:\n 1. Figures out what the right granularity to group by time is\n 2. Adds a missing order_by clause to a query\n 3. Updating \"time\" dimension in order_by to the right granularity","ref":"Plausible.Stats.QueryOptimizer.html#optimize/1"},{"type":"function","title":"Plausible.Stats.QueryOptimizer.split/1","doc":"Splits a query into event and sessions subcomponents as not all metrics can be\nqueried from a single table.\n\nevent:page dimension is treated in a special way, doing a breakdown of visit:entry_page\nfor sessions.","ref":"Plausible.Stats.QueryOptimizer.html#split/1"},{"type":"module","title":"Plausible.Stats.QueryResult","doc":"This struct contains the (JSON-encodable) response for a query and\nis responsible for building it from database query results.\n\nFor the convenience of API docs and consumers, the JSON result\nproduced by Jason.encode(query_result) is ordered.","ref":"Plausible.Stats.QueryResult.html"},{"type":"function","title":"Plausible.Stats.QueryResult.from/3","doc":"","ref":"Plausible.Stats.QueryResult.html#from/3"},{"type":"module","title":"Plausible.Stats.SQL.Expression","doc":"This module is responsible for generating SQL/Ecto expressions\nfor dimensions and metrics used in query SELECT statement.\n\nEach dimension and metric is tagged with with selected_as for easier\nusage down the line.","ref":"Plausible.Stats.SQL.Expression.html"},{"type":"macro","title":"Plausible.Stats.SQL.Expression.event_goal_join/2","doc":"","ref":"Plausible.Stats.SQL.Expression.html#event_goal_join/2"},{"type":"function","title":"Plausible.Stats.SQL.Expression.event_metric/1","doc":"","ref":"Plausible.Stats.SQL.Expression.html#event_metric/1"},{"type":"function","title":"Plausible.Stats.SQL.Expression.select_dimension/5","doc":"","ref":"Plausible.Stats.SQL.Expression.html#select_dimension/5"},{"type":"function","title":"Plausible.Stats.SQL.Expression.session_metric/2","doc":"","ref":"Plausible.Stats.SQL.Expression.html#session_metric/2"},{"type":"module","title":"Plausible.Stats.SQL.Fragments","doc":"Various macros and common SQL fragments used in Stats code.","ref":"Plausible.Stats.SQL.Fragments.html"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.bounce_rate/0","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#bounce_rate/0"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.coalesce_string/2","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#coalesce_string/2"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.get_by_key/3","doc":"Returns value of a key (usually property) under `meta.value` array or similar.\n\nThis macro is used for operating on custom properties.\nCallsites should also check whether key exists first in SQL via `has_key` macro.","ref":"Plausible.Stats.SQL.Fragments.html#get_by_key/3"},{"type":"macro","title":"Examples - Plausible.Stats.SQL.Fragments.get_by_key/3","doc":"`get_by_key(e, :meta, \"some_property_name\")` expands to SQL `meta.value[indexOf(meta.key, \"some_property\")]`","ref":"Plausible.Stats.SQL.Fragments.html#get_by_key/3-examples"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.has_key/3","doc":"Returns whether a key (usually property) exists under `meta.key` array or similar.\n\nThis macro is used for operating on custom properties.","ref":"Plausible.Stats.SQL.Fragments.html#has_key/3"},{"type":"macro","title":"Examples - Plausible.Stats.SQL.Fragments.has_key/3","doc":"`has_key(e, :meta, \"some_property_name\")` expands to SQL `has(meta.key, \"some_property_name\")`","ref":"Plausible.Stats.SQL.Fragments.html#has_key/3-examples"},{"type":"function","title":"Plausible.Stats.SQL.Fragments.meta_key_column/1","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#meta_key_column/1"},{"type":"function","title":"Plausible.Stats.SQL.Fragments.meta_value_column/1","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#meta_value_column/1"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.sample_percent/0","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#sample_percent/0"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.select_join_fields/4","doc":"Macro that helps join two Ecto queries by selecting fields from either one","ref":"Plausible.Stats.SQL.Fragments.html#select_join_fields/4"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.select_merge_as/3","doc":"Convenience Ecto macro for wrapping select_merge where each value gets in turn passed to selected_as.\n\n#","ref":"Plausible.Stats.SQL.Fragments.html#select_merge_as/3"},{"type":"macro","title":"Examples - Plausible.Stats.SQL.Fragments.select_merge_as/3","doc":"iex> select_merge_as(q, [t], %{ foo: t.column }) |> expand_macro_once\n \"select_merge(q, [], ^wrap_alias([t], %{foo: t.column}))\"","ref":"Plausible.Stats.SQL.Fragments.html#select_merge_as/3-examples"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.to_timezone/2","doc":"Converts time or date and time to the specified timezone.\n\nReference: https://clickhouse.com/docs/en/sql-reference/functions/date-time-functions/#totimezone","ref":"Plausible.Stats.SQL.Fragments.html#to_timezone/2"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.total/0","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#total/0"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.uniq/1","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#uniq/1"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.visit_duration/0","doc":"","ref":"Plausible.Stats.SQL.Fragments.html#visit_duration/0"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.weekstart_not_before/2","doc":"Returns the weekstart for `date`. If the weekstart is before the `not_before`\nboundary, `not_before` is returned.","ref":"Plausible.Stats.SQL.Fragments.html#weekstart_not_before/2"},{"type":"macro","title":"Examples - Plausible.Stats.SQL.Fragments.weekstart_not_before/2","doc":"In this pseudo-code example, the fragment returns the weekstart. The\n`not_before` boundary is set to the past Saturday, which is before the\nweekstart, therefore the cap does not apply.\n\n ```\n > this_wednesday = ~D[2022-11-09]\n > past_saturday = ~D[2022-11-05]\n > weekstart_not_before(this_wednesday, past_saturday)\n ~D[2022-11-07]\n ```\n\nIn this other example, the fragment returns Tuesday and not the weekstart.\nThe `not_before` boundary is set to Tuesday, which is past the weekstart,\ntherefore the cap applies.\n\n ```\n > this_wednesday = ~D[2022-11-09]\n > this_tuesday = ~D[2022-11-08]\n > weekstart_not_before(this_wednesday, this_tuesday)\n ~D[2022-11-08]\n ```","ref":"Plausible.Stats.SQL.Fragments.html#weekstart_not_before/2-examples"},{"type":"macro","title":"Plausible.Stats.SQL.Fragments.wrap_alias/2","doc":"Convenience Ecto macro for wrapping a map passed to select_merge_as such that each\nexpression gets wrapped in dynamic and set as selected_as.\n\n#","ref":"Plausible.Stats.SQL.Fragments.html#wrap_alias/2"},{"type":"macro","title":"Examples - Plausible.Stats.SQL.Fragments.wrap_alias/2","doc":"iex> wrap_alias([t], %{ foo: t.column }) |> expand_macro_once\n \"%{foo: dynamic([t], selected_as(t.column, :foo))}\"","ref":"Plausible.Stats.SQL.Fragments.html#wrap_alias/2-examples"},{"type":"module","title":"Plausible.Stats.SQL.SpecialMetrics","doc":"This module defines how special metrics like `conversion_rate` and\n`percentage` are calculated.","ref":"Plausible.Stats.SQL.SpecialMetrics.html"},{"type":"function","title":"Plausible.Stats.SQL.SpecialMetrics.add/3","doc":"","ref":"Plausible.Stats.SQL.SpecialMetrics.html#add/3"},{"type":"function","title":"Plausible.Stats.SQL.SpecialMetrics.maybe_add_global_conversion_rate/3","doc":"","ref":"Plausible.Stats.SQL.SpecialMetrics.html#maybe_add_global_conversion_rate/3"},{"type":"function","title":"Plausible.Stats.SQL.SpecialMetrics.maybe_add_group_conversion_rate/3","doc":"","ref":"Plausible.Stats.SQL.SpecialMetrics.html#maybe_add_group_conversion_rate/3"},{"type":"module","title":"Plausible.Stats.SQL.WhereBuilder","doc":"A module for building am ecto where clause of a query out of a query.","ref":"Plausible.Stats.SQL.WhereBuilder.html"},{"type":"function","title":"Plausible.Stats.SQL.WhereBuilder.build/3","doc":"Builds WHERE clause for a given Query against sessions or events table","ref":"Plausible.Stats.SQL.WhereBuilder.html#build/3"},{"type":"function","title":"Plausible.Stats.SQL.WhereBuilder.build_condition/2","doc":"Builds WHERE clause condition based off of a filter and a custom column name\nUsed for special business logic cases\n\nAccepts nil as the `filter` parameter, in which case the condition is a no-op (WHERE TRUE).","ref":"Plausible.Stats.SQL.WhereBuilder.html#build_condition/2"},{"type":"module","title":"Plausible.Stats.Sampling","doc":"Sampling related functions","ref":"Plausible.Stats.Sampling.html"},{"type":"function","title":"Plausible.Stats.Sampling.add_query_hint/1","doc":"","ref":"Plausible.Stats.Sampling.html#add_query_hint/1"},{"type":"function","title":"Plausible.Stats.Sampling.add_query_hint/2","doc":"","ref":"Plausible.Stats.Sampling.html#add_query_hint/2"},{"type":"function","title":"Plausible.Stats.Sampling.put_threshold/2","doc":"","ref":"Plausible.Stats.Sampling.html#put_threshold/2"},{"type":"module","title":"Plausible.Stats.TableDecider","doc":"This module contains logic for deciding which tables need to be queried given a query\nand metrics, with the purpose of reducing the number of queries and JOINs needed to perform.","ref":"Plausible.Stats.TableDecider.html"},{"type":"function","title":"Plausible.Stats.TableDecider.events_join_sessions?/1","doc":"","ref":"Plausible.Stats.TableDecider.html#events_join_sessions?/1"},{"type":"function","title":"Plausible.Stats.TableDecider.partition_metrics/2","doc":"","ref":"Plausible.Stats.TableDecider.html#partition_metrics/2"},{"type":"module","title":"Plausible.Stats.Time","doc":"Collection of functions to work with time in queries.","ref":"Plausible.Stats.Time.html"},{"type":"function","title":"Plausible.Stats.Time.date_or_weekstart/2","doc":"","ref":"Plausible.Stats.Time.html#date_or_weekstart/2"},{"type":"function","title":"Plausible.Stats.Time.format_datetime/1","doc":"","ref":"Plausible.Stats.Time.html#format_datetime/1"},{"type":"function","title":"Plausible.Stats.Time.time_dimension/1","doc":"Returns list of time bucket labels for the given query.","ref":"Plausible.Stats.Time.html#time_dimension/1"},{"type":"function","title":"Plausible.Stats.Time.time_labels/1","doc":"","ref":"Plausible.Stats.Time.html#time_labels/1"},{"type":"function","title":"Plausible.Stats.Time.utc_boundaries/2","doc":"","ref":"Plausible.Stats.Time.html#utc_boundaries/2"},{"type":"module","title":"Plausible.Stats.Timeseries","doc":"Builds timeseries results for v1 of our stats API and dashboards.\n\nAvoid adding new logic here - update QueryBuilder etc instead.","ref":"Plausible.Stats.Timeseries.html"},{"type":"function","title":"Plausible.Stats.Timeseries.timeseries/3","doc":"","ref":"Plausible.Stats.Timeseries.html#timeseries/3"},{"type":"module","title":"Plausible.Stats.Util","doc":"Utilities for modifying stat results","ref":"Plausible.Stats.Util.html"},{"type":"function","title":"Plausible.Stats.Util.keep_requested_metrics/2","doc":"Sometimes we need to manually add metrics in order to calculate the value for\nother metrics. E.g:\n\n* `__internal_visits` is fetched when querying bounce rate, visit duration,\n or views_per_visit, as it is needed to calculate these from imported data.\n\n* `visitors` metric might be added manually via `maybe_add_visitors_metric/1`,\n in order to be able to calculate conversion rate.\n\nThis function can be used for stripping those metrics from a breakdown (list),\nor an aggregate (map) result. We do not want to return metrics that we're not\nrequested.","ref":"Plausible.Stats.Util.html#keep_requested_metrics/2"},{"type":"function","title":"Plausible.Stats.Util.maybe_add_visitors_metric/1","doc":"This function adds the `visitors` metric into the list of\ngiven metrics if it's not already there and if it is needed\nfor any of the other metrics to be calculated.","ref":"Plausible.Stats.Util.html#maybe_add_visitors_metric/1"},{"type":"function","title":"Plausible.Stats.Util.shortname/2","doc":"","ref":"Plausible.Stats.Util.html#shortname/2"},{"type":"module","title":"Plausible.Test.Support.HTML","doc":"Floki wrappers to help make assertions about HTML/DOM structures","ref":"Plausible.Test.Support.HTML.html"},{"type":"function","title":"Plausible.Test.Support.HTML.class_of_element/2","doc":"","ref":"Plausible.Test.Support.HTML.html#class_of_element/2"},{"type":"function","title":"Plausible.Test.Support.HTML.element_exists?/2","doc":"","ref":"Plausible.Test.Support.HTML.html#element_exists?/2"},{"type":"function","title":"Plausible.Test.Support.HTML.find/2","doc":"","ref":"Plausible.Test.Support.HTML.html#find/2"},{"type":"function","title":"Plausible.Test.Support.HTML.form_exists?/2","doc":"","ref":"Plausible.Test.Support.HTML.html#form_exists?/2"},{"type":"function","title":"Plausible.Test.Support.HTML.name_of/1","doc":"","ref":"Plausible.Test.Support.HTML.html#name_of/1"},{"type":"function","title":"Plausible.Test.Support.HTML.submit_button/2","doc":"","ref":"Plausible.Test.Support.HTML.html#submit_button/2"},{"type":"function","title":"Plausible.Test.Support.HTML.text/1","doc":"","ref":"Plausible.Test.Support.HTML.html#text/1"},{"type":"function","title":"Plausible.Test.Support.HTML.text_of_attr/2","doc":"","ref":"Plausible.Test.Support.HTML.html#text_of_attr/2"},{"type":"function","title":"Plausible.Test.Support.HTML.text_of_attr/3","doc":"","ref":"Plausible.Test.Support.HTML.html#text_of_attr/3"},{"type":"function","title":"Plausible.Test.Support.HTML.text_of_element/2","doc":"","ref":"Plausible.Test.Support.HTML.html#text_of_element/2"},{"type":"module","title":"Plausible.Test.Support.HTTPMocker","doc":"Currently only supports post request, it's a drop-in replacement\nfor our exvcr usage that wasn't ever needed (e.g. we had no way to\nre-record the cassettes anyway).","ref":"Plausible.Test.Support.HTTPMocker.html"},{"type":"module","title":"Plausible.TestUtils","doc":"","ref":"Plausible.TestUtils.html"},{"type":"function","title":"Plausible.TestUtils.await_clickhouse_count/2","doc":"","ref":"Plausible.TestUtils.html#await_clickhouse_count/2"},{"type":"function","title":"Plausible.TestUtils.create_api_key/1","doc":"","ref":"Plausible.TestUtils.html#create_api_key/1"},{"type":"function","title":"Plausible.TestUtils.create_legacy_site_import/1","doc":"","ref":"Plausible.TestUtils.html#create_legacy_site_import/1"},{"type":"function","title":"Plausible.TestUtils.create_new_site/1","doc":"","ref":"Plausible.TestUtils.html#create_new_site/1"},{"type":"function","title":"Plausible.TestUtils.create_pageviews/1","doc":"","ref":"Plausible.TestUtils.html#create_pageviews/1"},{"type":"function","title":"Plausible.TestUtils.create_site/1","doc":"","ref":"Plausible.TestUtils.html#create_site/1"},{"type":"function","title":"Plausible.TestUtils.create_site_import/1","doc":"","ref":"Plausible.TestUtils.html#create_site_import/1"},{"type":"function","title":"Plausible.TestUtils.create_user/1","doc":"","ref":"Plausible.TestUtils.html#create_user/1"},{"type":"function","title":"Plausible.TestUtils.ensure_minio/0","doc":"","ref":"Plausible.TestUtils.html#ensure_minio/0"},{"type":"function","title":"Plausible.TestUtils.eventually/3","doc":"","ref":"Plausible.TestUtils.html#eventually/3"},{"type":"function","title":"Plausible.TestUtils.generate_usage_for/3","doc":"","ref":"Plausible.TestUtils.html#generate_usage_for/3"},{"type":"function","title":"Plausible.TestUtils.init_session/1","doc":"","ref":"Plausible.TestUtils.html#init_session/1"},{"type":"function","title":"Plausible.TestUtils.log_in/1","doc":"","ref":"Plausible.TestUtils.html#log_in/1"},{"type":"function","title":"Plausible.TestUtils.maybe_fake_minio/1","doc":"","ref":"Plausible.TestUtils.html#maybe_fake_minio/1"},{"type":"function","title":"Plausible.TestUtils.minio_running?/0","doc":"","ref":"Plausible.TestUtils.html#minio_running?/0"},{"type":"macro","title":"Plausible.TestUtils.patch_env/2","doc":"","ref":"Plausible.TestUtils.html#patch_env/2"},{"type":"function","title":"Plausible.TestUtils.populate_stats/1","doc":"","ref":"Plausible.TestUtils.html#populate_stats/1"},{"type":"function","title":"Plausible.TestUtils.populate_stats/2","doc":"","ref":"Plausible.TestUtils.html#populate_stats/2"},{"type":"function","title":"Plausible.TestUtils.populate_stats/3","doc":"","ref":"Plausible.TestUtils.html#populate_stats/3"},{"type":"function","title":"Plausible.TestUtils.random_ip/0","doc":"","ref":"Plausible.TestUtils.html#random_ip/0"},{"type":"function","title":"Plausible.TestUtils.relative_time/1","doc":"","ref":"Plausible.TestUtils.html#relative_time/1"},{"type":"macro","title":"Plausible.TestUtils.setup_patch_env/2","doc":"","ref":"Plausible.TestUtils.html#setup_patch_env/2"},{"type":"function","title":"Plausible.TestUtils.to_naive_truncate/1","doc":"","ref":"Plausible.TestUtils.html#to_naive_truncate/1"},{"type":"function","title":"Plausible.TestUtils.use_api_key/1","doc":"","ref":"Plausible.TestUtils.html#use_api_key/1"},{"type":"module","title":"Plausible.Themes","doc":"","ref":"Plausible.Themes.html"},{"type":"function","title":"Plausible.Themes.options/0","doc":"","ref":"Plausible.Themes.html#options/0"},{"type":"module","title":"Plausible.Timezones","doc":"","ref":"Plausible.Timezones.html"},{"type":"function","title":"Plausible.Timezones.options/1","doc":"","ref":"Plausible.Timezones.html#options/1"},{"type":"function","title":"Plausible.Timezones.to_date_in_timezone/2","doc":"","ref":"Plausible.Timezones.html#to_date_in_timezone/2"},{"type":"function","title":"Plausible.Timezones.to_datetime_in_timezone/2","doc":"","ref":"Plausible.Timezones.html#to_datetime_in_timezone/2"},{"type":"module","title":"Plausible.Users","doc":"User context","ref":"Plausible.Users.html"},{"type":"function","title":"Plausible.Users.accept_traffic_until/1","doc":"","ref":"Plausible.Users.html#accept_traffic_until/1"},{"type":"function","title":"Plausible.Users.allow_next_upgrade_override/1","doc":"","ref":"Plausible.Users.html#allow_next_upgrade_override/1"},{"type":"function","title":"Plausible.Users.bump_last_seen/2","doc":"","ref":"Plausible.Users.html#bump_last_seen/2"},{"type":"function","title":"Plausible.Users.has_email_code?/1","doc":"","ref":"Plausible.Users.html#has_email_code?/1"},{"type":"function","title":"Plausible.Users.last_subscription_join_query/0","doc":"","ref":"Plausible.Users.html#last_subscription_join_query/0"},{"type":"function","title":"Plausible.Users.maybe_reset_next_upgrade_override/1","doc":"","ref":"Plausible.Users.html#maybe_reset_next_upgrade_override/1"},{"type":"function","title":"Plausible.Users.on_trial?/1","doc":"","ref":"Plausible.Users.html#on_trial?/1"},{"type":"function","title":"Plausible.Users.trial_days_left/1","doc":"","ref":"Plausible.Users.html#trial_days_left/1"},{"type":"function","title":"Plausible.Users.update_accept_traffic_until/1","doc":"","ref":"Plausible.Users.html#update_accept_traffic_until/1"},{"type":"function","title":"Plausible.Users.with_subscription/1","doc":"","ref":"Plausible.Users.html#with_subscription/1"},{"type":"module","title":"Plausible.Verification","doc":"Module defining the user-agent used for site verification.","ref":"Plausible.Verification.html"},{"type":"function","title":"Plausible.Verification.user_agent/0","doc":"","ref":"Plausible.Verification.html#user_agent/0"},{"type":"behaviour","title":"Plausible.Verification.Check","doc":"Behaviour to be implemented by specific site verification checks.\n`report_progress_as()` doesn't necessarily reflect the actual check description,\nit serves as a user-facing message grouping mechanism, to prevent frequent message flashing when checks rotate often.\nEach check operates on `state()` and is expected to return it, optionally modified, by all means.\n`perform_safe/1` is used to guarantee no exceptions are thrown by faulty implementations, not to interrupt LiveView.","ref":"Plausible.Verification.Check.html"},{"type":"callback","title":"Plausible.Verification.Check.perform/1","doc":"","ref":"Plausible.Verification.Check.html#c:perform/1"},{"type":"callback","title":"Plausible.Verification.Check.report_progress_as/0","doc":"","ref":"Plausible.Verification.Check.html#c:report_progress_as/0"},{"type":"type","title":"Plausible.Verification.Check.state/0","doc":"","ref":"Plausible.Verification.Check.html#t:state/0"},{"type":"module","title":"Plausible.Verification.Checks","doc":"Checks that are performed during site verification.\nEach module defined in `@checks` implements the `Plausible.Verification.Check` behaviour.\nChecks are normally run asynchronously, except when synchronous execution is optionally required\nfor tests. Slowdowns can be optionally added, the user doesn't benefit from running the checks too quickly.\n\nIn async execution, each check notifies the caller by sending a message to it.","ref":"Plausible.Verification.Checks.html"},{"type":"function","title":"Plausible.Verification.Checks.interpret_diagnostics/1","doc":"","ref":"Plausible.Verification.Checks.html#interpret_diagnostics/1"},{"type":"function","title":"Plausible.Verification.Checks.run/3","doc":"","ref":"Plausible.Verification.Checks.html#run/3"},{"type":"module","title":"Plausible.Verification.Checks.CSP","doc":"Scans the Content Security Policy header to ensure that the Plausible domain is allowed.\nSee `Plausible.Verification.Checks` for the execution sequence.","ref":"Plausible.Verification.Checks.CSP.html"},{"type":"function","title":"Plausible.Verification.Checks.CSP.perform_safe/1","doc":"","ref":"Plausible.Verification.Checks.CSP.html#perform_safe/1"},{"type":"module","title":"Plausible.Verification.Checks.FetchBody","doc":"Fetches the body of the site and extracts the HTML document, if available, for\nfurther processing.\nSee `Plausible.Verification.Checks` for the execution sequence.","ref":"Plausible.Verification.Checks.FetchBody.html"},{"type":"function","title":"Plausible.Verification.Checks.FetchBody.perform_safe/1","doc":"","ref":"Plausible.Verification.Checks.FetchBody.html#perform_safe/1"},{"type":"module","title":"Plausible.Verification.Checks.Installation","doc":"Calls the browserless.io service (local instance can be spawned with `make browserless`)\nand runs verification/verify_plausible_installed.js via the [function API](https://docs.browserless.io/HTTP-APIs/function).\n\nThe successful execution assumes the following JSON payload:\n - `data.plausibleInstalled` - boolean indicating whether the `plausible()` window function was found\n - `data.callbackStatus` - integer. 202 indicates that the server acknowledged the test event.\n\nThe test event ingestion is discarded based on user-agent, see: `Plausible.Verification.user_agent/0`","ref":"Plausible.Verification.Checks.Installation.html"},{"type":"function","title":"Plausible.Verification.Checks.Installation.perform_safe/1","doc":"","ref":"Plausible.Verification.Checks.Installation.html#perform_safe/1"},{"type":"module","title":"Plausible.Verification.Checks.ScanBody","doc":"Naive way of detecting GTM and WordPress powered sites.","ref":"Plausible.Verification.Checks.ScanBody.html"},{"type":"function","title":"Plausible.Verification.Checks.ScanBody.perform_safe/1","doc":"","ref":"Plausible.Verification.Checks.ScanBody.html#perform_safe/1"},{"type":"module","title":"Plausible.Verification.Checks.Snippet","doc":"The check looks for Plausible snippets and tries to address the common\nintegration issues, such as bad placement, data-domain typos, unknown \nattributes frequently added by performance optimization plugins, etc.","ref":"Plausible.Verification.Checks.Snippet.html"},{"type":"function","title":"Plausible.Verification.Checks.Snippet.perform_safe/1","doc":"","ref":"Plausible.Verification.Checks.Snippet.html#perform_safe/1"},{"type":"module","title":"Plausible.Verification.Checks.SnippetCacheBust","doc":"A naive way of trying to figure out whether the latest site contents \nis wrapped with some CDN/caching layer.\nIn case no snippets were found, we'll try to bust the cache by appending a random query parameter\nand re-run `Plausible.Verification.Checks.FetchBody` and `Plausible.Verification.Checks.Snippet` checks.\nIf the result is different this time, we'll assume cache likely.","ref":"Plausible.Verification.Checks.SnippetCacheBust.html"},{"type":"function","title":"Plausible.Verification.Checks.SnippetCacheBust.perform_safe/1","doc":"","ref":"Plausible.Verification.Checks.SnippetCacheBust.html#perform_safe/1"},{"type":"module","title":"Plausible.Verification.Diagnostics","doc":"Module responsible for translating diagnostics to user-friendly errors and recommendations.","ref":"Plausible.Verification.Diagnostics.html"},{"type":"function","title":"Plausible.Verification.Diagnostics.interpret/2","doc":"","ref":"Plausible.Verification.Diagnostics.html#interpret/2"},{"type":"type","title":"Plausible.Verification.Diagnostics.t/0","doc":"","ref":"Plausible.Verification.Diagnostics.html#t:t/0"},{"type":"module","title":"Plausible.Verification.Diagnostics.Result","doc":"Diagnostics interpretation result.","ref":"Plausible.Verification.Diagnostics.Result.html"},{"type":"type","title":"Plausible.Verification.Diagnostics.Result.t/0","doc":"","ref":"Plausible.Verification.Diagnostics.Result.html#t:t/0"},{"type":"module","title":"Plausible.Verification.Errors","doc":"A go-to definition of all verification errors","ref":"Plausible.Verification.Errors.html"},{"type":"function","title":"Plausible.Verification.Errors.all/0","doc":"","ref":"Plausible.Verification.Errors.html#all/0"},{"type":"module","title":"Plausible.Verification.State","doc":"The struct and interface describing the state of the site verification process.\nAssigns are meant to be used to communicate between checks, while diagnostics\nare later on interpreted (translated into user-friendly messages and recommendations)\nvia `Plausible.Verification.Diagnostics` module.","ref":"Plausible.Verification.State.html"},{"type":"function","title":"Plausible.Verification.State.assign/2","doc":"","ref":"Plausible.Verification.State.html#assign/2"},{"type":"function","title":"Plausible.Verification.State.put_diagnostics/2","doc":"","ref":"Plausible.Verification.State.html#put_diagnostics/2"},{"type":"type","title":"Plausible.Verification.State.t/0","doc":"","ref":"Plausible.Verification.State.html#t:t/0"},{"type":"module","title":"Plausible.Verification.URL","doc":"Busting some caches by appending ?plausible_verification=12345 to it.","ref":"Plausible.Verification.URL.html"},{"type":"function","title":"Plausible.Verification.URL.bust_url/1","doc":"","ref":"Plausible.Verification.URL.html#bust_url/1"},{"type":"module","title":"Plausible.Workers.AcceptTrafficUntil","doc":"A worker meant to be run once a day that sends out e-mail notifications to site\nowners assuming:\n - their sites still receive traffic (i.e. have stats for yesterday)\n - `site.accept_traffic_until` is approaching either tomorrow or exactly in 7 days\n\nUsers having no sites or sites that receive no traffic, won't be notified.\nWe make a tiny effort here to make sure we send the same notification at most once a day.","ref":"Plausible.Workers.AcceptTrafficUntil.html"},{"type":"function","title":"Plausible.Workers.AcceptTrafficUntil.dry_run/1","doc":"","ref":"Plausible.Workers.AcceptTrafficUntil.html#dry_run/1"},{"type":"module","title":"Plausible.Workers.CheckUsage","doc":"","ref":"Plausible.Workers.CheckUsage.html"},{"type":"function","title":"Plausible.Workers.CheckUsage.check_enterprise_subscriber/2","doc":"","ref":"Plausible.Workers.CheckUsage.html#check_enterprise_subscriber/2"},{"type":"macro","title":"Plausible.Workers.CheckUsage.day_of_month/1","doc":"","ref":"Plausible.Workers.CheckUsage.html#day_of_month/1"},{"type":"macro","title":"Plausible.Workers.CheckUsage.last_day_of_month/1","doc":"","ref":"Plausible.Workers.CheckUsage.html#last_day_of_month/1"},{"type":"macro","title":"Plausible.Workers.CheckUsage.least/2","doc":"","ref":"Plausible.Workers.CheckUsage.html#least/2"},{"type":"function","title":"Plausible.Workers.CheckUsage.maybe_remove_grace_period/2","doc":"","ref":"Plausible.Workers.CheckUsage.html#maybe_remove_grace_period/2"},{"type":"macro","title":"Plausible.Workers.CheckUsage.yesterday/0","doc":"","ref":"Plausible.Workers.CheckUsage.html#yesterday/0"},{"type":"module","title":"Plausible.Workers.CleanInvitations","doc":"","ref":"Plausible.Workers.CleanInvitations.html"},{"type":"module","title":"Plausible.Workers.CleanUserSessions","doc":"Job removing expired user sessions. A grace period is applied.","ref":"Plausible.Workers.CleanUserSessions.html"},{"type":"function","title":"Plausible.Workers.CleanUserSessions.grace_period_duration/0","doc":"","ref":"Plausible.Workers.CleanUserSessions.html#grace_period_duration/0"},{"type":"module","title":"Plausible.Workers.ClickhouseCleanSites","doc":"Cleans deleted site data from ClickHouse asynchronously.\n\nWe batch up data deletions from ClickHouse as deleting a single site is\njust as expensive as deleting many.","ref":"Plausible.Workers.ClickhouseCleanSites.html"},{"type":"function","title":"Plausible.Workers.ClickhouseCleanSites.get_deleted_sites_with_clickhouse_data/0","doc":"","ref":"Plausible.Workers.ClickhouseCleanSites.html#get_deleted_sites_with_clickhouse_data/0"},{"type":"function","title":"Plausible.Workers.ClickhouseCleanSites.perform/1","doc":"","ref":"Plausible.Workers.ClickhouseCleanSites.html#perform/1"},{"type":"module","title":"Plausible.Workers.ExpireDomainChangeTransitions","doc":"Periodic worker that expires domain change transition period.\nOld domains are frozen for a given time, so users can still access them\nbefore redeploying their scripts and integrations.","ref":"Plausible.Workers.ExpireDomainChangeTransitions.html"},{"type":"module","title":"Plausible.Workers.ExportAnalytics","doc":"Worker for running CSV export jobs. Supports S3 and local storage.\nTo avoid blocking the queue, a timeout of 15 minutes is enforced.","ref":"Plausible.Workers.ExportAnalytics.html"},{"type":"function","title":"Plausible.Workers.ExportAnalytics.base_query/1","doc":"This base query filters export jobs for a site","ref":"Plausible.Workers.ExportAnalytics.html#base_query/1"},{"type":"module","title":"Plausible.Workers.ImportAnalytics","doc":"Worker for running analytics import jobs.","ref":"Plausible.Workers.ImportAnalytics.html"},{"type":"function","title":"Plausible.Workers.ImportAnalytics.import_complete/1","doc":"","ref":"Plausible.Workers.ImportAnalytics.html#import_complete/1"},{"type":"function","title":"Plausible.Workers.ImportAnalytics.import_fail/2","doc":"","ref":"Plausible.Workers.ImportAnalytics.html#import_fail/2"},{"type":"function","title":"Plausible.Workers.ImportAnalytics.import_fail_transient/1","doc":"","ref":"Plausible.Workers.ImportAnalytics.html#import_fail_transient/1"},{"type":"module","title":"Plausible.Workers.LocalImportAnalyticsCleaner","doc":"Worker for cleaning local files left after analytics import jobs.","ref":"Plausible.Workers.LocalImportAnalyticsCleaner.html"},{"type":"module","title":"Plausible.Workers.LockSites","doc":"","ref":"Plausible.Workers.LockSites.html"},{"type":"module","title":"Plausible.Workers.NotifyAnnualRenewal","doc":"","ref":"Plausible.Workers.NotifyAnnualRenewal.html"},{"type":"function","title":"Plausible.Workers.NotifyAnnualRenewal.perform/1","doc":"Sends a notification at most 7 days and at least 1 day before the renewal of an annual subscription","ref":"Plausible.Workers.NotifyAnnualRenewal.html#perform/1"},{"type":"module","title":"Plausible.Workers.NotifyExportedAnalytics","doc":"This worker delivers emails for successful and failed exports","ref":"Plausible.Workers.NotifyExportedAnalytics.html"},{"type":"module","title":"Plausible.Workers.RotateSalts","doc":"","ref":"Plausible.Workers.RotateSalts.html"},{"type":"module","title":"Plausible.Workers.ScheduleEmailReports","doc":"","ref":"Plausible.Workers.ScheduleEmailReports.html"},{"type":"function","title":"Plausible.Workers.ScheduleEmailReports.first_of_month_9am/1","doc":"","ref":"Plausible.Workers.ScheduleEmailReports.html#first_of_month_9am/1"},{"type":"function","title":"Plausible.Workers.ScheduleEmailReports.monday_9am/1","doc":"","ref":"Plausible.Workers.ScheduleEmailReports.html#monday_9am/1"},{"type":"function","title":"Plausible.Workers.ScheduleEmailReports.perform/1","doc":"Email reports should be sent on Monday at 9am according to the timezone\nof a site. This job runs every day at midnight to ensure that all sites\nhave a scheduled job for email reports.","ref":"Plausible.Workers.ScheduleEmailReports.html#perform/1"},{"type":"module","title":"Plausible.Workers.SendCheckStatsEmails","doc":"","ref":"Plausible.Workers.SendCheckStatsEmails.html"},{"type":"module","title":"Plausible.Workers.SendEmailReport","doc":"","ref":"Plausible.Workers.SendEmailReport.html"},{"type":"module","title":"Plausible.Workers.SendSiteSetupEmails","doc":"","ref":"Plausible.Workers.SendSiteSetupEmails.html"},{"type":"module","title":"Plausible.Workers.SendTrialNotifications","doc":"","ref":"Plausible.Workers.SendTrialNotifications.html"},{"type":"module","title":"Plausible.Workers.TrafficChangeNotifier","doc":"Oban service sending out traffic drop/spike notifications","ref":"Plausible.Workers.TrafficChangeNotifier.html"},{"type":"module","title":"PlausibleWeb","doc":"","ref":"PlausibleWeb.html"},{"type":"macro","title":"PlausibleWeb.__using__/1","doc":"When used, dispatch to the appropriate controller/view/etc.","ref":"PlausibleWeb.html#__using__/1"},{"type":"function","title":"PlausibleWeb.channel/0","doc":"","ref":"PlausibleWeb.html#channel/0"},{"type":"function","title":"PlausibleWeb.controller/0","doc":"","ref":"PlausibleWeb.html#controller/0"},{"type":"function","title":"PlausibleWeb.live_view/1","doc":"","ref":"PlausibleWeb.html#live_view/1"},{"type":"function","title":"PlausibleWeb.open_api_schema/0","doc":"","ref":"PlausibleWeb.html#open_api_schema/0"},{"type":"function","title":"PlausibleWeb.plugins_api_controller/0","doc":"","ref":"PlausibleWeb.html#plugins_api_controller/0"},{"type":"function","title":"PlausibleWeb.plugins_api_view/0","doc":"","ref":"PlausibleWeb.html#plugins_api_view/0"},{"type":"function","title":"PlausibleWeb.router/0","doc":"","ref":"PlausibleWeb.html#router/0"},{"type":"function","title":"PlausibleWeb.view/0","doc":"","ref":"PlausibleWeb.html#view/0"},{"type":"module","title":"PlausibleWeb.AdminController","doc":"","ref":"PlausibleWeb.AdminController.html"},{"type":"function","title":"PlausibleWeb.AdminController.current_plan/2","doc":"","ref":"PlausibleWeb.AdminController.html#current_plan/2"},{"type":"function","title":"PlausibleWeb.AdminController.usage/2","doc":"","ref":"PlausibleWeb.AdminController.html#usage/2"},{"type":"module","title":"PlausibleWeb.Api.ExternalController","doc":"","ref":"PlausibleWeb.Api.ExternalController.html"},{"type":"function","title":"PlausibleWeb.Api.ExternalController.error/2","doc":"","ref":"PlausibleWeb.Api.ExternalController.html#error/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalController.event/2","doc":"","ref":"PlausibleWeb.Api.ExternalController.html#event/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalController.health/2","doc":"","ref":"PlausibleWeb.Api.ExternalController.html#health/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalController.info/2","doc":"","ref":"PlausibleWeb.Api.ExternalController.html#info/2"},{"type":"module","title":"PlausibleWeb.Api.ExternalSitesController","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.create_site/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#create_site/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.delete_goal/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#delete_goal/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.delete_site/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#delete_site/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.find_or_create_goal/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#find_or_create_goal/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.find_or_create_shared_link/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#find_or_create_shared_link/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.get_site/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#get_site/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.goals_index/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#goals_index/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.index/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#index/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalSitesController.update_site/2","doc":"","ref":"PlausibleWeb.Api.ExternalSitesController.html#update_site/2"},{"type":"module","title":"PlausibleWeb.Api.ExternalStatsController","doc":"","ref":"PlausibleWeb.Api.ExternalStatsController.html"},{"type":"function","title":"PlausibleWeb.Api.ExternalStatsController.aggregate/2","doc":"","ref":"PlausibleWeb.Api.ExternalStatsController.html#aggregate/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalStatsController.breakdown/2","doc":"","ref":"PlausibleWeb.Api.ExternalStatsController.html#breakdown/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalStatsController.realtime_visitors/2","doc":"","ref":"PlausibleWeb.Api.ExternalStatsController.html#realtime_visitors/2"},{"type":"function","title":"PlausibleWeb.Api.ExternalStatsController.timeseries/2","doc":"","ref":"PlausibleWeb.Api.ExternalStatsController.html#timeseries/2"},{"type":"module","title":"PlausibleWeb.Api.Helpers","doc":"","ref":"PlausibleWeb.Api.Helpers.html"},{"type":"function","title":"PlausibleWeb.Api.Helpers.bad_request/2","doc":"","ref":"PlausibleWeb.Api.Helpers.html#bad_request/2"},{"type":"function","title":"PlausibleWeb.Api.Helpers.not_found/2","doc":"","ref":"PlausibleWeb.Api.Helpers.html#not_found/2"},{"type":"function","title":"PlausibleWeb.Api.Helpers.payment_required/2","doc":"","ref":"PlausibleWeb.Api.Helpers.html#payment_required/2"},{"type":"function","title":"PlausibleWeb.Api.Helpers.too_many_requests/2","doc":"","ref":"PlausibleWeb.Api.Helpers.html#too_many_requests/2"},{"type":"function","title":"PlausibleWeb.Api.Helpers.unauthorized/2","doc":"","ref":"PlausibleWeb.Api.Helpers.html#unauthorized/2"},{"type":"module","title":"PlausibleWeb.Api.InternalController","doc":"","ref":"PlausibleWeb.Api.InternalController.html"},{"type":"function","title":"PlausibleWeb.Api.InternalController.disable_feature/2","doc":"","ref":"PlausibleWeb.Api.InternalController.html#disable_feature/2"},{"type":"function","title":"PlausibleWeb.Api.InternalController.sites/2","doc":"","ref":"PlausibleWeb.Api.InternalController.html#sites/2"},{"type":"module","title":"PlausibleWeb.Api.PaddleController","doc":"","ref":"PlausibleWeb.Api.PaddleController.html"},{"type":"function","title":"PlausibleWeb.Api.PaddleController.currency/2","doc":"","ref":"PlausibleWeb.Api.PaddleController.html#currency/2"},{"type":"function","title":"PlausibleWeb.Api.PaddleController.verify_signature/2","doc":"","ref":"PlausibleWeb.Api.PaddleController.html#verify_signature/2"},{"type":"function","title":"PlausibleWeb.Api.PaddleController.webhook/2","doc":"","ref":"PlausibleWeb.Api.PaddleController.html#webhook/2"},{"type":"module","title":"PlausibleWeb.Api.StatsController","doc":"","ref":"PlausibleWeb.Api.StatsController.html"},{"type":"function","title":"PlausibleWeb.Api.StatsController.all_custom_prop_values/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#all_custom_prop_values/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.browser_versions/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#browser_versions/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.browsers/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#browsers/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.build_intervals/4","doc":"","ref":"PlausibleWeb.Api.StatsController.html#build_intervals/4"},{"type":"function","title":"PlausibleWeb.Api.StatsController.cities/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#cities/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.conversions/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#conversions/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.countries/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#countries/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.current_visitors/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#current_visitors/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.custom_prop_values/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#custom_prop_values/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.entry_pages/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#entry_pages/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.exit_pages/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#exit_pages/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.filter_suggestions/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#filter_suggestions/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.format_money/1","doc":"","ref":"PlausibleWeb.Api.StatsController.html#format_money/1"},{"type":"function","title":"PlausibleWeb.Api.StatsController.format_revenue_metric/1","doc":"","ref":"PlausibleWeb.Api.StatsController.html#format_revenue_metric/1"},{"type":"function","title":"PlausibleWeb.Api.StatsController.funnel/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#funnel/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.main_graph/2","doc":"Returns a time-series based on given parameters.","ref":"PlausibleWeb.Api.StatsController.html#main_graph/2"},{"type":"function","title":"Parameters - PlausibleWeb.Api.StatsController.main_graph/2","doc":"This API accepts the following parameters:\n\n * `period` - x-axis of the graph, e.g. `12mo`, `day`, `custom`.\n\n * `metric` - y-axis of the graph, e.g. `visits`, `visitors`, `pageviews`.\n See the Stats API [\"Metrics\"](https://plausible.io/docs/stats-api#metrics)\n section for more details. Defaults to `visitors`.\n\n * `interval` - granularity of the time-series data. You can think of it as\n a `GROUP BY` clause. Possible values are `minute`, `hour`, `date`, `week`,\n and `month`. The default depends on the `period` parameter. Check\n `Plausible.Query.from/2` for each default.\n\n * `filters` - optional filters to drill down data. See the Stats API\n [\"Filtering\"](https://plausible.io/docs/stats-api#filtering) section for\n more details.\n\n * `with_imported` - boolean indicating whether to include Google Analytics\n imported data or not. Defaults to `false`.\n\nFull example:\n```elixir\n%{\n \"from\" => \"2021-09-06\",\n \"interval\" => \"month\",\n \"metric\" => \"visitors\",\n \"period\" => \"custom\",\n \"to\" => \"2021-12-13\"\n}\n```","ref":"PlausibleWeb.Api.StatsController.html#main_graph/2-parameters"},{"type":"function","title":"Response - PlausibleWeb.Api.StatsController.main_graph/2","doc":"Returns a map with the following keys:\n\n * `plot` - list of values for the requested metric representing the y-axis\n of the graph.\n\n * `labels` - list of date times representing the x-axis of the graph.\n\n * `present_index` - index of the element representing the current date in\n `labels` and `plot` lists.\n\n * `interval` - the interval used for querying.\n\n * `includes_imported` - boolean indicating whether imported data\n was queried or not.\n\n * `imports_exist` - boolean indicating whether there are any completed\n imports for a given site or not.\n\n * `full_intervals` - map of dates indicating whether the interval has been\n cut off by the requested date range or not. For example, if looking at a\n month week-by-week, some weeks may be cut off by the month boundaries.\n It's useful to adjust the graph display slightly in case the interval is\n not 'full' so that the user understands why the numbers might be lower for\n those partial periods.\n\nFull example:\n```elixir\n%{\n \"full_intervals\" => %{\n \"2021-09-01\" => false,\n \"2021-10-01\" => true,\n \"2021-11-01\" => true,\n \"2021-12-01\" => false\n },\n \"imports_exist\" => false,\n \"interval\" => \"month\",\n \"labels\" => [\"2021-09-01\", \"2021-10-01\", \"2021-11-01\", \"2021-12-01\"],\n \"plot\" => [0, 0, 0, 0],\n \"present_index\" => nil,\n \"includes_imported\" => false\n}\n```","ref":"PlausibleWeb.Api.StatsController.html#main_graph/2-response"},{"type":"function","title":"PlausibleWeb.Api.StatsController.operating_system_versions/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#operating_system_versions/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.operating_systems/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#operating_systems/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.pages/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#pages/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.put_combined_name_with_version/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#put_combined_name_with_version/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.referrer_drilldown/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#referrer_drilldown/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.referrers/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#referrers/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.regions/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#regions/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.screen_sizes/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#screen_sizes/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.sources/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#sources/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.top_stats/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#top_stats/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.utm_campaigns/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#utm_campaigns/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.utm_contents/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#utm_contents/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.utm_mediums/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#utm_mediums/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.utm_sources/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#utm_sources/2"},{"type":"function","title":"PlausibleWeb.Api.StatsController.utm_terms/2","doc":"","ref":"PlausibleWeb.Api.StatsController.html#utm_terms/2"},{"type":"module","title":"PlausibleWeb.AuthController","doc":"","ref":"PlausibleWeb.AuthController.html"},{"type":"function","title":"PlausibleWeb.AuthController.activate/2","doc":"","ref":"PlausibleWeb.AuthController.html#activate/2"},{"type":"function","title":"PlausibleWeb.AuthController.activate_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#activate_form/2"},{"type":"function","title":"PlausibleWeb.AuthController.cancel_update_email/2","doc":"","ref":"PlausibleWeb.AuthController.html#cancel_update_email/2"},{"type":"function","title":"PlausibleWeb.AuthController.create_api_key/2","doc":"","ref":"PlausibleWeb.AuthController.html#create_api_key/2"},{"type":"function","title":"PlausibleWeb.AuthController.delete_api_key/2","doc":"","ref":"PlausibleWeb.AuthController.html#delete_api_key/2"},{"type":"function","title":"PlausibleWeb.AuthController.delete_me/2","doc":"","ref":"PlausibleWeb.AuthController.html#delete_me/2"},{"type":"function","title":"PlausibleWeb.AuthController.disable_2fa/2","doc":"","ref":"PlausibleWeb.AuthController.html#disable_2fa/2"},{"type":"function","title":"PlausibleWeb.AuthController.generate_2fa_recovery_codes/2","doc":"","ref":"PlausibleWeb.AuthController.html#generate_2fa_recovery_codes/2"},{"type":"function","title":"PlausibleWeb.AuthController.google_auth_callback/2","doc":"","ref":"PlausibleWeb.AuthController.html#google_auth_callback/2"},{"type":"function","title":"PlausibleWeb.AuthController.initiate_2fa_setup/2","doc":"","ref":"PlausibleWeb.AuthController.html#initiate_2fa_setup/2"},{"type":"function","title":"PlausibleWeb.AuthController.login/2","doc":"","ref":"PlausibleWeb.AuthController.html#login/2"},{"type":"function","title":"PlausibleWeb.AuthController.login_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#login_form/2"},{"type":"function","title":"PlausibleWeb.AuthController.logout/2","doc":"","ref":"PlausibleWeb.AuthController.html#logout/2"},{"type":"function","title":"PlausibleWeb.AuthController.new_api_key/2","doc":"","ref":"PlausibleWeb.AuthController.html#new_api_key/2"},{"type":"function","title":"PlausibleWeb.AuthController.password_reset/2","doc":"","ref":"PlausibleWeb.AuthController.html#password_reset/2"},{"type":"function","title":"PlausibleWeb.AuthController.password_reset_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#password_reset_form/2"},{"type":"function","title":"PlausibleWeb.AuthController.password_reset_request/2","doc":"","ref":"PlausibleWeb.AuthController.html#password_reset_request/2"},{"type":"function","title":"PlausibleWeb.AuthController.password_reset_request_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#password_reset_request_form/2"},{"type":"function","title":"PlausibleWeb.AuthController.request_activation_code/2","doc":"","ref":"PlausibleWeb.AuthController.html#request_activation_code/2"},{"type":"function","title":"PlausibleWeb.AuthController.save_settings/2","doc":"","ref":"PlausibleWeb.AuthController.html#save_settings/2"},{"type":"function","title":"PlausibleWeb.AuthController.update_email/2","doc":"","ref":"PlausibleWeb.AuthController.html#update_email/2"},{"type":"function","title":"PlausibleWeb.AuthController.user_settings/2","doc":"","ref":"PlausibleWeb.AuthController.html#user_settings/2"},{"type":"function","title":"PlausibleWeb.AuthController.verify_2fa/2","doc":"","ref":"PlausibleWeb.AuthController.html#verify_2fa/2"},{"type":"function","title":"PlausibleWeb.AuthController.verify_2fa_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#verify_2fa_form/2"},{"type":"function","title":"PlausibleWeb.AuthController.verify_2fa_recovery_code/2","doc":"","ref":"PlausibleWeb.AuthController.html#verify_2fa_recovery_code/2"},{"type":"function","title":"PlausibleWeb.AuthController.verify_2fa_recovery_code_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#verify_2fa_recovery_code_form/2"},{"type":"function","title":"PlausibleWeb.AuthController.verify_2fa_setup/2","doc":"","ref":"PlausibleWeb.AuthController.html#verify_2fa_setup/2"},{"type":"function","title":"PlausibleWeb.AuthController.verify_2fa_setup_form/2","doc":"","ref":"PlausibleWeb.AuthController.html#verify_2fa_setup_form/2"},{"type":"module","title":"PlausibleWeb.AuthPlug","doc":"Plug for populating conn assigns with user data\non the basis of authenticated session token.\n\nMust be kept in sync with `PlausibleWeb.Live.AuthContext`.","ref":"PlausibleWeb.AuthPlug.html"},{"type":"function","title":"PlausibleWeb.AuthPlug.call/2","doc":"","ref":"PlausibleWeb.AuthPlug.html#call/2"},{"type":"function","title":"PlausibleWeb.AuthPlug.init/1","doc":"","ref":"PlausibleWeb.AuthPlug.html#init/1"},{"type":"module","title":"PlausibleWeb.AuthView","doc":"","ref":"PlausibleWeb.AuthView.html"},{"type":"function","title":"PlausibleWeb.AuthView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.AuthView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.AuthView.activate.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#activate.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.delimit_integer/1","doc":"","ref":"PlausibleWeb.AuthView.html#delimit_integer/1"},{"type":"function","title":"PlausibleWeb.AuthView.format_invoices/1","doc":"","ref":"PlausibleWeb.AuthView.html#format_invoices/1"},{"type":"function","title":"PlausibleWeb.AuthView.generate_2fa_recovery_codes.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#generate_2fa_recovery_codes.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.initiate_2fa_setup.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#initiate_2fa_setup.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.login_form.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#login_form.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.new_api_key.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#new_api_key.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.password_reset_form.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#password_reset_form.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.password_reset_request_form.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#password_reset_request_form.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.password_reset_request_success.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#password_reset_request_success.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.present_subscription_status/1","doc":"","ref":"PlausibleWeb.AuthView.html#present_subscription_status/1"},{"type":"function","title":"PlausibleWeb.AuthView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.AuthView.html#render/2"},{"type":"function","title":"PlausibleWeb.AuthView.subscription_colors/1","doc":"","ref":"PlausibleWeb.AuthView.html#subscription_colors/1"},{"type":"function","title":"PlausibleWeb.AuthView.subscription_interval/1","doc":"","ref":"PlausibleWeb.AuthView.html#subscription_interval/1"},{"type":"function","title":"PlausibleWeb.AuthView.subscription_quota/2","doc":"","ref":"PlausibleWeb.AuthView.html#subscription_quota/2"},{"type":"function","title":"PlausibleWeb.AuthView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.AuthView.html#template_not_found/2"},{"type":"function","title":"PlausibleWeb.AuthView.user_settings.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#user_settings.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.verify_2fa.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#verify_2fa.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.verify_2fa_recovery_code.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#verify_2fa_recovery_code.html/1"},{"type":"function","title":"PlausibleWeb.AuthView.verify_2fa_setup.html/1","doc":"","ref":"PlausibleWeb.AuthView.html#verify_2fa_setup.html/1"},{"type":"module","title":"PlausibleWeb.AuthorizeSiteAccess","doc":"","ref":"PlausibleWeb.AuthorizeSiteAccess.html"},{"type":"function","title":"PlausibleWeb.AuthorizeSiteAccess.call/2","doc":"","ref":"PlausibleWeb.AuthorizeSiteAccess.html#call/2"},{"type":"function","title":"PlausibleWeb.AuthorizeSiteAccess.init/1","doc":"","ref":"PlausibleWeb.AuthorizeSiteAccess.html#init/1"},{"type":"module","title":"PlausibleWeb.AvatarController","doc":"This module proxies requests to BASE_URL/avatar/:hash to www.gravatar.com/avatar/:hash.\n\nThe purpose is to make use of Gravatar's convenient avatar service without exposing information\nthat could be used for tracking the Plausible user. Compared to requesting the Gravatar directly\nfrom the browser, this proxy module protects the Plausible user from disclosing to Gravatar:\n1. The client IP address\n2. User-Agent\n3. Referer header which can be used to track which site the user is visiting (i.e. plausible.io or self-hosted URL)\n\nThe downside is the added latency from the request having to go through the Plausible server, rather than contacting the\nlocal CDN server operated by Gravatar's service.","ref":"PlausibleWeb.AvatarController.html"},{"type":"function","title":"PlausibleWeb.AvatarController.avatar/2","doc":"","ref":"PlausibleWeb.AvatarController.html#avatar/2"},{"type":"module","title":"PlausibleWeb.BillingController","doc":"","ref":"PlausibleWeb.BillingController.html"},{"type":"function","title":"PlausibleWeb.BillingController.change_plan/2","doc":"","ref":"PlausibleWeb.BillingController.html#change_plan/2"},{"type":"function","title":"PlausibleWeb.BillingController.change_plan_preview/2","doc":"","ref":"PlausibleWeb.BillingController.html#change_plan_preview/2"},{"type":"function","title":"PlausibleWeb.BillingController.choose_plan/2","doc":"","ref":"PlausibleWeb.BillingController.html#choose_plan/2"},{"type":"function","title":"PlausibleWeb.BillingController.ping_subscription/2","doc":"","ref":"PlausibleWeb.BillingController.html#ping_subscription/2"},{"type":"function","title":"PlausibleWeb.BillingController.upgrade_success/2","doc":"","ref":"PlausibleWeb.BillingController.html#upgrade_success/2"},{"type":"function","title":"PlausibleWeb.BillingController.upgrade_to_enterprise_plan/2","doc":"","ref":"PlausibleWeb.BillingController.html#upgrade_to_enterprise_plan/2"},{"type":"module","title":"PlausibleWeb.BillingView","doc":"","ref":"PlausibleWeb.BillingView.html"},{"type":"function","title":"PlausibleWeb.BillingView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.BillingView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.BillingView.change_enterprise_plan_contact_us.html/1","doc":"","ref":"PlausibleWeb.BillingView.html#change_enterprise_plan_contact_us.html/1"},{"type":"function","title":"PlausibleWeb.BillingView.change_plan_preview.html/1","doc":"","ref":"PlausibleWeb.BillingView.html#change_plan_preview.html/1"},{"type":"function","title":"PlausibleWeb.BillingView.choose_plan.html/1","doc":"","ref":"PlausibleWeb.BillingView.html#choose_plan.html/1"},{"type":"function","title":"PlausibleWeb.BillingView.present_currency/1","doc":"","ref":"PlausibleWeb.BillingView.html#present_currency/1"},{"type":"function","title":"PlausibleWeb.BillingView.present_date/1","doc":"","ref":"PlausibleWeb.BillingView.html#present_date/1"},{"type":"function","title":"PlausibleWeb.BillingView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.BillingView.html#render/2"},{"type":"function","title":"PlausibleWeb.BillingView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.BillingView.html#template_not_found/2"},{"type":"function","title":"PlausibleWeb.BillingView.upgrade_success.html/1","doc":"","ref":"PlausibleWeb.BillingView.html#upgrade_success.html/1"},{"type":"function","title":"PlausibleWeb.BillingView.upgrade_to_enterprise_plan.html/1","doc":"","ref":"PlausibleWeb.BillingView.html#upgrade_to_enterprise_plan.html/1"},{"type":"module","title":"PlausibleWeb.Captcha","doc":"","ref":"PlausibleWeb.Captcha.html"},{"type":"function","title":"PlausibleWeb.Captcha.enabled?/0","doc":"","ref":"PlausibleWeb.Captcha.html#enabled?/0"},{"type":"function","title":"PlausibleWeb.Captcha.sitekey/0","doc":"","ref":"PlausibleWeb.Captcha.html#sitekey/0"},{"type":"function","title":"PlausibleWeb.Captcha.verify/1","doc":"","ref":"PlausibleWeb.Captcha.html#verify/1"},{"type":"module","title":"PlausibleWeb.Components.Billing.PlanBenefits","doc":"This module exposes functions for rendering and returning plan\nbenefits for Growth, Business, and Enterprise plans.","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html"},{"type":"function","title":"PlausibleWeb.Components.Billing.PlanBenefits.for_business/2","doc":"Returns Business benefits for the given Business plan.\n\nA second argument is also required - list of Growth benefits. This\nis because we don't want to list the same benefits in both Growth\nand Business. Everything in Growth is also included in Business.","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#for_business/2"},{"type":"function","title":"PlausibleWeb.Components.Billing.PlanBenefits.for_enterprise/1","doc":"This function only takes a list of business benefits. Since all\nlimits and features of enterprise plans are configurable, we can\nsay on the upgrade page that enterprise plans include everything\nin Business.","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#for_enterprise/1"},{"type":"function","title":"PlausibleWeb.Components.Billing.PlanBenefits.for_growth/1","doc":"This function takes a growth plan and returns a list representing\nthe different benefits a user gets when subscribing to this plan.","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#for_growth/1"},{"type":"function","title":"PlausibleWeb.Components.Billing.PlanBenefits.render/1","doc":"This function takes a list of benefits returned by either one of:\n\n* `for_growth/1`\n* `for_business/2`\n* `for_enterprise/1`.\n\nand renders them as HTML.\n\nThe benefits in the given list can be either strings or functions\nreturning a Phoenix component. This allows, for example, to render\nlinks within the plan benefit text.","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Billing.PlanBenefits.render/1","doc":"* `benefits` (`:list`) (required)\n* `class` (`:string`) - Defaults to `nil`.","ref":"PlausibleWeb.Components.Billing.PlanBenefits.html#render/1-attributes"},{"type":"module","title":"PlausibleWeb.Components.FirstDashboardLaunchBanner","doc":"A banner that appears on the first dashboard launch","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html"},{"type":"function","title":"PlausibleWeb.Components.FirstDashboardLaunchBanner.render/1","doc":"","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.FirstDashboardLaunchBanner.render/1","doc":"* `site` (`Plausible.Site`) (required)","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#render/1-attributes"},{"type":"function","title":"PlausibleWeb.Components.FirstDashboardLaunchBanner.set/1","doc":"","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#set/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.FirstDashboardLaunchBanner.set/1","doc":"* `site` (`Plausible.Site`) (required)","ref":"PlausibleWeb.Components.FirstDashboardLaunchBanner.html#set/1-attributes"},{"type":"module","title":"PlausibleWeb.Components.FlowProgress","doc":"Component for provisioning/registration flows displaying\nprogress status. See `PlausibleWeb.Flows` for the list of\nflow definitions.","ref":"PlausibleWeb.Components.FlowProgress.html"},{"type":"function","title":"PlausibleWeb.Components.FlowProgress.render/1","doc":"","ref":"PlausibleWeb.Components.FlowProgress.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.FlowProgress.render/1","doc":"* `flow` (`:string`) (required) - Must be one of `\"register\"`, `\"invitation\"`, `\"domain_change\"`, `\"review\"`, or `\"provisioning\"`.\n* `current_step` (`:string`) (required) - Must be one of `\"Register\"`, `\"Activate account\"`, `\"Add site info\"`, `\"Install Plausible\"`, `\"Verify installation\"`, or `\"Set up new domain\"`.","ref":"PlausibleWeb.Components.FlowProgress.html#render/1-attributes"},{"type":"module","title":"PlausibleWeb.Components.Generic","doc":"Generic reusable components","ref":"PlausibleWeb.Components.Generic.html"},{"type":"function","title":"PlausibleWeb.Components.Generic.button/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#button/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.button/1","doc":"* `type` (`:string`) - Defaults to `\"button\"`.\n* `theme` (`:string`) - Defaults to `\"primary\"`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* `disabled` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#button/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.button/1","doc":"* `inner_block`","ref":"PlausibleWeb.Components.Generic.html#button/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.button_link/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#button_link/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.button_link/1","doc":"* `href` (`:string`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.\n* `theme` (`:string`) - Defaults to `\"primary\"`.\n* `disabled` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#button_link/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.button_link/1","doc":"* `inner_block`","ref":"PlausibleWeb.Components.Generic.html#button_link/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.docs_info/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#docs_info/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.docs_info/1","doc":"* `slug` (`:string`) (required)","ref":"PlausibleWeb.Components.Generic.html#docs_info/1-attributes"},{"type":"function","title":"PlausibleWeb.Components.Generic.dropdown/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#dropdown/1"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.dropdown/1","doc":"* `button` (required) - Accepts attributes:\n\n * `class` (`:string`)\n* `panel` (required) - Accepts attributes:\n\n * `class` (`:string`)","ref":"PlausibleWeb.Components.Generic.html#dropdown/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.dropdown_link/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#dropdown_link/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.dropdown_link/1","doc":"* `href` (`:string`) (required)\n* `new_tab` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#dropdown_link/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.dropdown_link/1","doc":"* `inner_block` (required)","ref":"PlausibleWeb.Components.Generic.html#dropdown_link/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.dynamic_icon/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#dynamic_icon/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.dynamic_icon/1","doc":"* `name` (`:atom`) (required)\n* `outline` (`:boolean`) - Defaults to `true`.\n* `solid` (`:boolean`) - Defaults to `false`.\n* `mini` (`:boolean`) - Defaults to `false`.\n* Global attributes are accepted. Supports all globals plus: `[\"fill\", \"stroke\", \"stroke-width\"]`.","ref":"PlausibleWeb.Components.Generic.html#dynamic_icon/1-attributes"},{"type":"function","title":"PlausibleWeb.Components.Generic.focus_box/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#focus_box/1"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.focus_box/1","doc":"* `title`\n* `subtitle`\n* `inner_block` (required)\n* `footer`","ref":"PlausibleWeb.Components.Generic.html#focus_box/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.focus_list/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#focus_list/1"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.focus_list/1","doc":"* `item` (required)","ref":"PlausibleWeb.Components.Generic.html#focus_list/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.notice/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#notice/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.notice/1","doc":"* `title` (`:any`) - Defaults to `nil`.\n* `size` (`:atom`) - Defaults to `:sm`.\n* `theme` (`:atom`) - Defaults to `:yellow`.\n* `dismissable_id` (`:any`) - Defaults to `nil`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#notice/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.notice/1","doc":"* `inner_block`","ref":"PlausibleWeb.Components.Generic.html#notice/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.spinner/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#spinner/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.spinner/1","doc":"* `class` (`:any`) - Defaults to `\"\"`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#spinner/1-attributes"},{"type":"function","title":"PlausibleWeb.Components.Generic.styled_link/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#styled_link/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.styled_link/1","doc":"* `id` (`:any`) - Defaults to `nil`.\n* `href` (`:string`) - Defaults to `\"#\"`.\n* `new_tab` (`:boolean`) - Defaults to `false`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* `method` (`:string`) - Defaults to `\"get\"`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#styled_link/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.styled_link/1","doc":"* `inner_block`","ref":"PlausibleWeb.Components.Generic.html#styled_link/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.tooltip/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#tooltip/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.tooltip/1","doc":"* `wrapper_class` (`:any`) - Defaults to `\"\"`.\n* `class` (`:any`) - Defaults to `\"\"`.\n* `icon?` (`:boolean`) - Defaults to `true`.\n* `sticky?` (`:boolean`) - Defaults to `true`.\n* `position` (`:string`) - Defaults to `\"bottom-10 margin-x-auto left-10 right-10\"`.","ref":"PlausibleWeb.Components.Generic.html#tooltip/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.tooltip/1","doc":"* `inner_block` (required)\n* `tooltip_content` (required)","ref":"PlausibleWeb.Components.Generic.html#tooltip/1-slots"},{"type":"function","title":"PlausibleWeb.Components.Generic.unstyled_link/1","doc":"","ref":"PlausibleWeb.Components.Generic.html#unstyled_link/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Generic.unstyled_link/1","doc":"* `href` (`:string`) (required)\n* `new_tab` (`:boolean`) - Defaults to `false`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* `id` (`:any`) - Defaults to `nil`.\n* `method` (`:string`) - Defaults to `\"get\"`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Components.Generic.html#unstyled_link/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Generic.unstyled_link/1","doc":"* `inner_block`","ref":"PlausibleWeb.Components.Generic.html#unstyled_link/1-slots"},{"type":"module","title":"PlausibleWeb.Components.Google","doc":"Google-related components","ref":"PlausibleWeb.Components.Google.html"},{"type":"function","title":"PlausibleWeb.Components.Google.button/1","doc":"","ref":"PlausibleWeb.Components.Google.html#button/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Google.button/1","doc":"* `to` (`:string`) (required)\n* `id` (`:string`) (required)","ref":"PlausibleWeb.Components.Google.html#button/1-attributes"},{"type":"function","title":"PlausibleWeb.Components.Google.logo/1","doc":"","ref":"PlausibleWeb.Components.Google.html#logo/1"},{"type":"module","title":"PlausibleWeb.Components.Settings","doc":"An umbrella module for the Integrations settings section","ref":"PlausibleWeb.Components.Settings.html"},{"type":"function","title":"PlausibleWeb.Components.Settings.settings_search_console/1","doc":"","ref":"PlausibleWeb.Components.Settings.html#settings_search_console/1"},{"type":"module","title":"PlausibleWeb.Components.Site.Feature","doc":"Phoenix Component for rendering a user-facing feature toggle\ncapable of flipping booleans in `Plausible.Site` via the `toggle_feature` controller action.","ref":"PlausibleWeb.Components.Site.Feature.html"},{"type":"function","title":"PlausibleWeb.Components.Site.Feature.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Components.Site.Feature.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Components.Site.Feature.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Components.Site.Feature.html#render/2"},{"type":"function","title":"PlausibleWeb.Components.Site.Feature.target/4","doc":"","ref":"PlausibleWeb.Components.Site.Feature.html#target/4"},{"type":"function","title":"PlausibleWeb.Components.Site.Feature.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Components.Site.Feature.html#template_not_found/2"},{"type":"function","title":"PlausibleWeb.Components.Site.Feature.toggle/1","doc":"","ref":"PlausibleWeb.Components.Site.Feature.html#toggle/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.Site.Feature.toggle/1","doc":"* `site` (`Plausible.Site`) (required)\n* `feature_mod` (`:atom`) (required) - Must be one of `Plausible.Billing.Feature.Goals`, `Plausible.Billing.Feature.StatsAPI`, `Plausible.Billing.Feature.Props`, `Plausible.Billing.Feature.Funnels`, or `Plausible.Billing.Feature.RevenueGoals`.\n* `conn` (`Plug.Conn`) (required)","ref":"PlausibleWeb.Components.Site.Feature.html#toggle/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.Site.Feature.toggle/1","doc":"* `inner_block`","ref":"PlausibleWeb.Components.Site.Feature.html#toggle/1-slots"},{"type":"module","title":"PlausibleWeb.Components.TwoFactor","doc":"Reusable components specific to 2FA","ref":"PlausibleWeb.Components.TwoFactor.html"},{"type":"function","title":"PlausibleWeb.Components.TwoFactor.modal/1","doc":"","ref":"PlausibleWeb.Components.TwoFactor.html#modal/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.TwoFactor.modal/1","doc":"* `id` (`:string`) (required)\n* `state_param` (`:string`) (required)\n* `form_data` (`:any`) (required)\n* `form_target` (`:string`) (required)\n* `onsubmit` (`:string`) - Defaults to `nil`.\n* `title` (`:string`) (required)","ref":"PlausibleWeb.Components.TwoFactor.html#modal/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Components.TwoFactor.modal/1","doc":"* `icon` (required)\n* `inner_block` (required)\n* `buttons` (required)","ref":"PlausibleWeb.Components.TwoFactor.html#modal/1-slots"},{"type":"function","title":"PlausibleWeb.Components.TwoFactor.qr_code/1","doc":"","ref":"PlausibleWeb.Components.TwoFactor.html#qr_code/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.TwoFactor.qr_code/1","doc":"* `text` (`:string`) (required)\n* `scale` (`:integer`) - Defaults to `4`.","ref":"PlausibleWeb.Components.TwoFactor.html#qr_code/1-attributes"},{"type":"function","title":"PlausibleWeb.Components.TwoFactor.verify_2fa_input/1","doc":"","ref":"PlausibleWeb.Components.TwoFactor.html#verify_2fa_input/1"},{"type":"function","title":"Attributes - PlausibleWeb.Components.TwoFactor.verify_2fa_input/1","doc":"* `id` (`:string`) - Defaults to `\"verify-button\"`.\n* `form` (`:any`) (required)\n* `field` (`:any`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.","ref":"PlausibleWeb.Components.TwoFactor.html#verify_2fa_input/1-attributes"},{"type":"module","title":"PlausibleWeb.ConnCase","doc":"This module defines the test case to be used by\ntests that require setting up a connection.\n\nSuch tests rely on `Phoenix.ConnTest` and also\nimport other functionality to make it easier\nto build common data structures and query the data layer.\n\nFinally, if the test case interacts with the database,\nit cannot be async. For this reason, every test runs\ninside a transaction which is reset at the beginning\nof the test unless the test case is marked as async.","ref":"PlausibleWeb.ConnCase.html"},{"type":"module","title":"PlausibleWeb.ControllerHelpers","doc":"","ref":"PlausibleWeb.ControllerHelpers.html"},{"type":"function","title":"PlausibleWeb.ControllerHelpers.debug_metadata/1","doc":"","ref":"PlausibleWeb.ControllerHelpers.html#debug_metadata/1"},{"type":"function","title":"PlausibleWeb.ControllerHelpers.render_error/2","doc":"","ref":"PlausibleWeb.ControllerHelpers.html#render_error/2"},{"type":"function","title":"PlausibleWeb.ControllerHelpers.render_error/3","doc":"","ref":"PlausibleWeb.ControllerHelpers.html#render_error/3"},{"type":"module","title":"PlausibleWeb.Controllers.API.Revenue","doc":"Revenue specific functions for the API scope","ref":"PlausibleWeb.Controllers.API.Revenue.html"},{"type":"function","title":"PlausibleWeb.Controllers.API.Revenue.format_money/1","doc":"","ref":"PlausibleWeb.Controllers.API.Revenue.html#format_money/1"},{"type":"function","title":"PlausibleWeb.Controllers.API.Revenue.format_revenue_metric/1","doc":"","ref":"PlausibleWeb.Controllers.API.Revenue.html#format_revenue_metric/1"},{"type":"module","title":"PlausibleWeb.DebugController","doc":"","ref":"PlausibleWeb.DebugController.html"},{"type":"function","title":"PlausibleWeb.DebugController.clickhouse/2","doc":"","ref":"PlausibleWeb.DebugController.html#clickhouse/2"},{"type":"module","title":"PlausibleWeb.DebugView","doc":"","ref":"PlausibleWeb.DebugView.html"},{"type":"function","title":"PlausibleWeb.DebugView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.DebugView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.DebugView.clickhouse.html/1","doc":"","ref":"PlausibleWeb.DebugView.html#clickhouse.html/1"},{"type":"function","title":"PlausibleWeb.DebugView.controller_name/1","doc":"","ref":"PlausibleWeb.DebugView.html#controller_name/1"},{"type":"function","title":"PlausibleWeb.DebugView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.DebugView.html#render/2"},{"type":"function","title":"PlausibleWeb.DebugView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.DebugView.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Dogfood","doc":"Plausible tracking itself functions","ref":"PlausibleWeb.Dogfood.html"},{"type":"function","title":"PlausibleWeb.Dogfood.api_destination/0","doc":"Temporary override to do more testing of the new ingest.plausible.io endpoint for accepting events. In staging and locally\nwill fall back to staging.plausible.io/api/event and localhost:8000/api/event respectively.","ref":"PlausibleWeb.Dogfood.html#api_destination/0"},{"type":"function","title":"PlausibleWeb.Dogfood.domain/1","doc":"","ref":"PlausibleWeb.Dogfood.html#domain/1"},{"type":"function","title":"PlausibleWeb.Dogfood.script_url/0","doc":"","ref":"PlausibleWeb.Dogfood.html#script_url/0"},{"type":"module","title":"PlausibleWeb.Email","doc":"","ref":"PlausibleWeb.Email.html"},{"type":"function","title":"PlausibleWeb.Email.activation_email/2","doc":"","ref":"PlausibleWeb.Email.html#activation_email/2"},{"type":"function","title":"PlausibleWeb.Email.approaching_accept_traffic_until/1","doc":"","ref":"PlausibleWeb.Email.html#approaching_accept_traffic_until/1"},{"type":"function","title":"PlausibleWeb.Email.approaching_accept_traffic_until_tomorrow/1","doc":"","ref":"PlausibleWeb.Email.html#approaching_accept_traffic_until_tomorrow/1"},{"type":"function","title":"PlausibleWeb.Email.base_email/0","doc":"","ref":"PlausibleWeb.Email.html#base_email/0"},{"type":"function","title":"PlausibleWeb.Email.base_email/1","doc":"","ref":"PlausibleWeb.Email.html#base_email/1"},{"type":"function","title":"PlausibleWeb.Email.cancellation_email/1","doc":"","ref":"PlausibleWeb.Email.html#cancellation_email/1"},{"type":"function","title":"PlausibleWeb.Email.check_stats_email/1","doc":"","ref":"PlausibleWeb.Email.html#check_stats_email/1"},{"type":"function","title":"PlausibleWeb.Email.create_site_email/1","doc":"","ref":"PlausibleWeb.Email.html#create_site_email/1"},{"type":"function","title":"PlausibleWeb.Email.dashboard_locked/3","doc":"","ref":"PlausibleWeb.Email.html#dashboard_locked/3"},{"type":"function","title":"PlausibleWeb.Email.drop_notification/5","doc":"","ref":"PlausibleWeb.Email.html#drop_notification/5"},{"type":"function","title":"PlausibleWeb.Email.enterprise_over_limit_internal_email/4","doc":"","ref":"PlausibleWeb.Email.html#enterprise_over_limit_internal_email/4"},{"type":"function","title":"PlausibleWeb.Email.error_report/3","doc":"","ref":"PlausibleWeb.Email.html#error_report/3"},{"type":"function","title":"PlausibleWeb.Email.existing_user_invitation/1","doc":"","ref":"PlausibleWeb.Email.html#existing_user_invitation/1"},{"type":"function","title":"PlausibleWeb.Email.export_failure/2","doc":"","ref":"PlausibleWeb.Email.html#export_failure/2"},{"type":"function","title":"PlausibleWeb.Email.export_success/3","doc":"","ref":"PlausibleWeb.Email.html#export_success/3"},{"type":"function","title":"PlausibleWeb.Email.import_failure/2","doc":"","ref":"PlausibleWeb.Email.html#import_failure/2"},{"type":"function","title":"PlausibleWeb.Email.import_success/2","doc":"","ref":"PlausibleWeb.Email.html#import_success/2"},{"type":"function","title":"PlausibleWeb.Email.invitation_accepted/1","doc":"","ref":"PlausibleWeb.Email.html#invitation_accepted/1"},{"type":"function","title":"PlausibleWeb.Email.invitation_rejected/1","doc":"","ref":"PlausibleWeb.Email.html#invitation_rejected/1"},{"type":"function","title":"PlausibleWeb.Email.mailer_email_from/0","doc":"","ref":"PlausibleWeb.Email.html#mailer_email_from/0"},{"type":"function","title":"PlausibleWeb.Email.new_user_invitation/1","doc":"","ref":"PlausibleWeb.Email.html#new_user_invitation/1"},{"type":"function","title":"PlausibleWeb.Email.over_limit_email/3","doc":"","ref":"PlausibleWeb.Email.html#over_limit_email/3"},{"type":"function","title":"PlausibleWeb.Email.ownership_transfer_accepted/1","doc":"","ref":"PlausibleWeb.Email.html#ownership_transfer_accepted/1"},{"type":"function","title":"PlausibleWeb.Email.ownership_transfer_rejected/1","doc":"","ref":"PlausibleWeb.Email.html#ownership_transfer_rejected/1"},{"type":"function","title":"PlausibleWeb.Email.ownership_transfer_request/2","doc":"","ref":"PlausibleWeb.Email.html#ownership_transfer_request/2"},{"type":"function","title":"PlausibleWeb.Email.password_reset_email/2","doc":"","ref":"PlausibleWeb.Email.html#password_reset_email/2"},{"type":"function","title":"PlausibleWeb.Email.priority_email/0","doc":"Unlike the default 'base' emails, priority emails cannot be unsubscribed from. This is achieved\n by sending them through a dedicated 'priority' message stream in Postmark.","ref":"PlausibleWeb.Email.html#priority_email/0"},{"type":"function","title":"PlausibleWeb.Email.priority_email/1","doc":"","ref":"PlausibleWeb.Email.html#priority_email/1"},{"type":"function","title":"PlausibleWeb.Email.render/3","doc":"Render an Phoenix template and set the body on the email.\n\nPass an atom as the template name (:welcome_email) to render HTML *and* plain\ntext emails. Use a string if you only want to render one type, e.g.\n\"welcome_email.text\" or \"welcome_email.html\". Scroll to the top for more examples.","ref":"PlausibleWeb.Email.html#render/3"},{"type":"function","title":"PlausibleWeb.Email.site_member_removed/1","doc":"","ref":"PlausibleWeb.Email.html#site_member_removed/1"},{"type":"function","title":"PlausibleWeb.Email.site_setup_help/2","doc":"","ref":"PlausibleWeb.Email.html#site_setup_help/2"},{"type":"function","title":"PlausibleWeb.Email.site_setup_success/2","doc":"","ref":"PlausibleWeb.Email.html#site_setup_success/2"},{"type":"function","title":"PlausibleWeb.Email.spike_notification/5","doc":"","ref":"PlausibleWeb.Email.html#spike_notification/5"},{"type":"function","title":"PlausibleWeb.Email.stats_report/2","doc":"","ref":"PlausibleWeb.Email.html#stats_report/2"},{"type":"function","title":"PlausibleWeb.Email.trial_one_week_reminder/1","doc":"","ref":"PlausibleWeb.Email.html#trial_one_week_reminder/1"},{"type":"function","title":"PlausibleWeb.Email.trial_over_email/1","doc":"","ref":"PlausibleWeb.Email.html#trial_over_email/1"},{"type":"function","title":"PlausibleWeb.Email.trial_upgrade_email/3","doc":"","ref":"PlausibleWeb.Email.html#trial_upgrade_email/3"},{"type":"function","title":"PlausibleWeb.Email.two_factor_disabled_email/1","doc":"","ref":"PlausibleWeb.Email.html#two_factor_disabled_email/1"},{"type":"function","title":"PlausibleWeb.Email.two_factor_enabled_email/1","doc":"","ref":"PlausibleWeb.Email.html#two_factor_enabled_email/1"},{"type":"function","title":"PlausibleWeb.Email.welcome_email/1","doc":"","ref":"PlausibleWeb.Email.html#welcome_email/1"},{"type":"function","title":"PlausibleWeb.Email.yearly_expiration_notification/1","doc":"","ref":"PlausibleWeb.Email.html#yearly_expiration_notification/1"},{"type":"function","title":"PlausibleWeb.Email.yearly_renewal_notification/1","doc":"","ref":"PlausibleWeb.Email.html#yearly_renewal_notification/1"},{"type":"module","title":"PlausibleWeb.EmailView","doc":"","ref":"PlausibleWeb.EmailView.html"},{"type":"function","title":"PlausibleWeb.EmailView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.EmailView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.EmailView.activation_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#activation_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.approaching_accept_traffic_until.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#approaching_accept_traffic_until.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.cancellation_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#cancellation_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.check_stats_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#check_stats_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.create_site_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#create_site_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.csv_import.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#csv_import.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.dashboard_locked.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#dashboard_locked.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.date_format/1","doc":"","ref":"PlausibleWeb.EmailView.html#date_format/1"},{"type":"function","title":"PlausibleWeb.EmailView.drop_notification.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#drop_notification.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.enterprise_over_limit_internal.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#enterprise_over_limit_internal.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.error_report_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#error_report_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.existing_user_invitation.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#existing_user_invitation.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.export_failure.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#export_failure.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.export_success.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#export_success.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.google_analytics_import.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#google_analytics_import.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.greet_recipient/1","doc":"","ref":"PlausibleWeb.EmailView.html#greet_recipient/1"},{"type":"function","title":"PlausibleWeb.EmailView.invitation_accepted.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#invitation_accepted.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.invitation_rejected.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#invitation_rejected.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.new_user_invitation.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#new_user_invitation.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.over_limit.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#over_limit.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.ownership_transfer_accepted.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#ownership_transfer_accepted.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.ownership_transfer_rejected.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#ownership_transfer_rejected.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.ownership_transfer_request.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#ownership_transfer_request.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.password_reset_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#password_reset_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.plausible_url/0","doc":"","ref":"PlausibleWeb.EmailView.html#plausible_url/0"},{"type":"function","title":"PlausibleWeb.EmailView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.EmailView.html#render/2"},{"type":"function","title":"PlausibleWeb.EmailView.sentry_link/2","doc":"","ref":"PlausibleWeb.EmailView.html#sentry_link/2"},{"type":"function","title":"PlausibleWeb.EmailView.site_member_removed.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#site_member_removed.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.site_setup_help_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#site_setup_help_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.site_setup_success_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#site_setup_success_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.spike_notification.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#spike_notification.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.EmailView.html#template_not_found/2"},{"type":"function","title":"PlausibleWeb.EmailView.trial_one_week_reminder.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#trial_one_week_reminder.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.trial_over_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#trial_over_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.trial_upgrade_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#trial_upgrade_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.two_factor_disabled_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#two_factor_disabled_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.two_factor_enabled_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#two_factor_enabled_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.welcome_email.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#welcome_email.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.yearly_expiration_notification.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#yearly_expiration_notification.html/1"},{"type":"function","title":"PlausibleWeb.EmailView.yearly_renewal_notification.html/1","doc":"","ref":"PlausibleWeb.EmailView.html#yearly_renewal_notification.html/1"},{"type":"module","title":"PlausibleWeb.Endpoint","doc":"","ref":"PlausibleWeb.Endpoint.html"},{"type":"function","title":"PlausibleWeb.Endpoint.broadcast/3","doc":"","ref":"PlausibleWeb.Endpoint.html#broadcast/3"},{"type":"function","title":"PlausibleWeb.Endpoint.broadcast!/3","doc":"","ref":"PlausibleWeb.Endpoint.html#broadcast!/3"},{"type":"function","title":"PlausibleWeb.Endpoint.broadcast_from/4","doc":"","ref":"PlausibleWeb.Endpoint.html#broadcast_from/4"},{"type":"function","title":"PlausibleWeb.Endpoint.broadcast_from!/4","doc":"","ref":"PlausibleWeb.Endpoint.html#broadcast_from!/4"},{"type":"function","title":"PlausibleWeb.Endpoint.call/2","doc":"","ref":"PlausibleWeb.Endpoint.html#call/2"},{"type":"function","title":"PlausibleWeb.Endpoint.child_spec/1","doc":"Returns the child specification to start the endpoint\nunder a supervision tree.","ref":"PlausibleWeb.Endpoint.html#child_spec/1"},{"type":"function","title":"PlausibleWeb.Endpoint.config/2","doc":"Returns the endpoint configuration for `key`\n\nReturns `default` if the key does not exist.","ref":"PlausibleWeb.Endpoint.html#config/2"},{"type":"function","title":"PlausibleWeb.Endpoint.config_change/2","doc":"Reloads the configuration given the application environment changes.","ref":"PlausibleWeb.Endpoint.html#config_change/2"},{"type":"function","title":"PlausibleWeb.Endpoint.host/0","doc":"Returns the host for the given endpoint.","ref":"PlausibleWeb.Endpoint.html#host/0"},{"type":"function","title":"PlausibleWeb.Endpoint.init/1","doc":"","ref":"PlausibleWeb.Endpoint.html#init/1"},{"type":"function","title":"PlausibleWeb.Endpoint.local_broadcast/3","doc":"","ref":"PlausibleWeb.Endpoint.html#local_broadcast/3"},{"type":"function","title":"PlausibleWeb.Endpoint.local_broadcast_from/4","doc":"","ref":"PlausibleWeb.Endpoint.html#local_broadcast_from/4"},{"type":"function","title":"PlausibleWeb.Endpoint.path/1","doc":"Generates the path information when routing to this endpoint.","ref":"PlausibleWeb.Endpoint.html#path/1"},{"type":"function","title":"PlausibleWeb.Endpoint.runtime_session/2","doc":"","ref":"PlausibleWeb.Endpoint.html#runtime_session/2"},{"type":"function","title":"PlausibleWeb.Endpoint.runtime_session_opts/0","doc":"","ref":"PlausibleWeb.Endpoint.html#runtime_session_opts/0"},{"type":"function","title":"PlausibleWeb.Endpoint.script_name/0","doc":"Generates the script name.","ref":"PlausibleWeb.Endpoint.html#script_name/0"},{"type":"function","title":"PlausibleWeb.Endpoint.secure_cookie?/0","doc":"","ref":"PlausibleWeb.Endpoint.html#secure_cookie?/0"},{"type":"function","title":"PlausibleWeb.Endpoint.server_info/1","doc":"Returns the address and port that the server is running on","ref":"PlausibleWeb.Endpoint.html#server_info/1"},{"type":"function","title":"PlausibleWeb.Endpoint.start_link/1","doc":"Starts the endpoint supervision tree.\n\nAll other options are merged into the endpoint configuration.","ref":"PlausibleWeb.Endpoint.html#start_link/1"},{"type":"function","title":"PlausibleWeb.Endpoint.static_integrity/1","doc":"Generates a base64-encoded cryptographic hash (sha512) to a static file\nin `priv/static`. Meant to be used for Subresource Integrity with CDNs.","ref":"PlausibleWeb.Endpoint.html#static_integrity/1"},{"type":"function","title":"PlausibleWeb.Endpoint.static_lookup/1","doc":"Returns a two item tuple with the first item being the `static_path`\nand the second item being the `static_integrity`.","ref":"PlausibleWeb.Endpoint.html#static_lookup/1"},{"type":"function","title":"PlausibleWeb.Endpoint.static_path/1","doc":"Generates a route to a static file in `priv/static`.","ref":"PlausibleWeb.Endpoint.html#static_path/1"},{"type":"function","title":"PlausibleWeb.Endpoint.static_url/0","doc":"Generates the static URL without any path information.\n\nIt uses the configuration under `:static_url` to generate\nsuch. It falls back to `:url` if `:static_url` is not set.","ref":"PlausibleWeb.Endpoint.html#static_url/0"},{"type":"function","title":"PlausibleWeb.Endpoint.struct_url/0","doc":"Generates the endpoint base URL but as a `URI` struct.\n\nIt uses the configuration under `:url` to generate such.\nUseful for manipulating the URL data and passing it to\nURL helpers.","ref":"PlausibleWeb.Endpoint.html#struct_url/0"},{"type":"function","title":"PlausibleWeb.Endpoint.subscribe/2","doc":"","ref":"PlausibleWeb.Endpoint.html#subscribe/2"},{"type":"function","title":"PlausibleWeb.Endpoint.unsubscribe/1","doc":"","ref":"PlausibleWeb.Endpoint.html#unsubscribe/1"},{"type":"function","title":"PlausibleWeb.Endpoint.url/0","doc":"Generates the endpoint base URL without any path information.\n\nIt uses the configuration under `:url` to generate such.","ref":"PlausibleWeb.Endpoint.html#url/0"},{"type":"function","title":"PlausibleWeb.Endpoint.websocket_url/0","doc":"","ref":"PlausibleWeb.Endpoint.html#websocket_url/0"},{"type":"module","title":"PlausibleWeb.ErrorHelpers","doc":"","ref":"PlausibleWeb.ErrorHelpers.html"},{"type":"function","title":"PlausibleWeb.ErrorHelpers.error_tag/2","doc":"","ref":"PlausibleWeb.ErrorHelpers.html#error_tag/2"},{"type":"function","title":"PlausibleWeb.ErrorHelpers.translate_error/1","doc":"","ref":"PlausibleWeb.ErrorHelpers.html#translate_error/1"},{"type":"module","title":"PlausibleWeb.ErrorReportController","doc":"","ref":"PlausibleWeb.ErrorReportController.html"},{"type":"function","title":"PlausibleWeb.ErrorReportController.submit_error_report/2","doc":"","ref":"PlausibleWeb.ErrorReportController.html#submit_error_report/2"},{"type":"module","title":"PlausibleWeb.ErrorView","doc":"","ref":"PlausibleWeb.ErrorView.html"},{"type":"function","title":"PlausibleWeb.ErrorView.404_error.html/1","doc":"","ref":"PlausibleWeb.ErrorView.html#404_error.html/1"},{"type":"function","title":"PlausibleWeb.ErrorView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.ErrorView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.ErrorView.generic_error.html/1","doc":"","ref":"PlausibleWeb.ErrorView.html#generic_error.html/1"},{"type":"function","title":"PlausibleWeb.ErrorView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.ErrorView.html#render/2"},{"type":"function","title":"PlausibleWeb.ErrorView.server_error.html/1","doc":"","ref":"PlausibleWeb.ErrorView.html#server_error.html/1"},{"type":"function","title":"PlausibleWeb.ErrorView.server_error_report_thanks.html/1","doc":"","ref":"PlausibleWeb.ErrorView.html#server_error_report_thanks.html/1"},{"type":"function","title":"PlausibleWeb.ErrorView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.ErrorView.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Favicon","doc":"A Plug that fetches favicon images from DuckDuckGo and returns them\nto the Plausible frontend.\n\nThe proxying is there so we can reduce the number of third-party domains that\nthe browser clients need to connect to. Our goal is to have 0 third-party domain\nconnections on the website for privacy reasons.\n\nThis module also maps between categorized sources and their respective URLs for favicons.\nWhat does that mean exactly? During ingestion we use `PlausibleWeb.RefInspector.parse/1` to\ncategorize our referrer sources like so:\n\ngoogle.com -> Google\ngoogle.co.uk -> Google\ngoogle.com.au -> Google\n\nSo when we show Google as a source in the dashboard, the request to this plug will come as:\nhttps://plausible/io/favicon/sources/Google\n\nNow, when we want to show a favicon for Google, we need to convert Google -> google.com or\nsome other hostname owned by Google:\nhttps://icons.duckduckgo.com/ip3/google.com.ico\n\nThe mapping from source category -> source hostname is stored in \"priv/referer_favicon_domains.json\" and\nmanaged by `Mix.Tasks.GenerateReferrerFavicons.run/1`","ref":"PlausibleWeb.Favicon.html"},{"type":"function","title":"PlausibleWeb.Favicon.call/2","doc":"Proxies HTTP request to DuckDuckGo favicon service. Swallows hop-by-hop HTTP\nheaders that should not be forwarded as defined in [RFC 2616](https://www.rfc-editor.org/rfc/rfc2616#section-13.5.1)","ref":"PlausibleWeb.Favicon.html#call/2"},{"type":"function","title":"Placeholder - PlausibleWeb.Favicon.call/2","doc":"Cases where we show a placeholder icon instead:\n\n1. In case of network error to DuckDuckGo\n2. In case of non-2xx status code from DuckDuckGo\n3. In case of broken image response body from DuckDuckGo\n\nI'm not sure why DDG sometimes returns a broken PNG image in their response\nbut we filter that out. When the icon request fails, we show a placeholder\nfavicon instead. The placeholder is an emoji from\n[https://favicon.io/emoji-favicons/](https://favicon.io/emoji-favicons/)\n\nDuckDuckGo favicon service has some issues with [SVG favicons](https://css-tricks.com/svg-favicons-and-all-the-fun-things-we-can-do-with-them/).\nFor some reason, they return them with `content-type=image/x-icon` whereas SVG\nicons should be returned with `content-type=image/svg+xml`. This Plug detects\nwhen the response body starts with `\n<.input name=\"my-input\" errors={[\"oh no!\"]} />","ref":"PlausibleWeb.Live.Components.Form.html#input/1-examples"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Form.input/1","doc":"* `id` (`:any`) - Defaults to `nil`.\n* `name` (`:any`)\n* `label` (`:string`) - Defaults to `nil`.\n* `value` (`:any`)\n* `type` (`:string`) - Defaults to `\"text\"`.\n* `field` (`Phoenix.HTML.FormField`) - a form field struct retrieved from the form, for example: @form[:email].\n* `errors` (`:list`) - Defaults to `[]`.\n* `checked` (`:boolean`) - the checked flag for checkbox inputs.\n* `prompt` (`:string`) - the prompt for select inputs. Defaults to `nil`.\n* `options` (`:list`) - the options to pass to Phoenix.HTML.Form.options_for_select/2.\n* `multiple` (`:boolean`) - the multiple flag for select inputs. Defaults to `false`.\n* Global attributes are accepted. Supports all globals plus: `[\"accept\", \"autocomplete\", \"capture\", \"cols\", \"disabled\", \"form\", \"list\", \"max\", \"maxlength\", \"min\", \"minlength\", \"multiple\", \"pattern\", \"placeholder\", \"readonly\", \"required\", \"rows\", \"size\", \"step\"]`.","ref":"PlausibleWeb.Live.Components.Form.html#input/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Live.Components.Form.input/1","doc":"* `inner_block`","ref":"PlausibleWeb.Live.Components.Form.html#input/1-slots"},{"type":"function","title":"PlausibleWeb.Live.Components.Form.input_with_clipboard/1","doc":"","ref":"PlausibleWeb.Live.Components.Form.html#input_with_clipboard/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Form.input_with_clipboard/1","doc":"* `id` (`:string`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.\n* `name` (`:string`) (required)\n* `label` (`:string`) (required)\n* `value` (`:string`) - Defaults to `\"\"`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Live.Components.Form.html#input_with_clipboard/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Components.Form.label/1","doc":"Renders a label.","ref":"PlausibleWeb.Live.Components.Form.html#label/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Form.label/1","doc":"* `for` (`:string`) - Defaults to `nil`.","ref":"PlausibleWeb.Live.Components.Form.html#label/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Live.Components.Form.label/1","doc":"* `inner_block` (required)","ref":"PlausibleWeb.Live.Components.Form.html#label/1-slots"},{"type":"function","title":"PlausibleWeb.Live.Components.Form.password_input_with_strength/1","doc":"","ref":"PlausibleWeb.Live.Components.Form.html#password_input_with_strength/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Form.password_input_with_strength/1","doc":"* `id` (`:any`) - Defaults to `nil`.\n* `label` (`:string`) - Defaults to `nil`.\n* `field` (`Phoenix.HTML.FormField`) (required) - a form field struct retrieved from the form, for example: @form[:password].\n* `strength` (`:any`)\n* Global attributes are accepted. Supports all globals plus: `[\"autocomplete\", \"disabled\", \"form\", \"maxlength\", \"minlength\", \"readonly\", \"required\", \"size\"]`.","ref":"PlausibleWeb.Live.Components.Form.html#password_input_with_strength/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Components.Form.password_length_hint/1","doc":"","ref":"PlausibleWeb.Live.Components.Form.html#password_length_hint/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Form.password_length_hint/1","doc":"* `minimum` (`:integer`) (required)\n* `class` (`:any`)\n* `ok_class` (`:any`)\n* `error_class` (`:any`)\n* `field` (`Phoenix.HTML.FormField`) (required) - a form field struct retrieved from the form, for example: @form[:password].","ref":"PlausibleWeb.Live.Components.Form.html#password_length_hint/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Components.Form.strength_meter/1","doc":"","ref":"PlausibleWeb.Live.Components.Form.html#strength_meter/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Form.strength_meter/1","doc":"* `score` (`:integer`) - Defaults to `0`.\n* `warning` (`:string`) - Defaults to `\"\"`.\n* `suggestions` (`:list`) - Defaults to `[]`.","ref":"PlausibleWeb.Live.Components.Form.html#strength_meter/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Components.Form.translate_error/1","doc":"","ref":"PlausibleWeb.Live.Components.Form.html#translate_error/1"},{"type":"module","title":"PlausibleWeb.Live.Components.Modal","doc":"LiveView implementation of modal component.\n\nThis component is a general purpose modal implementation for LiveView\nwith emphasis on keeping nested components largely agnostic of the fact\nthat they are placed in a modal and maintaining good user experience\non connections with high latency.","ref":"PlausibleWeb.Live.Components.Modal.html"},{"type":"module","title":"Usage - PlausibleWeb.Live.Components.Modal","doc":"An example use case for a modal is embedding a form inside\nexisting live view which allows adding new entries of some kind:\n\n```\n<.live_component module={Modal} id=\"some-form-modal\" :let={modal_unique_id}>\n <.live_component\n module={SomeForm}\n id={\"some-form-#{modal_unique_id}\"}\n on_save_form={\n fn entry, socket ->\n send(self(), {:entry_added, entry})\n Modal.close(socket, \"some-form-modal\")\n end\n }\n />\n\n```\n\nThen somewhere in the same live view the modal is rendered in:\n\n```\n<.button x-data x-on:click={Modal.JS.open(\"goals-form-modal\")}>\n + Add Entry\n\n```","ref":"PlausibleWeb.Live.Components.Modal.html#module-usage"},{"type":"module","title":"Explanation - PlausibleWeb.Live.Components.Modal","doc":"The component embedded inside the modal is always rendered when\nthe live view is mounted but is kept hidden until `Modal.JS.open`\nis called on it. On subsequent openings within the same session\nthe contents of the modal are completely remounted. This assures\nthat any stateful components inside the modal are reset to their\ninitial state. The modal component provides `modal_unique_id`\nas an argument to its inner block. Appending this ID to every\nlive components' ID nested inside the modal is important for\nconsistent state reset on every reopening. This also applies\nto live components nested inside live components embedded directly\nin the modal's inner block - then the unique ID should be also\npassed down as an attribute and appended accordingly. Appending can\nbe skipped if embedded component handles state reset explicitly\n(via, for instance, `phx-click-away` callback).\n\n`Modal` exposes a number of functions for managing window state:\n\n * `Modal.JS.preopen/1` - to preopen the modal on the frontend.\n Useful when the actual opening is done server-side with\n `Modal.open/2` - helps avoid lack of feedback to the end user\n when server-side state change before opening the modal is\n still in progress.\n * `Modal.JS.open/1` - to open the modal from the frontend. It's\n important to make sure the element triggering that call is\n wrapped in an Alpine UI component - or is an Alpine component\n itself - adding `x-data` attribute without any value is enough\n to ensure that.\n * `Modal.open/2` - to open the modal from the backend; usually\n called from `handle_event/2` of component wrapping the modal\n and providing the state. Should be used together with\n `Modal.JS.preopen/1` for optimal user experience.\n * `Modal.close/2` - to close the modal from the backend; usually\n done inside wrapped component's `handle_event/2`. The example\n quoted above shows one way to implement this, under that assumption\n that the component exposes a callback, like this:\n\n ```\n defmodule SomeForm do\n use Phoenix.LiveComponent\n\n def update(assigns, socket) do\n # ...\n\n {:ok, assign(socket, :on_save_form, assigns.on_save_form)}\n end\n\n #...\n\n def handle_event(\"save-form\", %{\"form\" => form}, socket) do\n case save_entry(form) do\n {:ok, entry} ->\n {:noreply, socket.assigns.on_save_form(entry, socket)}\n\n # error case handling ...\n end\n end\n end\n ```\n\n Using callback approach has an added benefit of making the\n component more flexible.","ref":"PlausibleWeb.Live.Components.Modal.html#module-explanation"},{"type":"function","title":"PlausibleWeb.Live.Components.Modal.close/2","doc":"","ref":"PlausibleWeb.Live.Components.Modal.html#close/2"},{"type":"function","title":"PlausibleWeb.Live.Components.Modal.open/2","doc":"","ref":"PlausibleWeb.Live.Components.Modal.html#open/2"},{"type":"function","title":"PlausibleWeb.Live.Components.Modal.render/1","doc":"","ref":"PlausibleWeb.Live.Components.Modal.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Modal.render/1","doc":"* `id` (`:any`) (required)\n* `class` (`:string`) - Defaults to `\"\"`.\n* `preload?` (`:boolean`) - Defaults to `true`.","ref":"PlausibleWeb.Live.Components.Modal.html#render/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Live.Components.Modal.render/1","doc":"* `inner_block` (required)","ref":"PlausibleWeb.Live.Components.Modal.html#render/1-slots"},{"type":"module","title":"PlausibleWeb.Live.Components.Pagination","doc":"Pagination components for LiveViews.","ref":"PlausibleWeb.Live.Components.Pagination.html"},{"type":"function","title":"PlausibleWeb.Live.Components.Pagination.pagination/1","doc":"","ref":"PlausibleWeb.Live.Components.Pagination.html#pagination/1"},{"type":"module","title":"PlausibleWeb.Live.Components.Verification","doc":"This component is responsible for rendering the verification progress\nand diagnostics.","ref":"PlausibleWeb.Live.Components.Verification.html"},{"type":"function","title":"PlausibleWeb.Live.Components.Verification.render/1","doc":"","ref":"PlausibleWeb.Live.Components.Verification.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Verification.render/1","doc":"* `domain` (`:string`) (required)\n* `message` (`:string`) - Defaults to `\"We're visiting your site to ensure that everything is working\"`.\n* `finished?` (`:boolean`) - Defaults to `false`.\n* `success?` (`:boolean`) - Defaults to `false`.\n* `interpretation` (`Plausible.Verification.Diagnostics.Result`) - Defaults to `nil`.\n* `attempts` (`:integer`) - Defaults to `0`.\n* `flow` (`:string`) - Defaults to `\"\"`.\n* `installation_type` (`:string`) - Defaults to `nil`.\n* `awaiting_first_pageview?` (`:boolean`) - Defaults to `false`.","ref":"PlausibleWeb.Live.Components.Verification.html#render/1-attributes"},{"type":"module","title":"PlausibleWeb.Live.Components.Visitors","doc":"Component rendering mini-graph of site's visitors over the last 24 hours.\n\nThe `gradient_defs` component should be rendered once before using `chart`\none or more times.\n\nAccepts input generated via `Plausible.Stats.Clickhouse.last_24h_visitors_hourly_intervals/2`.","ref":"PlausibleWeb.Live.Components.Visitors.html"},{"type":"function","title":"PlausibleWeb.Live.Components.Visitors.chart/1","doc":"","ref":"PlausibleWeb.Live.Components.Visitors.html#chart/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Components.Visitors.chart/1","doc":"* `intervals` (`:list`) (required)\n* `height` (`:integer`) - Defaults to `50`.\n* `tick` (`:integer`) - Defaults to `20`.","ref":"PlausibleWeb.Live.Components.Visitors.html#chart/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Components.Visitors.gradient_defs/1","doc":"","ref":"PlausibleWeb.Live.Components.Visitors.html#gradient_defs/1"},{"type":"module","title":"PlausibleWeb.Live.Flash","doc":"Flash component for LiveViews - works also when embedded within dead views","ref":"PlausibleWeb.Live.Flash.html"},{"type":"function","title":"PlausibleWeb.Live.Flash.clear_flash_button/1","doc":"","ref":"PlausibleWeb.Live.Flash.html#clear_flash_button/1"},{"type":"function","title":"PlausibleWeb.Live.Flash.flash/1","doc":"","ref":"PlausibleWeb.Live.Flash.html#flash/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Flash.flash/1","doc":"* `key` (`:string`) - Defaults to `nil`.\n* `on_close` (`:any`) - Defaults to `\"lv:clear-flash\"`.\n* `class` (`:string`) - Defaults to `\"\"`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Live.Flash.html#flash/1-attributes"},{"type":"function","title":"Slots - PlausibleWeb.Live.Flash.flash/1","doc":"* `icon` (required)\n* `title`\n* `message` (required)","ref":"PlausibleWeb.Live.Flash.html#flash/1-slots"},{"type":"function","title":"PlausibleWeb.Live.Flash.flash_messages/1","doc":"","ref":"PlausibleWeb.Live.Flash.html#flash_messages/1"},{"type":"function","title":"PlausibleWeb.Live.Flash.icon_error/1","doc":"","ref":"PlausibleWeb.Live.Flash.html#icon_error/1"},{"type":"function","title":"PlausibleWeb.Live.Flash.icon_success/1","doc":"","ref":"PlausibleWeb.Live.Flash.html#icon_success/1"},{"type":"function","title":"PlausibleWeb.Live.Flash.put_live_flash/3","doc":"","ref":"PlausibleWeb.Live.Flash.html#put_live_flash/3"},{"type":"module","title":"PlausibleWeb.Live.FunnelSettings","doc":"LiveView allowing listing, creating and deleting funnels.","ref":"PlausibleWeb.Live.FunnelSettings.html"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.handle_event/3","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.handle_info/2","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.mount/3","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.render/1","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.FunnelSettings.Form","doc":"Phoenix LiveComponent that renders a form used for setting up funnels.\nMakes use of dynamically placed `PlausibleWeb.Live.FunnelSettings.ComboBox` components\nto allow building searchable funnel definitions out of list of goals available.","ref":"PlausibleWeb.Live.FunnelSettings.Form.html"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.add_step_button/1","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#add_step_button/1"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.evaluation/1","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#evaluation/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.FunnelSettings.Form.evaluation/1","doc":"* `at` (`:integer`) (required)\n* `result` (`:map`) (required)","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#evaluation/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.handle_event/3","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.handle_info/2","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.mount/3","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.remove_step_button/1","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#remove_step_button/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.FunnelSettings.Form.remove_step_button/1","doc":"* `step_idx` (`:integer`) (required)","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#remove_step_button/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.Form.render/1","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.Form.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.FunnelSettings.List","doc":"Phoenix LiveComponent module that renders a list of funnels with their names\nand the number of steps they have.\n\nEach funnel is displayed with a delete button, which triggers a confirmation\nmessage before deleting the funnel from the UI. If there are no funnels\nconfigured for the site, a message is displayed indicating so.","ref":"PlausibleWeb.Live.FunnelSettings.List.html"},{"type":"function","title":"PlausibleWeb.Live.FunnelSettings.List.render/1","doc":"","ref":"PlausibleWeb.Live.FunnelSettings.List.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.GoalSettings","doc":"LiveView allowing listing, creating and deleting goals.","ref":"PlausibleWeb.Live.GoalSettings.html"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.handle_event/3","doc":"","ref":"PlausibleWeb.Live.GoalSettings.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.handle_info/2","doc":"","ref":"PlausibleWeb.Live.GoalSettings.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.mount/3","doc":"","ref":"PlausibleWeb.Live.GoalSettings.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.render/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.GoalSettings.Form","doc":"Live view for the goal creation form","ref":"PlausibleWeb.Live.GoalSettings.Form.html"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.create_form/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#create_form/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.custom_event_fields/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#custom_event_fields/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.GoalSettings.Form.custom_event_fields/1","doc":"* `f` (`Phoenix.HTML.Form`)\n* `site` (`Plausible.Site`)\n* `current_user` (`Plausible.Auth.User`)\n* `suffix` (`:string`)\n* `existing_goals` (`:list`)\n* `goal_options` (`:list`)\n* `goal` (`Plausible.Goal`) - Defaults to `nil`.\n* `has_access_to_revenue_goals?` (`:boolean`)\n* Global attributes are accepted.","ref":"PlausibleWeb.Live.GoalSettings.Form.html#custom_event_fields/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.edit_form/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#edit_form/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.handle_event/3","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.pageview_fields/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#pageview_fields/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.GoalSettings.Form.pageview_fields/1","doc":"* `f` (`Phoenix.HTML.Form`)\n* `site` (`Plausible.Site`)\n* `suffix` (`:string`)\n* `goal` (`Plausible.Goal`) - Defaults to `nil`.\n* Global attributes are accepted.","ref":"PlausibleWeb.Live.GoalSettings.Form.html#pageview_fields/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.pageviews_tab/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#pageviews_tab/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.render/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.revenue_goal_settings/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#revenue_goal_settings/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.suggest_event_names/3","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#suggest_event_names/3"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.suggest_page_paths/2","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#suggest_page_paths/2"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.tabs/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#tabs/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.Form.update/2","doc":"","ref":"PlausibleWeb.Live.GoalSettings.Form.html#update/2"},{"type":"module","title":"PlausibleWeb.Live.GoalSettings.List","doc":"Phoenix LiveComponent module that renders a list of goals","ref":"PlausibleWeb.Live.GoalSettings.List.html"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.List.custom_event_description/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.List.html#custom_event_description/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.List.goal_description/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.List.html#goal_description/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.List.pageview_description/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.List.html#pageview_description/1"},{"type":"function","title":"PlausibleWeb.Live.GoalSettings.List.render/1","doc":"","ref":"PlausibleWeb.Live.GoalSettings.List.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.GoalSettings.List.render/1","doc":"* `goals` (`:list`) (required)\n* `domain` (`:string`) (required)\n* `filter_text` (`:string`)\n* `site` (`Plausible.Site`) (required)","ref":"PlausibleWeb.Live.GoalSettings.List.html#render/1-attributes"},{"type":"module","title":"PlausibleWeb.Live.ImportsExportsSettings","doc":"LiveView allowing listing and deleting imports.","ref":"PlausibleWeb.Live.ImportsExportsSettings.html"},{"type":"function","title":"PlausibleWeb.Live.ImportsExportsSettings.handle_info/2","doc":"","ref":"PlausibleWeb.Live.ImportsExportsSettings.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.ImportsExportsSettings.mount/3","doc":"","ref":"PlausibleWeb.Live.ImportsExportsSettings.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.ImportsExportsSettings.render/1","doc":"","ref":"PlausibleWeb.Live.ImportsExportsSettings.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.Installation","doc":"User assistance module around Plausible installation instructions/onboarding","ref":"PlausibleWeb.Live.Installation.html"},{"type":"function","title":"PlausibleWeb.Live.Installation.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Installation.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Installation.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Installation.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Installation.handle_params/3","doc":"","ref":"PlausibleWeb.Live.Installation.html#handle_params/3"},{"type":"function","title":"PlausibleWeb.Live.Installation.mount/3","doc":"","ref":"PlausibleWeb.Live.Installation.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Installation.render/1","doc":"","ref":"PlausibleWeb.Live.Installation.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.Installation.render_snippet_404/0","doc":"","ref":"PlausibleWeb.Live.Installation.html#render_snippet_404/0"},{"type":"function","title":"PlausibleWeb.Live.Installation.render_snippet_404/1","doc":"","ref":"PlausibleWeb.Live.Installation.html#render_snippet_404/1"},{"type":"function","title":"PlausibleWeb.Live.Installation.script_extension_params/0","doc":"","ref":"PlausibleWeb.Live.Installation.html#script_extension_params/0"},{"type":"module","title":"PlausibleWeb.Live.Plugins.API.Settings","doc":"LiveView allowing listing, creating and revoking Plugins API tokens.","ref":"PlausibleWeb.Live.Plugins.API.Settings.html"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.Settings.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.Settings.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.Settings.mount/3","doc":"","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.Settings.render/1","doc":"","ref":"PlausibleWeb.Live.Plugins.API.Settings.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.Plugins.API.TokenForm","doc":"Live view for the goal creation form","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.TokenForm.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.TokenForm.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.TokenForm.mount/3","doc":"","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Plugins.API.TokenForm.render/1","doc":"","ref":"PlausibleWeb.Live.Plugins.API.TokenForm.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.PropsSettings","doc":"LiveView allowing listing, allowing and disallowing custom event properties.","ref":"PlausibleWeb.Live.PropsSettings.html"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.handle_event/3","doc":"","ref":"PlausibleWeb.Live.PropsSettings.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.handle_info/2","doc":"","ref":"PlausibleWeb.Live.PropsSettings.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.mount/3","doc":"","ref":"PlausibleWeb.Live.PropsSettings.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.render/1","doc":"","ref":"PlausibleWeb.Live.PropsSettings.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.PropsSettings.Form","doc":"Live view for the custom props creation form","ref":"PlausibleWeb.Live.PropsSettings.Form.html"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.Form.handle_event/3","doc":"","ref":"PlausibleWeb.Live.PropsSettings.Form.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.Form.handle_info/2","doc":"","ref":"PlausibleWeb.Live.PropsSettings.Form.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.Form.mount/3","doc":"","ref":"PlausibleWeb.Live.PropsSettings.Form.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.Form.render/1","doc":"","ref":"PlausibleWeb.Live.PropsSettings.Form.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.PropsSettings.List","doc":"Phoenix LiveComponent module that renders a list of custom properties","ref":"PlausibleWeb.Live.PropsSettings.List.html"},{"type":"function","title":"PlausibleWeb.Live.PropsSettings.List.render/1","doc":"","ref":"PlausibleWeb.Live.PropsSettings.List.html#render/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.PropsSettings.List.render/1","doc":"* `props` (`:list`) (required)\n* `domain` (`:string`) (required)\n* `filter_text` (`:string`)","ref":"PlausibleWeb.Live.PropsSettings.List.html#render/1-attributes"},{"type":"module","title":"PlausibleWeb.Live.RegisterForm","doc":"LiveView for registration form.","ref":"PlausibleWeb.Live.RegisterForm.html"},{"type":"function","title":"PlausibleWeb.Live.RegisterForm.handle_event/3","doc":"","ref":"PlausibleWeb.Live.RegisterForm.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.RegisterForm.handle_info/2","doc":"","ref":"PlausibleWeb.Live.RegisterForm.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.RegisterForm.mount/3","doc":"","ref":"PlausibleWeb.Live.RegisterForm.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.RegisterForm.render/1","doc":"","ref":"PlausibleWeb.Live.RegisterForm.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.ResetPasswordForm","doc":"LiveView for password reset form.","ref":"PlausibleWeb.Live.ResetPasswordForm.html"},{"type":"function","title":"PlausibleWeb.Live.ResetPasswordForm.handle_event/3","doc":"","ref":"PlausibleWeb.Live.ResetPasswordForm.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.ResetPasswordForm.handle_info/2","doc":"","ref":"PlausibleWeb.Live.ResetPasswordForm.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.ResetPasswordForm.mount/3","doc":"","ref":"PlausibleWeb.Live.ResetPasswordForm.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.ResetPasswordForm.render/1","doc":"","ref":"PlausibleWeb.Live.ResetPasswordForm.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.SentryContext","doc":"This module tries to supply LiveViews with some common Sentry context\n(without it, there is practically none).\n\nUse via `use PlausibleWeb.Live.SentryContext` in your LiveView module,\nor preferably via `use PlausibleWeb, :live_view`.\n\nIn case you have multiple LiveViews, there is `use PlausibleWeb, live_view: :no_sentry_context`\nexposed that allows you to skip using this module. This is because\nonly the root LiveView has access to `connect_info` and an exception will be\nthrown otherwise.","ref":"PlausibleWeb.Live.SentryContext.html"},{"type":"function","title":"PlausibleWeb.Live.SentryContext.on_mount/4","doc":"","ref":"PlausibleWeb.Live.SentryContext.html#on_mount/4"},{"type":"module","title":"PlausibleWeb.Live.Shields.Countries","doc":"LiveView for IP Addresses Shield","ref":"PlausibleWeb.Live.Shields.Countries.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.Countries.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Shields.Countries.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.Countries.mount/3","doc":"","ref":"PlausibleWeb.Live.Shields.Countries.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.Countries.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.Countries.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.Shields.CountryRules","doc":"LiveView allowing Country Rules management","ref":"PlausibleWeb.Live.Shields.CountryRules.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.CountryRules.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Shields.CountryRules.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.CountryRules.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.CountryRules.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.Shields.CountryRules.send_flash/2","doc":"","ref":"PlausibleWeb.Live.Shields.CountryRules.html#send_flash/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.CountryRules.update/2","doc":"","ref":"PlausibleWeb.Live.Shields.CountryRules.html#update/2"},{"type":"module","title":"PlausibleWeb.Live.Shields.HostnameRules","doc":"LiveView allowing hostname Rules management","ref":"PlausibleWeb.Live.Shields.HostnameRules.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.HostnameRules.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.HostnameRules.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.Shields.HostnameRules.send_flash/2","doc":"","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#send_flash/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.HostnameRules.suggest_hostnames/3","doc":"","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#suggest_hostnames/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.HostnameRules.update/2","doc":"","ref":"PlausibleWeb.Live.Shields.HostnameRules.html#update/2"},{"type":"module","title":"PlausibleWeb.Live.Shields.Hostnames","doc":"LiveView for Hostnames Shield","ref":"PlausibleWeb.Live.Shields.Hostnames.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.Hostnames.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Shields.Hostnames.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.Hostnames.mount/3","doc":"","ref":"PlausibleWeb.Live.Shields.Hostnames.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.Hostnames.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.Hostnames.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.Shields.IPAddresses","doc":"LiveView for IP Addresses Shield","ref":"PlausibleWeb.Live.Shields.IPAddresses.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPAddresses.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Shields.IPAddresses.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPAddresses.mount/3","doc":"","ref":"PlausibleWeb.Live.Shields.IPAddresses.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPAddresses.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.IPAddresses.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.Shields.IPRules","doc":"LiveView allowing IP Rules management","ref":"PlausibleWeb.Live.Shields.IPRules.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPRules.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Shields.IPRules.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPRules.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.IPRules.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPRules.send_flash/2","doc":"","ref":"PlausibleWeb.Live.Shields.IPRules.html#send_flash/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.IPRules.update/2","doc":"","ref":"PlausibleWeb.Live.Shields.IPRules.html#update/2"},{"type":"module","title":"PlausibleWeb.Live.Shields.PageRules","doc":"LiveView allowing page Rules management","ref":"PlausibleWeb.Live.Shields.PageRules.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.PageRules.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Shields.PageRules.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.PageRules.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.PageRules.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.Shields.PageRules.send_flash/2","doc":"","ref":"PlausibleWeb.Live.Shields.PageRules.html#send_flash/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.PageRules.suggest_page_paths/3","doc":"","ref":"PlausibleWeb.Live.Shields.PageRules.html#suggest_page_paths/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.PageRules.update/2","doc":"","ref":"PlausibleWeb.Live.Shields.PageRules.html#update/2"},{"type":"module","title":"PlausibleWeb.Live.Shields.Pages","doc":"LiveView for IP Addresses Shield","ref":"PlausibleWeb.Live.Shields.Pages.html"},{"type":"function","title":"PlausibleWeb.Live.Shields.Pages.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Shields.Pages.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Shields.Pages.mount/3","doc":"","ref":"PlausibleWeb.Live.Shields.Pages.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Shields.Pages.render/1","doc":"","ref":"PlausibleWeb.Live.Shields.Pages.html#render/1"},{"type":"module","title":"PlausibleWeb.Live.Sites","doc":"LiveView for sites index.","ref":"PlausibleWeb.Live.Sites.html"},{"type":"function","title":"PlausibleWeb.Live.Sites.ellipsis_menu/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#ellipsis_menu/1"},{"type":"function","title":"PlausibleWeb.Live.Sites.favicon/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#favicon/1"},{"type":"function","title":"PlausibleWeb.Live.Sites.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Sites.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Sites.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Sites.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Sites.handle_params/3","doc":"","ref":"PlausibleWeb.Live.Sites.html#handle_params/3"},{"type":"function","title":"PlausibleWeb.Live.Sites.icon_pin/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#icon_pin/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Sites.icon_pin/1","doc":"* Global attributes are accepted.","ref":"PlausibleWeb.Live.Sites.html#icon_pin/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Sites.invitation/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#invitation/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Sites.invitation/1","doc":"* `site` (`Plausible.Site`) (required)\n* `invitation` (`Plausible.Auth.Invitation`) (required)\n* `hourly_stats` (`:map`) (required)","ref":"PlausibleWeb.Live.Sites.html#invitation/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Sites.invitation_modal/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#invitation_modal/1"},{"type":"function","title":"PlausibleWeb.Live.Sites.mount/3","doc":"","ref":"PlausibleWeb.Live.Sites.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Sites.percentage_change/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#percentage_change/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Sites.percentage_change/1","doc":"* `change` (`:integer`) (required)","ref":"PlausibleWeb.Live.Sites.html#percentage_change/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Sites.render/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#render/1"},{"type":"function","title":"PlausibleWeb.Live.Sites.search_form/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#search_form/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Sites.search_form/1","doc":"* `filter_text` (`:string`) - Defaults to `\"\"`.\n* `uri` (`URI`) (required)","ref":"PlausibleWeb.Live.Sites.html#search_form/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Sites.site/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#site/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Sites.site/1","doc":"* `site` (`Plausible.Site`) (required)\n* `hourly_stats` (`:map`) (required)","ref":"PlausibleWeb.Live.Sites.html#site/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Sites.site_stats/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#site_stats/1"},{"type":"function","title":"Attributes - PlausibleWeb.Live.Sites.site_stats/1","doc":"* `hourly_stats` (`:map`) (required)","ref":"PlausibleWeb.Live.Sites.html#site_stats/1-attributes"},{"type":"function","title":"PlausibleWeb.Live.Sites.upgrade_nag_screen/1","doc":"","ref":"PlausibleWeb.Live.Sites.html#upgrade_nag_screen/1"},{"type":"module","title":"PlausibleWeb.Live.Verification","doc":"LiveView coordinating the site verification process.\nOnboarding new sites, renders a standalone component. \nEmbedded modal variant is available for general site settings.","ref":"PlausibleWeb.Live.Verification.html"},{"type":"function","title":"PlausibleWeb.Live.Verification.handle_event/3","doc":"","ref":"PlausibleWeb.Live.Verification.html#handle_event/3"},{"type":"function","title":"PlausibleWeb.Live.Verification.handle_info/2","doc":"","ref":"PlausibleWeb.Live.Verification.html#handle_info/2"},{"type":"function","title":"PlausibleWeb.Live.Verification.mount/3","doc":"","ref":"PlausibleWeb.Live.Verification.html#mount/3"},{"type":"function","title":"PlausibleWeb.Live.Verification.render/1","doc":"","ref":"PlausibleWeb.Live.Verification.html#render/1"},{"type":"module","title":"PlausibleWeb.MJML.StatsReport","doc":"MJML rendered for the weekly or monthly report e-mail","ref":"PlausibleWeb.MJML.StatsReport.html"},{"type":"function","title":"PlausibleWeb.MJML.StatsReport.debug_mjml_template/0","doc":"Returns the raw MJML template. Useful for debugging rendering issues.","ref":"PlausibleWeb.MJML.StatsReport.html#debug_mjml_template/0"},{"type":"function","title":"PlausibleWeb.MJML.StatsReport.render/1","doc":"Safely render the MJML template using Phoenix.HTML","ref":"PlausibleWeb.MJML.StatsReport.html#render/1"},{"type":"module","title":"PlausibleWeb.PageController","doc":"","ref":"PlausibleWeb.PageController.html"},{"type":"function","title":"PlausibleWeb.PageController.index/2","doc":"The root path is never accessible in Plausible.Cloud because it is handled by the upstream reverse proxy.\n\nThis controller action is only ever triggered in self-hosted Plausible.","ref":"PlausibleWeb.PageController.html#index/2"},{"type":"module","title":"PlausibleWeb.PageView","doc":"","ref":"PlausibleWeb.PageView.html"},{"type":"function","title":"PlausibleWeb.PageView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.PageView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.PageView.index.html/1","doc":"","ref":"PlausibleWeb.PageView.html#index.html/1"},{"type":"function","title":"PlausibleWeb.PageView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.PageView.html#render/2"},{"type":"function","title":"PlausibleWeb.PageView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.PageView.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities","doc":"Controller for Plugins API Capabilities - doesn't enforce authentication,\nserves as a comprehensive health check","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.index/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#index/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.open_api_operation/1","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#open_api_operation/1"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.shared_security/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#shared_security/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Capabilities.shared_tags/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Capabilities.html#shared_tags/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps","doc":"Controller for the CustomProp resource under Plugins API","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.disable/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#disable/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.enable/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#enable/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.open_api_operation/1","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#open_api_operation/1"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.shared_security/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#shared_security/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.CustomProps.shared_tags/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.CustomProps.html#shared_tags/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Controllers.Funnels","doc":"Controller for the Funnel resource under Plugins API","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.create/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#create/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.get/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#get/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.index/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#index/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.open_api_operation/1","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#open_api_operation/1"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.shared_security/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#shared_security/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Funnels.shared_tags/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Funnels.html#shared_tags/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Controllers.Goals","doc":"Controller for the Goal resource under Plugins API","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.create/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#create/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.delete/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#delete/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.delete_bulk/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#delete_bulk/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.get/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#get/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.index/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#index/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.open_api_operation/1","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#open_api_operation/1"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.shared_security/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#shared_security/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.Goals.shared_tags/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.Goals.html#shared_tags/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks","doc":"Controller for the Shared Link resource under Plugins API","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.create/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#create/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.get/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#get/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.index/2","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#index/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.open_api_operation/1","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#open_api_operation/1"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.shared_security/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#shared_security/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.shared_tags/0","doc":"","ref":"PlausibleWeb.Plugins.API.Controllers.SharedLinks.html#shared_tags/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Errors","doc":"Common responses for Plugins API","ref":"PlausibleWeb.Plugins.API.Errors.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Errors.error/3","doc":"","ref":"PlausibleWeb.Plugins.API.Errors.html#error/3"},{"type":"function","title":"PlausibleWeb.Plugins.API.Errors.unauthorized/1","doc":"","ref":"PlausibleWeb.Plugins.API.Errors.html#unauthorized/1"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Capabilities","doc":"OpenAPI schema for Capabilities","ref":"PlausibleWeb.Plugins.API.Schemas.Capabilities.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Capabilities.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Capabilities.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Capabilities.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Capabilities.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp","doc":"OpenAPI schema for Goal","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest","doc":"OpenAPI schema for Custom Property disable request","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.DisableRequest.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest","doc":"OpenAPI schema for Custom Property creation request","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.EnableRequest.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse","doc":"OpenAPI schema for SharedLink list response","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.CustomProp.ListResponse.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Error","doc":"OpenAPI schema for an error included in a response","ref":"PlausibleWeb.Plugins.API.Schemas.Error.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Error.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Error.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Error.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Error.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Funnel","doc":"OpenAPI schema for Funnel","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest","doc":"OpenAPI schema for Funnel creation request - get or creates goals along the way","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.CreateRequest.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse","doc":"OpenAPI schema for Funnel list response","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Funnel.ListResponse.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal","doc":"OpenAPI schema for Goal","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest","doc":"OpenAPI schema for Goal creation request","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent","doc":"OpenAPI schema for Custom Event Goal creation request","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.CustomEvent.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview","doc":"OpenAPI schema for Pageview Goal creation request","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Pageview.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue","doc":"OpenAPI schema for Custom Event Goal creation request","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CreateRequest.Revenue.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent","doc":"OpenAPI schema for Custom Event Goal object","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.CustomEvent.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest","doc":"OpenAPI schema for bulk Goal deletion request","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.DeleteBulkRequest.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse","doc":"OpenAPI schema for Goals list response","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.ListResponse.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview","doc":"OpenAPI schema for Pageview Goal object","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Pageview.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue","doc":"OpenAPI schema for Revenue Goal object","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Revenue.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Type","doc":"OpenAPI schema for common Goal Type\n\nFuture-proof: funnels etc.","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Goal.Type.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Link","doc":"OpenAPI Link schema","ref":"PlausibleWeb.Plugins.API.Schemas.Link.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Link.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Link.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Link.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Link.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.NotFound","doc":"OpenAPI schema for a generic 404 response","ref":"PlausibleWeb.Plugins.API.Schemas.NotFound.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.NotFound.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.NotFound.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.NotFound.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.NotFound.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata","doc":"Pagination metadata OpenAPI schema","ref":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.PaginationMetadata.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired","doc":"OpenAPI schema for a generic 402 response","ref":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.PaymentRequired.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink","doc":"OpenAPI schema for SharedLink object","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest","doc":"OpenAPI schema for SharedLink creation request","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.CreateRequest.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse","doc":"OpenAPI schema for SharedLink list response","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.SharedLink.ListResponse.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.Unauthorized","doc":"OpenAPI schema for a generic 401 response","ref":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.Unauthorized.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity","doc":"OpenAPI schema for a generic 422 response","ref":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.schema/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.html#schema/0"},{"type":"type","title":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.t/0","doc":"","ref":"PlausibleWeb.Plugins.API.Schemas.UnprocessableEntity.html#t:t/0"},{"type":"module","title":"PlausibleWeb.Plugins.API.Spec","doc":"OpenAPI specification for the Plugins API","ref":"PlausibleWeb.Plugins.API.Spec.html"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.Capabilities","doc":"View for rendering Capabilities on the Plugins API","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Capabilities.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Capabilities.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Capabilities.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.Capabilities.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.CustomProp","doc":"View for rendering Custom Props in the Plugins API","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.CustomProp.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.CustomProp.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.CustomProp.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.CustomProp.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.Error","doc":"View for rendering Plugins REST API errors","ref":"PlausibleWeb.Plugins.API.Views.Error.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Error.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.Error.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Error.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.Error.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Error.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.Error.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.Funnel","doc":"View for rendering Funnels in the Plugins API","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Funnel.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Funnel.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Funnel.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.Funnel.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.Goal","doc":"View for rendering Goals in the Plugins API","ref":"PlausibleWeb.Plugins.API.Views.Goal.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Goal.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.Goal.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Goal.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.Goal.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Goal.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.Goal.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.Pagination","doc":"A view capable of rendering pagination metadata included\nin responses containing lists of objects.","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Pagination.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Pagination.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Pagination.render_metadata_links/4","doc":"","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#render_metadata_links/4"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.Pagination.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.Pagination.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Plugins.API.Views.SharedLink","doc":"View for rendering Shared Links in the Plugins API","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.SharedLink.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.SharedLink.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html#render/2"},{"type":"function","title":"PlausibleWeb.Plugins.API.Views.SharedLink.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Plugins.API.Views.SharedLink.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.PluginsAPICase","doc":"This module defines the test case to be used by\ntests that require setting up a Plugins API connection.","ref":"PlausibleWeb.PluginsAPICase.html"},{"type":"module","title":"PlausibleWeb.Plugs.AuthorizePluginsAPI","doc":"Plug for Basic HTTP Authentication using\nPlugins API Tokens lookup.","ref":"PlausibleWeb.Plugs.AuthorizePluginsAPI.html"},{"type":"function","title":"PlausibleWeb.Plugs.AuthorizePluginsAPI.call/2","doc":"","ref":"PlausibleWeb.Plugs.AuthorizePluginsAPI.html#call/2"},{"type":"function","title":"PlausibleWeb.Plugs.AuthorizePluginsAPI.init/1","doc":"","ref":"PlausibleWeb.Plugs.AuthorizePluginsAPI.html#init/1"},{"type":"module","title":"PlausibleWeb.Plugs.AuthorizePublicAPI","doc":"Plug for authorizing access to Stats and Sites APIs.\n\nThe plug expects `:api_scope` to be provided in the assigns. The scope\nwill then be used to check for API key validity. The assign can be\nprovided in the router configuration in a following way:\n\n scope \"/api/v1/stats\", PlausibleWeb.Api, assigns: %{api_scope: \"some:scope:*\"} do\n pipe_through [:public_api, PlausibleWeb.Plugs.AuthorizePublicAPI]\n\n # route definitions follow\n # ...\n end\n\nThe scope from `:api_scope` is checked for match against all scopes from API key's\n`scopes` field. If the scope is among `@implicit_scopes`, it's considered to be\npresent for any valid API key. Scopes are checked for match by prefix, so if we have\n`some:scope:*` in matching route `:api_scope` and the API key has `some:*` in its\n`scopes` field, they will match.\n\nAfter a match is found, additional verification can be conducted, like in case of\n`stats:read:*`, where valid site ID is expected among parameters too.\n\nAll API requests are rate limited per API key, enforcing a given hourly request limit.","ref":"PlausibleWeb.Plugs.AuthorizePublicAPI.html"},{"type":"function","title":"PlausibleWeb.Plugs.AuthorizePublicAPI.call/2","doc":"","ref":"PlausibleWeb.Plugs.AuthorizePublicAPI.html#call/2"},{"type":"function","title":"PlausibleWeb.Plugs.AuthorizePublicAPI.init/1","doc":"","ref":"PlausibleWeb.Plugs.AuthorizePublicAPI.html#init/1"},{"type":"module","title":"PlausibleWeb.Plugs.ErrorHandler","doc":"A thin macro wrapper around Plug.ErrorHandler that adds Sentry context\n containing a readable support hash presented to the users.\n To be used in the user-facing APIs, so that we don't leak internal\n server errors.\n\n Usage: `use PlausibleWeb.Plugs.ErrorHandler`","ref":"PlausibleWeb.Plugs.ErrorHandler.html"},{"type":"module","title":"PlausibleWeb.Plugs.MaybeDisableRegistration","doc":"Plug toggling registration according to selfhosted state.","ref":"PlausibleWeb.Plugs.MaybeDisableRegistration.html"},{"type":"function","title":"PlausibleWeb.Plugs.MaybeDisableRegistration.call/2","doc":"","ref":"PlausibleWeb.Plugs.MaybeDisableRegistration.html#call/2"},{"type":"function","title":"PlausibleWeb.Plugs.MaybeDisableRegistration.init/1","doc":"","ref":"PlausibleWeb.Plugs.MaybeDisableRegistration.html#init/1"},{"type":"module","title":"PlausibleWeb.Plugs.NoRobots","doc":"Rejects bot requests by any means available.\n\nWe're adding `x-robots-tag` to the response header and annotate the conn\nwith \"noindex, nofollow\" under `private.robots` key.\n\nThe only exception is, if the request is trying to access our live demo\nat plausible.io/plausible.io - in which case we'll allow indexing, but deny\nfollowing links and skip the bot detection, in kind robots we trust.","ref":"PlausibleWeb.Plugs.NoRobots.html"},{"type":"module","title":"PlausibleWeb.Plugs.UserSessionTouch","doc":"Plug for bumping timeout on user session on every dashboard request.","ref":"PlausibleWeb.Plugs.UserSessionTouch.html"},{"type":"function","title":"PlausibleWeb.Plugs.UserSessionTouch.call/2","doc":"","ref":"PlausibleWeb.Plugs.UserSessionTouch.html#call/2"},{"type":"function","title":"PlausibleWeb.Plugs.UserSessionTouch.init/1","doc":"","ref":"PlausibleWeb.Plugs.UserSessionTouch.html#init/1"},{"type":"module","title":"PlausibleWeb.RefInspector","doc":"","ref":"PlausibleWeb.RefInspector.html"},{"type":"function","title":"PlausibleWeb.RefInspector.format_referrer/1","doc":"","ref":"PlausibleWeb.RefInspector.html#format_referrer/1"},{"type":"function","title":"PlausibleWeb.RefInspector.parse/1","doc":"","ref":"PlausibleWeb.RefInspector.html#parse/1"},{"type":"function","title":"PlausibleWeb.RefInspector.right_uri?/1","doc":"","ref":"PlausibleWeb.RefInspector.html#right_uri?/1"},{"type":"module","title":"PlausibleWeb.RemoteIP","doc":"Implements the strategy of retrieving client's remote IP","ref":"PlausibleWeb.RemoteIP.html"},{"type":"function","title":"PlausibleWeb.RemoteIP.get/1","doc":"","ref":"PlausibleWeb.RemoteIP.html#get/1"},{"type":"module","title":"PlausibleWeb.RequireAccountPlug","doc":"","ref":"PlausibleWeb.RequireAccountPlug.html"},{"type":"function","title":"PlausibleWeb.RequireAccountPlug.call/2","doc":"","ref":"PlausibleWeb.RequireAccountPlug.html#call/2"},{"type":"function","title":"PlausibleWeb.RequireAccountPlug.init/1","doc":"","ref":"PlausibleWeb.RequireAccountPlug.html#init/1"},{"type":"module","title":"PlausibleWeb.RequireLoggedOutPlug","doc":"","ref":"PlausibleWeb.RequireLoggedOutPlug.html"},{"type":"function","title":"PlausibleWeb.RequireLoggedOutPlug.call/2","doc":"","ref":"PlausibleWeb.RequireLoggedOutPlug.html#call/2"},{"type":"function","title":"PlausibleWeb.RequireLoggedOutPlug.init/1","doc":"","ref":"PlausibleWeb.RequireLoggedOutPlug.html#init/1"},{"type":"module","title":"PlausibleWeb.Router","doc":"","ref":"PlausibleWeb.Router.html"},{"type":"function","title":"PlausibleWeb.Router.api/2","doc":"","ref":"PlausibleWeb.Router.html#api/2"},{"type":"function","title":"PlausibleWeb.Router.app_layout/2","doc":"","ref":"PlausibleWeb.Router.html#app_layout/2"},{"type":"function","title":"PlausibleWeb.Router.browser/2","doc":"","ref":"PlausibleWeb.Router.html#browser/2"},{"type":"function","title":"PlausibleWeb.Router.call/2","doc":"Callback invoked by Plug on every request.","ref":"PlausibleWeb.Router.html#call/2"},{"type":"function","title":"PlausibleWeb.Router.csrf/2","doc":"","ref":"PlausibleWeb.Router.html#csrf/2"},{"type":"function","title":"PlausibleWeb.Router.external_api/2","doc":"","ref":"PlausibleWeb.Router.html#external_api/2"},{"type":"function","title":"PlausibleWeb.Router.flags/2","doc":"","ref":"PlausibleWeb.Router.html#flags/2"},{"type":"function","title":"PlausibleWeb.Router.init/1","doc":"Callback required by Plug that initializes the router\nfor serving web requests.","ref":"PlausibleWeb.Router.html#init/1"},{"type":"function","title":"PlausibleWeb.Router.internal_stats_api/2","doc":"","ref":"PlausibleWeb.Router.html#internal_stats_api/2"},{"type":"function","title":"PlausibleWeb.Router.kaffy_browser/2","doc":"","ref":"PlausibleWeb.Router.html#kaffy_browser/2"},{"type":"function","title":"PlausibleWeb.Router.plugins_api/2","doc":"","ref":"PlausibleWeb.Router.html#plugins_api/2"},{"type":"function","title":"PlausibleWeb.Router.plugins_api_auth/2","doc":"","ref":"PlausibleWeb.Router.html#plugins_api_auth/2"},{"type":"function","title":"PlausibleWeb.Router.public_api/2","doc":"","ref":"PlausibleWeb.Router.html#public_api/2"},{"type":"function","title":"PlausibleWeb.Router.shared_link/2","doc":"","ref":"PlausibleWeb.Router.html#shared_link/2"},{"type":"module","title":"PlausibleWeb.SessionTimeoutPlug","doc":"NOTE: This plug will be replaced with a different\nsession expiration mechanism once server-side persisted\nsessions are rolled out.","ref":"PlausibleWeb.SessionTimeoutPlug.html"},{"type":"function","title":"PlausibleWeb.SessionTimeoutPlug.call/2","doc":"","ref":"PlausibleWeb.SessionTimeoutPlug.html#call/2"},{"type":"function","title":"PlausibleWeb.SessionTimeoutPlug.init/1","doc":"","ref":"PlausibleWeb.SessionTimeoutPlug.html#init/1"},{"type":"module","title":"PlausibleWeb.Site.MembershipController","doc":"This controller deals with user management via the UI in Site Settings -> People. It's important to enforce permissions in this controller.\n\n Owner - Can manage users, can trigger a 'transfer ownership' request\n Admin - Can manage users\n Viewer - Can not access user management settings\n Anyone - Can accept invitations\n\n Everything else should be explicitly disallowed.","ref":"PlausibleWeb.Site.MembershipController.html"},{"type":"function","title":"PlausibleWeb.Site.MembershipController.invite_member/2","doc":"","ref":"PlausibleWeb.Site.MembershipController.html#invite_member/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipController.invite_member_form/2","doc":"","ref":"PlausibleWeb.Site.MembershipController.html#invite_member_form/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipController.remove_member/2","doc":"","ref":"PlausibleWeb.Site.MembershipController.html#remove_member/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipController.transfer_ownership/2","doc":"","ref":"PlausibleWeb.Site.MembershipController.html#transfer_ownership/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipController.transfer_ownership_form/2","doc":"","ref":"PlausibleWeb.Site.MembershipController.html#transfer_ownership_form/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipController.update_role/2","doc":"Updates the role of a user. The user being updated could be the same or different from the user taking\n the action. When updating the role, it's important to enforce permissions:\n\n Owner - Can update anyone's role except for themselves. If they want to change their own role, they have to use the 'transfer ownership' feature.\n Admin - Can update anyone's role except for owners. Can downgrade their own access to 'viewer'. Can promote a viewer to admin.","ref":"PlausibleWeb.Site.MembershipController.html#update_role/2"},{"type":"module","title":"PlausibleWeb.Site.MembershipView","doc":"","ref":"PlausibleWeb.Site.MembershipView.html"},{"type":"function","title":"PlausibleWeb.Site.MembershipView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.Site.MembershipView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.Site.MembershipView.invite_member_form.html/1","doc":"","ref":"PlausibleWeb.Site.MembershipView.html#invite_member_form.html/1"},{"type":"function","title":"PlausibleWeb.Site.MembershipView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.Site.MembershipView.html#render/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.Site.MembershipView.html#template_not_found/2"},{"type":"function","title":"PlausibleWeb.Site.MembershipView.transfer_ownership_form.html/1","doc":"","ref":"PlausibleWeb.Site.MembershipView.html#transfer_ownership_form.html/1"},{"type":"module","title":"PlausibleWeb.SiteController","doc":"","ref":"PlausibleWeb.SiteController.html"},{"type":"function","title":"PlausibleWeb.SiteController.add_monthly_report_recipient/2","doc":"","ref":"PlausibleWeb.SiteController.html#add_monthly_report_recipient/2"},{"type":"function","title":"PlausibleWeb.SiteController.add_traffic_change_notification_recipient/2","doc":"","ref":"PlausibleWeb.SiteController.html#add_traffic_change_notification_recipient/2"},{"type":"function","title":"PlausibleWeb.SiteController.add_weekly_report_recipient/2","doc":"","ref":"PlausibleWeb.SiteController.html#add_weekly_report_recipient/2"},{"type":"function","title":"PlausibleWeb.SiteController.change_domain/2","doc":"","ref":"PlausibleWeb.SiteController.html#change_domain/2"},{"type":"function","title":"PlausibleWeb.SiteController.change_domain_submit/2","doc":"","ref":"PlausibleWeb.SiteController.html#change_domain_submit/2"},{"type":"function","title":"PlausibleWeb.SiteController.create_shared_link/2","doc":"","ref":"PlausibleWeb.SiteController.html#create_shared_link/2"},{"type":"function","title":"PlausibleWeb.SiteController.create_site/2","doc":"","ref":"PlausibleWeb.SiteController.html#create_site/2"},{"type":"function","title":"PlausibleWeb.SiteController.csv_import/2","doc":"","ref":"PlausibleWeb.SiteController.html#csv_import/2"},{"type":"function","title":"PlausibleWeb.SiteController.delete_google_auth/2","doc":"","ref":"PlausibleWeb.SiteController.html#delete_google_auth/2"},{"type":"function","title":"PlausibleWeb.SiteController.delete_shared_link/2","doc":"","ref":"PlausibleWeb.SiteController.html#delete_shared_link/2"},{"type":"function","title":"PlausibleWeb.SiteController.delete_site/2","doc":"","ref":"PlausibleWeb.SiteController.html#delete_site/2"},{"type":"function","title":"PlausibleWeb.SiteController.disable_monthly_report/2","doc":"","ref":"PlausibleWeb.SiteController.html#disable_monthly_report/2"},{"type":"function","title":"PlausibleWeb.SiteController.disable_traffic_change_notification/2","doc":"","ref":"PlausibleWeb.SiteController.html#disable_traffic_change_notification/2"},{"type":"function","title":"PlausibleWeb.SiteController.disable_weekly_report/2","doc":"","ref":"PlausibleWeb.SiteController.html#disable_weekly_report/2"},{"type":"function","title":"PlausibleWeb.SiteController.download_export/2","doc":"","ref":"PlausibleWeb.SiteController.html#download_export/2"},{"type":"function","title":"PlausibleWeb.SiteController.edit_shared_link/2","doc":"","ref":"PlausibleWeb.SiteController.html#edit_shared_link/2"},{"type":"function","title":"PlausibleWeb.SiteController.enable_monthly_report/2","doc":"","ref":"PlausibleWeb.SiteController.html#enable_monthly_report/2"},{"type":"function","title":"PlausibleWeb.SiteController.enable_traffic_change_notification/2","doc":"","ref":"PlausibleWeb.SiteController.html#enable_traffic_change_notification/2"},{"type":"function","title":"PlausibleWeb.SiteController.enable_weekly_report/2","doc":"","ref":"PlausibleWeb.SiteController.html#enable_weekly_report/2"},{"type":"function","title":"PlausibleWeb.SiteController.forget_import/2","doc":"","ref":"PlausibleWeb.SiteController.html#forget_import/2"},{"type":"function","title":"PlausibleWeb.SiteController.forget_imported/2","doc":"","ref":"PlausibleWeb.SiteController.html#forget_imported/2"},{"type":"function","title":"PlausibleWeb.SiteController.make_private/2","doc":"","ref":"PlausibleWeb.SiteController.html#make_private/2"},{"type":"function","title":"PlausibleWeb.SiteController.make_public/2","doc":"","ref":"PlausibleWeb.SiteController.html#make_public/2"},{"type":"function","title":"PlausibleWeb.SiteController.new/2","doc":"","ref":"PlausibleWeb.SiteController.html#new/2"},{"type":"function","title":"PlausibleWeb.SiteController.new_shared_link/2","doc":"","ref":"PlausibleWeb.SiteController.html#new_shared_link/2"},{"type":"function","title":"PlausibleWeb.SiteController.remove_monthly_report_recipient/2","doc":"","ref":"PlausibleWeb.SiteController.html#remove_monthly_report_recipient/2"},{"type":"function","title":"PlausibleWeb.SiteController.remove_traffic_change_notification_recipient/2","doc":"","ref":"PlausibleWeb.SiteController.html#remove_traffic_change_notification_recipient/2"},{"type":"function","title":"PlausibleWeb.SiteController.remove_weekly_report_recipient/2","doc":"","ref":"PlausibleWeb.SiteController.html#remove_weekly_report_recipient/2"},{"type":"function","title":"PlausibleWeb.SiteController.reset_stats/2","doc":"","ref":"PlausibleWeb.SiteController.html#reset_stats/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_danger_zone/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_danger_zone/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_email_reports/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_email_reports/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_funnels/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_funnels/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_general/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_general/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_goals/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_goals/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_imports_exports/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_imports_exports/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_integrations/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_integrations/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_people/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_people/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_props/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_props/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_shields/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_shields/2"},{"type":"function","title":"PlausibleWeb.SiteController.settings_visibility/2","doc":"","ref":"PlausibleWeb.SiteController.html#settings_visibility/2"},{"type":"function","title":"PlausibleWeb.SiteController.update_feature_visibility/2","doc":"","ref":"PlausibleWeb.SiteController.html#update_feature_visibility/2"},{"type":"function","title":"PlausibleWeb.SiteController.update_google_auth/2","doc":"","ref":"PlausibleWeb.SiteController.html#update_google_auth/2"},{"type":"function","title":"PlausibleWeb.SiteController.update_settings/2","doc":"","ref":"PlausibleWeb.SiteController.html#update_settings/2"},{"type":"function","title":"PlausibleWeb.SiteController.update_shared_link/2","doc":"","ref":"PlausibleWeb.SiteController.html#update_shared_link/2"},{"type":"function","title":"PlausibleWeb.SiteController.update_traffic_change_notification/2","doc":"","ref":"PlausibleWeb.SiteController.html#update_traffic_change_notification/2"},{"type":"module","title":"PlausibleWeb.SiteView","doc":"","ref":"PlausibleWeb.SiteView.html"},{"type":"function","title":"PlausibleWeb.SiteView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.SiteView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.SiteView.change_domain.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#change_domain.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.csv_import.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#csv_import.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.edit_shared_link.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#edit_shared_link.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.new.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#new.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.new_shared_link.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#new_shared_link.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.plausible_url/0","doc":"","ref":"PlausibleWeb.SiteView.html#plausible_url/0"},{"type":"function","title":"PlausibleWeb.SiteView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.SiteView.html#render/2"},{"type":"function","title":"PlausibleWeb.SiteView.render_snippet/1","doc":"","ref":"PlausibleWeb.SiteView.html#render_snippet/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_danger_zone.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_danger_zone.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_email_reports.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_email_reports.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_funnels.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_funnels.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_general.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_general.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_goals.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_goals.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_imports_exports.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_imports_exports.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_integrations.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_integrations.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_people.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_people.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_props.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_props.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_search_console.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_search_console.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_shields.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_shields.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.settings_visibility.html/1","doc":"","ref":"PlausibleWeb.SiteView.html#settings_visibility.html/1"},{"type":"function","title":"PlausibleWeb.SiteView.shared_link_dest/2","doc":"","ref":"PlausibleWeb.SiteView.html#shared_link_dest/2"},{"type":"function","title":"PlausibleWeb.SiteView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.SiteView.html#template_not_found/2"},{"type":"function","title":"PlausibleWeb.SiteView.traffic_change_form/1","doc":"","ref":"PlausibleWeb.SiteView.html#traffic_change_form/1"},{"type":"function","title":"PlausibleWeb.SiteView.with_indefinite_article/1","doc":"","ref":"PlausibleWeb.SiteView.html#with_indefinite_article/1"},{"type":"module","title":"PlausibleWeb.StatsController","doc":"This controller is responsible for rendering stats dashboards.\n\nThe stats dashboards are currently the only part of the app that uses client-side\nrendering. Since the dashboards are heavily interactive, they are built with React\nwhich is an appropriate choice for highly interactive browser UIs.\n\n \nsequenceDiagram\n Browser->>StatsController: GET /mydomain.com\n StatsController-->>Browser: StatsView.render(\"stats.html\")\n Note left of Browser: ReactDom.render(Dashboard)\n\n Browser -) Api.StatsController: GET /api/stats/mydomain.com/top-stats\n Api.StatsController --) Browser: {\"top_stats\": [...]}\n Note left of Browser: TopStats.render()\n\n Browser -) Api.StatsController: GET /api/stats/mydomain.com/main-graph\n Api.StatsController --) Browser: [{\"plot\": [...], \"labels\": [...]}, ...]\n Note left of Browser: VisitorGraph.render()\n\n Browser -) Api.StatsController: GET /api/stats/mydomain.com/sources\n Api.StatsController --) Browser: [{\"name\": \"Google\", \"visitors\": 292150}, ...]\n Note left of Browser: Sources.render()\n\n Note over Browser,StatsController: And so on, for all reports in the viewport\n \n\nThis reasoning for this sequence is as follows:\n 1. First paint is fast because it doesn't do any data aggregation yet - good UX\n 2. The basic structure of the dashboard is rendered with spinners before reports are ready - good UX\n 2. Rendering on the frontend allows for maximum interactivity. Re-rendering and re-fetching can be as granular as needed.\n 3. Routing on the frontend allows the user to navigate the dashboard without reloading the page and losing context\n 4. Rendering on the frontend allows caching results in the browser to reduce pressure on backends and storage\n 3.1 No client-side caching has been implemented yet. This is still theoretical. See https://github.com/plausible/analytics/discussions/1278\n 3.2 This is a big potential opportunity, because analytics data is mostly immutable. Clients can cache all historical data.\n 5. Since frontend rendering & navigation is harder to build and maintain than regular server-rendered HTML, we don't use SPA-style rendering anywhere else\n .The only place currently where the benefits outweigh the costs is the dashboard.","ref":"PlausibleWeb.StatsController.html"},{"type":"function","title":"PlausibleWeb.StatsController.authenticate_shared_link/2","doc":"","ref":"PlausibleWeb.StatsController.html#authenticate_shared_link/2"},{"type":"function","title":"PlausibleWeb.StatsController.csv_export/2","doc":"The export is limited to 300 entries for other reports and 100 entries for pages because bigger result sets\nstart causing failures. Since we request data like time on page or bounce_rate for pages in a separate query\nusing the IN filter, it causes the requests to balloon in payload size.","ref":"PlausibleWeb.StatsController.html#csv_export/2"},{"type":"function","title":"PlausibleWeb.StatsController.shared_link/2","doc":"Authorizes and renders a shared link:\n 1. Shared link with no password protection: needs to just make sure the shared link entry is still\n in our database. This check makes sure shared link access can be revoked by the site admins. If the\n shared link exists, render it directly.\n\n 2. Shared link with password protection: Same checks as without the password, but an extra step is taken to\n protect the page with a password. When the user passes the password challenge, a cookie is set with Plausible.Auth.Token.sign_shared_link().\n The cookie allows the user to access the dashboard for 24 hours without entering the password again.\n\n #","ref":"PlausibleWeb.StatsController.html#shared_link/2"},{"type":"function","title":"Backwards compatibility - PlausibleWeb.StatsController.shared_link/2","doc":"The URL format for shared links was changed in [this pull request](https://github.com/plausible/analytics/pull/752) in order\n to make the URLs easier to bookmark. The old format is supported along with the new in order to not break old links.\n\n See: https://plausible.io/docs/shared-links","ref":"PlausibleWeb.StatsController.html#shared_link/2-backwards-compatibility"},{"type":"function","title":"PlausibleWeb.StatsController.stats/2","doc":"","ref":"PlausibleWeb.StatsController.html#stats/2"},{"type":"module","title":"PlausibleWeb.StatsView","doc":"","ref":"PlausibleWeb.StatsView.html"},{"type":"function","title":"PlausibleWeb.StatsView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.StatsView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.StatsView.large_number_format/1","doc":"","ref":"PlausibleWeb.StatsView.html#large_number_format/1"},{"type":"function","title":"PlausibleWeb.StatsView.plausible_url/0","doc":"","ref":"PlausibleWeb.StatsView.html#plausible_url/0"},{"type":"function","title":"PlausibleWeb.StatsView.pretty_stats_url/1","doc":"Returns a readable stats URL.\n\nNative Phoenix router functions percent-encode all diacritics, resulting in\nugly URLs, e.g. `https://plausible.io/café.com` transforms into\n`https://plausible.io/caf%C3%A9.com`.\n\nThis function encodes only the slash (`/`) character from the site's domain.","ref":"PlausibleWeb.StatsView.html#pretty_stats_url/1"},{"type":"function","title":"Examples - PlausibleWeb.StatsView.pretty_stats_url/1","doc":"iex> PlausibleWeb.StatsView.pretty_stats_url(%Plausible.Site{domain: \"user.gittea.io/repo\"})\n \"http://localhost:8000/user.gittea.io%2Frepo\"\n\n iex> PlausibleWeb.StatsView.pretty_stats_url(%Plausible.Site{domain: \"anakin.test\"})\n \"http://localhost:8000/anakin.test\"\n\n iex> PlausibleWeb.StatsView.pretty_stats_url(%Plausible.Site{domain: \"café.test\"})\n \"http://localhost:8000/café.test\"","ref":"PlausibleWeb.StatsView.html#pretty_stats_url/1-examples"},{"type":"function","title":"PlausibleWeb.StatsView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.StatsView.html#render/2"},{"type":"function","title":"PlausibleWeb.StatsView.shared_link_password.html/1","doc":"","ref":"PlausibleWeb.StatsView.html#shared_link_password.html/1"},{"type":"function","title":"PlausibleWeb.StatsView.site_locked.html/1","doc":"","ref":"PlausibleWeb.StatsView.html#site_locked.html/1"},{"type":"function","title":"PlausibleWeb.StatsView.stats.html/1","doc":"","ref":"PlausibleWeb.StatsView.html#stats.html/1"},{"type":"function","title":"PlausibleWeb.StatsView.stats_container_class/1","doc":"","ref":"PlausibleWeb.StatsView.html#stats_container_class/1"},{"type":"function","title":"PlausibleWeb.StatsView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.StatsView.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.Tracker","doc":"","ref":"PlausibleWeb.Tracker.html"},{"type":"function","title":"PlausibleWeb.Tracker.call/2","doc":"","ref":"PlausibleWeb.Tracker.html#call/2"},{"type":"function","title":"PlausibleWeb.Tracker.child_spec/1","doc":"Returns a specification to start this module under a supervisor.\n\nSee `Supervisor`.","ref":"PlausibleWeb.Tracker.html#child_spec/1"},{"type":"function","title":"PlausibleWeb.Tracker.init/1","doc":"","ref":"PlausibleWeb.Tracker.html#init/1"},{"type":"module","title":"PlausibleWeb.TwoFactor.Session","doc":"Functions for managing session data related to Two-Factor\nAuthentication.","ref":"PlausibleWeb.TwoFactor.Session.html"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.clear_2fa_user/1","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#clear_2fa_user/1"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.clear_remember_2fa/1","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#clear_remember_2fa/1"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.get_2fa_user/1","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#get_2fa_user/1"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.maybe_set_remember_2fa/3","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#maybe_set_remember_2fa/3"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.remember_2fa?/2","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#remember_2fa?/2"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.remember_2fa_days/0","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#remember_2fa_days/0"},{"type":"function","title":"PlausibleWeb.TwoFactor.Session.set_2fa_user/2","doc":"","ref":"PlausibleWeb.TwoFactor.Session.html#set_2fa_user/2"},{"type":"module","title":"PlausibleWeb.UnsubscribeController","doc":"","ref":"PlausibleWeb.UnsubscribeController.html"},{"type":"function","title":"PlausibleWeb.UnsubscribeController.monthly_report/2","doc":"","ref":"PlausibleWeb.UnsubscribeController.html#monthly_report/2"},{"type":"function","title":"PlausibleWeb.UnsubscribeController.weekly_report/2","doc":"","ref":"PlausibleWeb.UnsubscribeController.html#weekly_report/2"},{"type":"module","title":"PlausibleWeb.UnsubscribeView","doc":"","ref":"PlausibleWeb.UnsubscribeView.html"},{"type":"function","title":"PlausibleWeb.UnsubscribeView.__resource__/0","doc":"The resource name, as an atom, for this view","ref":"PlausibleWeb.UnsubscribeView.html#__resource__/0"},{"type":"function","title":"PlausibleWeb.UnsubscribeView.render/2","doc":"Renders the given template locally.","ref":"PlausibleWeb.UnsubscribeView.html#render/2"},{"type":"function","title":"PlausibleWeb.UnsubscribeView.success.html/1","doc":"","ref":"PlausibleWeb.UnsubscribeView.html#success.html/1"},{"type":"function","title":"PlausibleWeb.UnsubscribeView.template_not_found/2","doc":"Callback invoked when no template is found.\nBy default it raises but can be customized\nto render a particular template.","ref":"PlausibleWeb.UnsubscribeView.html#template_not_found/2"},{"type":"module","title":"PlausibleWeb.UserAuth","doc":"Functions for user session management.\n\nIn it's current shape, both current (legacy) and soon to be implemented (new)\nuser sessions are supported side by side.\n\nThe legacy token is still accepted from the session cookie. Once 14 days\npass (the current time window for which session cookie is valid without\nany activity), the legacy cookies won't be accepted anymore (legacy token\nretrieval is tracked with logging) and the logic will be cleaned of branching\nfor legacy session.","ref":"PlausibleWeb.UserAuth.html"},{"type":"function","title":"PlausibleWeb.UserAuth.convert_legacy_session/1","doc":"","ref":"PlausibleWeb.UserAuth.html#convert_legacy_session/1"},{"type":"function","title":"PlausibleWeb.UserAuth.get_user_session/1","doc":"","ref":"PlausibleWeb.UserAuth.html#get_user_session/1"},{"type":"function","title":"PlausibleWeb.UserAuth.log_in_user/3","doc":"","ref":"PlausibleWeb.UserAuth.html#log_in_user/3"},{"type":"function","title":"PlausibleWeb.UserAuth.log_out_user/1","doc":"","ref":"PlausibleWeb.UserAuth.html#log_out_user/1"},{"type":"function","title":"PlausibleWeb.UserAuth.set_logged_in_cookie/1","doc":"Sets the `logged_in` cookie share with the static site for determining\nwhether client is authenticated.\n\nAs it's a separate cookie, there's a chance it might fall out of sync\nwith session cookie state due to manual deletion or premature expiration.","ref":"PlausibleWeb.UserAuth.html#set_logged_in_cookie/1"},{"type":"function","title":"PlausibleWeb.UserAuth.touch_user_session/1","doc":"","ref":"PlausibleWeb.UserAuth.html#touch_user_session/1"},{"type":"function","title":"PlausibleWeb.UserAuth.touch_user_session/2","doc":"","ref":"PlausibleWeb.UserAuth.html#touch_user_session/2"},{"type":"extras","title":"Introduction","doc":"# Plausible Analytics\n\n \n \n \n \n \n \n Simple Metrics |\n Lightweight Script |\n Privacy Focused |\n Open Source |\n Docs |\n Contributing \n \n \n\n[Plausible Analytics](https://plausible.io/) is an easy to use, lightweight (< 1 KB), open source and privacy-friendly alternative to Google Analytics. It doesn’t use cookies and is fully compliant with GDPR, CCPA and PECR. You can self-host Plausible Community Edition or have us manage Plausible Analytics for you in the cloud. Here's [the live demo of our own website stats](https://plausible.io/plausible.io). Made and hosted in the EU 🇪🇺\n\nWe are dedicated to making web analytics more privacy-friendly. Our mission is to reduce corporate surveillance by providing an alternative web analytics tool which doesn’t come from the AdTech world. We are completely independent and solely funded by our subscribers.\n\n![Plausible Analytics](https://plausible.io/docs/img/plausible-analytics.png)","ref":"readme.html"},{"type":"extras","title":"Why Plausible? - Introduction","doc":"Here's what makes Plausible a great Google Analytics alternative and why we're trusted by 12,000+ paying subscribers to deliver their website and business insights:\n\n- **Clutter Free**: Plausible Analytics provides [simple web analytics](https://plausible.io/simple-web-analytics) and it cuts through the noise. No layers of menus, no need for custom reports. Get all the important insights on one single page. No training necessary.\n- **GDPR/CCPA/PECR compliant**: Measure traffic, not individuals. No personal data or IP addresses are ever stored in our database. We don't use cookies or any other persistent identifiers. [Read more about our data policy](https://plausible.io/data-policy)\n- **Lightweight**: Plausible Analytics works by loading a script on your website, like Google Analytics. Our script is [45x smaller](https://plausible.io/lightweight-web-analytics), making your website quicker to load. You can also send events directly to our [events API](https://plausible.io/docs/events-api).\n- **Email or Slack reports**: Keep an eye on your traffic with weekly and/or monthly email or Slack reports. You can also get traffic spike notifications.\n- **Invite team members and share stats**: You have the option to be transparent and open your web analytics to everyone. Your website stats are private by default but you can choose to make them public so anyone with your custom link can view them. You can [invite team members](https://plausible.io/docs/users-roles) and assign user roles too.\n- **Define key goals and track conversions**: Create custom events with custom dimensions to track conversions and attribution to understand and identify the trends that matter. Includes easy ways to track outbound link clicks, file downloads and 404 error pages.\n- **Search keywords**: Integrate your dashboard with Google Search Console to get the most accurate reporting on your search keywords.\n- **SPA support**: Plausible is built with modern web frameworks in mind and it works automatically with any pushState based router on the frontend. We also support frameworks that use the URL hash for routing. See [our documentation](https://plausible.io/docs/hash-based-routing).\n- **Smooth transition from Google Analytics**: There's a realtime dashboard, entry pages report and integration with Search Console. You can track your paid campaigns and conversions. You can invite team members. You can even [import your historical Google Analytics stats](https://plausible.io/docs/google-analytics-import). Learn how to [get the most out of your Plausible experience](https://plausible.io/docs/your-plausible-experience) and join thousands who have already migrated from Google Analytics.\n\nInterested to learn more? [Read more on our website](https://plausible.io), learn more about the team and the goals of the project on [our about page](https://plausible.io/about) or explore [the documentation](https://plausible.io/docs).","ref":"readme.html#why-plausible"},{"type":"extras","title":"Why is Plausible Analytics Cloud not free like Google Analytics? - Introduction","doc":"Plausible Analytics is an independently owned and actively developed project. To keep the project development going, to stay in business, to continue putting effort into building a better product and to cover our costs, we need to charge a fee.\n\nGoogle Analytics is free because Google has built their company and their wealth by collecting and analyzing huge amounts of personal information from web users and using these personal and behavioral insights to sell advertisements.\n\nPlausible has no part in that business model. No personal data is being collected and analyzed either. With Plausible, you 100% own and control all of your website data. This data is not being shared with or sold to any third-parties.\n\nWe choose the subscription business model rather than the business model of surveillance capitalism. See reasons why we believe you should [stop using Google Analytics on your website](https://plausible.io/blog/remove-google-analytics).","ref":"readme.html#why-is-plausible-analytics-cloud-not-free-like-google-analytics"},{"type":"extras","title":"Getting started with Plausible - Introduction","doc":"The easiest way to get started with Plausible Analytics is with [our official managed service in the cloud](https://plausible.io/#pricing). It takes 2 minutes to start counting your stats with a worldwide CDN, high availability, backups, security and maintenance all done for you by us.\n\nIn order to be compliant with the GDPR and the Schrems II ruling, all visitor data for our managed service in the cloud is exclusively processed on servers and cloud infrastructure owned and operated by European providers. Your website data never leaves the EU.\n\nOur managed hosting can save a substantial amount of developer time and resources. For most sites this ends up being the best value option and the revenue goes to funding the maintenance and further development of Plausible. So you’ll be supporting open source software and getting a great service!\n\n#","ref":"readme.html#getting-started-with-plausible"},{"type":"extras","title":"Can Plausible be self-hosted? - Introduction","doc":"Plausible is [open source web analytics](https://plausible.io/open-source-website-analytics) and we have a free as in beer and self-hosted solution called [Plausible Community Edition (CE)](https://plausible.io/self-hosted-web-analytics). Here are the differences between Plausible Analytics managed hosting in the cloud and the Plausible CE:\n\n| | Plausible Analytics Cloud | Plausible Community Edition |\n| ------------- | ------------- | ------------- |\n| **Infrastructure management** | Easy and convenient. It takes 2 minutes to start counting your stats with a worldwide CDN, high availability, backups, security and maintenance all done for you by us. We manage everything so you don’t have to worry about anything and can focus on your stats. | You do it all yourself. You need to get a server and you need to manage your infrastructure. You are responsible for installation, maintenance, upgrades, server capacity, uptime, backup, security, stability, consistency, loading time and so on.|\n| **Release schedule** | Continuously developed and improved with new features and updates multiple times per week. | [It's a long term release](https://plausible.io/blog/building-open-source) published twice per year so latest features and improvements won't be immediately available.|\n| **Premium features** | All features available as listed in [our pricing plans](https://plausible.io/#pricing). | Selected premium features such as funnels and ecommerce revenue goals are not available as we aim to ensure a [protective barrier around our cloud offering](https://plausible.io/blog/community-edition).|\n| **Bot filtering** | Advanced bot filtering for more accurate stats. Our algorithm detects and excludes non-human traffic patterns. We also exclude known bots by the User-Agent header and filter out traffic from data centers and referrer spam domains. | Basic bot filtering that targets the most common non-human traffic based on the User-Agent header and referrer spam domains.|\n| **Server location** | All visitor data is exclusively processed on EU-owned cloud infrastructure. We keep your site data on a secure, encrypted and green energy powered server in Germany. This ensures that your site data is protected by the strict European Union data privacy laws and ensures compliance with GDPR. Your website data never leaves the EU. | You have full control and can host your instance on any server in any country that you wish. Host it on a server in your basement or host it with any cloud provider wherever you want, even those that are not GDPR compliant.|\n| **Data portability** | You see all your site stats and metrics on our modern-looking, simple to use and fast loading dashboard. You can only see the stats aggregated in the dashboard. You can download the stats using the [CSV export](https://plausible.io/docs/export-stats), [stats API](https://plausible.io/docs/stats-api) or tools such as the [Data Studio Connector](https://plausible.io/docs/integration-guides#google-data-studio). | Do you want access to the raw data? Self-hosting gives you that option. You can take the data directly from the ClickHouse database. |\n| **Premium support** | Real support delivered by real human beings who build and maintain Plausible. | Premium support is not included. CE is community supported only.|\n| **Costs** | There's a cost associated with providing an analytics service so we charge a subscription fee. We choose the subscription business model rather than the business model of surveillance capitalism. Your money funds further development of Plausible. | You need to pay for your server, CDN, backups and whatever other cost there is associated with running the infrastructure. You never have to pay any fees to us. Your money goes to 3rd party companies with no connection to us.|\n\nInterested in self-hosting Plausible CE on your server? Take a look at our [Plausible CE installation instructions](https://github.com/plausible/community-edition/).\n\nPlausible CE is a community supported project and there are no guarantees that you will get support from the creators of Plausible to troubleshoot your self-hosting issues. There is a [community supported forum](https://github.com/plausible/analytics/discussions/categories/self-hosted-support) where you can ask for help.\n\nOur only source of funding is our premium, managed service for running Plausible in the cloud.","ref":"readme.html#can-plausible-be-self-hosted"},{"type":"extras","title":"Technology - Introduction","doc":"Plausible Analytics is a standard Elixir/Phoenix application backed by a PostgreSQL database for general data and a Clickhouse\ndatabase for stats. On the frontend we use [TailwindCSS](https://tailwindcss.com/) for styling and React to make the dashboard interactive.","ref":"readme.html#technology"},{"type":"extras","title":"Contributors - Introduction","doc":"For anyone wishing to contribute to Plausible, we recommend taking a look at [our contributor guide](https://github.com/plausible/analytics/blob/master/CONTRIBUTING.md).","ref":"readme.html#contributors"},{"type":"extras","title":"Feedback & Roadmap - Introduction","doc":"We welcome feedback from our community. We have a public roadmap driven by the features suggested by the community members. Take a look at our [feedback board](https://plausible.io/feedback). Please let us know if you have any requests and vote on open issues so we can better prioritize.\n\nTo stay up to date with all the latest news and product updates, make sure to follow us on [X (formerly Twitter)](https://twitter.com/plausiblehq), [LinkedIn](https://www.linkedin.com/company/plausible-analytics/) or [Mastodon](https://fosstodon.org/@plausible).","ref":"readme.html#feedback-roadmap"},{"type":"extras","title":"License & Trademarks - Introduction","doc":"Plausible CE is open source under the GNU Affero General Public License Version 3 (AGPLv3) or any later version. You can [find it here](https://github.com/plausible/analytics/blob/master/LICENSE.md).\n\nTo avoid issues with AGPL virality, we've released the JavaScript tracker which gets included on your website under the MIT license. You can [find it here](https://github.com/plausible/analytics/blob/master/tracker/LICENSE.md).\n\nCopyright (c) 2018-present Plausible Insights OÜ. Plausible Analytics name and logo are trademarks of Plausible Insights OÜ. Please see our [trademark guidelines](https://plausible.io/trademark) for info on acceptable usage.","ref":"readme.html#license-trademarks"},{"type":"extras","title":"Contributing","doc":"# Contributing\n\nWe welcome everyone to contribute to Plausible. This document is to help you on setting up your environment, finding a task, and opening pull requests.","ref":"contributing.html"},{"type":"extras","title":"Development setup - Contributing","doc":"The easiest way to get up and running is to [install](https://docs.docker.com/get-docker/) and use Docker for running both Postgres and Clickhouse.\n\nMake sure Docker, Elixir, Erlang and Node.js are all installed on your development machine. The [`.tool-versions`](https://github.com/plausible/analytics/blob/master/.tool-versions) file is available to use with [asdf](https://github.com/asdf-vm/asdf) or similar tools.\n\n#","ref":"contributing.html#development-setup"},{"type":"extras","title":"Start the environment - Contributing","doc":"1. Run both `make postgres` and `make clickhouse`.\n2. You can set up everything with `make install`, alternatively run each command separately:\n 1. Run `mix deps.get`. This will download the required Elixir dependencies.\n 2. Run `mix ecto.create`. This will create the required databases in both Postgres and Clickhouse.\n 3. Run `mix ecto.migrate` to build the database schema.\n 4. Run `mix run priv/repo/seeds.exs` to seed the database. Check the [Seeds](#Seeds) section for more.\n 5. Run `npm ci --prefix assets` to install the required client-side dependencies.\n 6. Run `npm ci --prefix tracker` to install the required tracker dependencies.\n 7. Run `mix assets.setup` to install Tailwind and Esbuild\n 8. Run `npm run deploy --prefix tracker` to generate tracker files in `priv/tracker/js`\n 9. Run `mix download_country_database` to fetch geolocation database\n3. Run `make server` or `mix phx.server` to start the Phoenix server.\n4. The system is now available on `localhost:8000`.\n\n#","ref":"contributing.html#start-the-environment"},{"type":"extras","title":"Seeds - Contributing","doc":"You can optionally seed your database to automatically create an account and a site with stats:\n\n1. Run `mix run priv/repo/seeds.exs` to seed the database.\n2. Start the server with `make server` and navigate to `http://localhost:8000/login`.\n3. Log in with the following e-mail and password combination: `user@plausible.test` and `plausible`.\n4. You should now have a `dummy.site` site with generated stats.\n\nAlternatively, you can manually create a new account:\n\n1. Navigate to `http://localhost:8000/register` and fill in the form.\n2. Fill in the rest of the forms and for the domain use `dummy.site`\n3. Skip the JS snippet and click start collecting data.\n4. Run `mix send_pageview` from the terminal to generate a fake pageview event for the dummy site.\n5. You should now be all set!\n\n#","ref":"contributing.html#seeds"},{"type":"extras","title":"Stopping Docker containers - Contributing","doc":"1. Stop and remove the Postgres container with `make postgres-stop`.\n2. Stop and remove the Clickhouse container with `make clickhouse-stop`.\n\nVolumes are preserved. You'll find that the Postgres and Clickhouse state are retained when you bring them up again the next time: no need to re-register and so on.\n\nNote: Since we are deleting the containers, be careful when deleting volumes with `docker volume prune`. You might accidentally delete the database and would have to go through re-registration process.\n\n#","ref":"contributing.html#stopping-docker-containers"},{"type":"extras","title":"Pre-commit hooks - Contributing","doc":"`pre-commit` requires Python to be available locally and covers Elixir, JavaScript, and CSS. Set up with `pip install --user pre-commit` followed by `pre-commit install`. Conversely, if the prompts are far too bothersome, remove with `pre-commit uninstall`.","ref":"contributing.html#pre-commit-hooks"},{"type":"extras","title":"Finding a task - Contributing","doc":"Bugs can be found in our [issue tracker](https://github.com/plausible/analytics/issues). Issues are usually up for grabs.\n\nNew features need to be discussed with the core team and the community first. If you're tackling a feature, please make sure it has been already discussed in the [Discussions tab](https://github.com/plausible/analytics/discussions). We kindly ask contributors to use the discussion comment section to propose a solution before opening a pull request.\n\nPull requests without an associated issue or discussion may still be merged, but we will focus on changes that have already been talked through.","ref":"contributing.html#finding-a-task"}],"content_type":"text/markdown"} \ No newline at end of file diff --git a/search.html b/search.html index 1aed970d92..f09cde09ea 100644 --- a/search.html +++ b/search.html @@ -138,7 +138,7 @@ Pages

    - +

  • filters - optional filters to drill down data. See the Stats API "Filtering" section for more details.