diff options
| -rw-r--r-- | CHANGELOG.md | 2 | ||||
| -rw-r--r-- | config/config.exs | 7 | ||||
| -rw-r--r-- | config/description.exs | 37 | ||||
| -rw-r--r-- | docs/API/prometheus.md | 26 | ||||
| -rw-r--r-- | docs/clients.md | 35 | ||||
| -rw-r--r-- | lib/pleroma/helpers/inet_helper.ex | 19 | ||||
| -rw-r--r-- | lib/pleroma/web/endpoint.ex | 40 | ||||
| -rw-r--r-- | test/pleroma/web/endpoint/metrics_exporter_test.exs | 68 | 
8 files changed, 212 insertions, 22 deletions
| diff --git a/CHANGELOG.md b/CHANGELOG.md index 1f15a8b95..6ca56a90d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,12 +12,14 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).  - Media preview proxy (requires `ffmpeg` and `ImageMagick` to be installed and media proxy to be enabled; see `:media_preview_proxy` config for more details).  - Pleroma API: Importing the mutes users from CSV files.  - Experimental websocket-based federation between Pleroma instances. +- App metrics: ability to restrict access to specified IP whitelist.  ### Changed  - **Breaking** Requires `libmagic` (or `file`) to guess file types.  - **Breaking:** Pleroma Admin API: emoji packs and files routes changed.  - **Breaking:** Sensitive/NSFW statuses no longer disable link previews. +- **Breaking:** App metrics endpoint (`/api/pleroma/app_metrics`) is disabled by default, check `docs/API/prometheus.md` on enabling and configuring.   - Search: Users are now findable by their urls.  - Renamed `:await_up_timeout` in `:connections_pool` namespace to `:connect_timeout`, old name is deprecated.  - Renamed `:timeout` in `pools` namespace to `:recv_timeout`, old name is deprecated. diff --git a/config/config.exs b/config/config.exs index 124f30a77..bd611fd42 100644 --- a/config/config.exs +++ b/config/config.exs @@ -635,7 +635,12 @@ config :pleroma, Pleroma.Emails.UserEmail,  config :pleroma, Pleroma.Emails.NewUsersDigestEmail, enabled: false -config :prometheus, Pleroma.Web.Endpoint.MetricsExporter, path: "/api/pleroma/app_metrics" +config :prometheus, Pleroma.Web.Endpoint.MetricsExporter, +  enabled: false, +  auth: false, +  ip_whitelist: [], +  path: "/api/pleroma/app_metrics", +  format: :text  config :pleroma, Pleroma.ScheduledActivity,    daily_user_limit: 25, diff --git a/config/description.exs b/config/description.exs index 0da1da57d..55363c45a 100644 --- a/config/description.exs +++ b/config/description.exs @@ -3716,5 +3716,42 @@ config :pleroma, :config_description, [          suggestions: [2]        }      ] +  }, +  %{ +    group: :prometheus, +    key: Pleroma.Web.Endpoint.MetricsExporter, +    type: :group, +    description: "Prometheus app metrics endpoint configuration", +    children: [ +      %{ +        key: :enabled, +        type: :boolean, +        description: "[Pleroma extension] Enables app metrics endpoint." +      }, +      %{ +        key: :ip_whitelist, +        type: [{:list, :string}, {:list, :charlist}, {:list, :tuple}], +        description: +          "[Pleroma extension] If non-empty, restricts access to app metrics endpoint to specified IP addresses." +      }, +      %{ +        key: :auth, +        type: [:boolean, :tuple], +        description: "Enables HTTP Basic Auth for app metrics endpoint.", +        suggestion: [false, {:basic, "myusername", "mypassword"}] +      }, +      %{ +        key: :path, +        type: :string, +        description: "App metrics endpoint URI path.", +        suggestions: ["/api/pleroma/app_metrics"] +      }, +      %{ +        key: :format, +        type: :atom, +        description: "App metrics endpoint output format.", +        suggestions: [:text, :protobuf] +      } +    ]    }  ] diff --git a/docs/API/prometheus.md b/docs/API/prometheus.md index 19c564e3c..a5158d905 100644 --- a/docs/API/prometheus.md +++ b/docs/API/prometheus.md @@ -2,15 +2,37 @@  Pleroma includes support for exporting metrics via the [prometheus_ex](https://github.com/deadtrickster/prometheus.ex) library. +Config example: + +``` +config :prometheus, Pleroma.Web.Endpoint.MetricsExporter, +  enabled: true, +  auth: {:basic, "myusername", "mypassword"}, +  ip_whitelist: ["127.0.0.1"], +  path: "/api/pleroma/app_metrics", +  format: :text +``` + +* `enabled` (Pleroma extension) enables the endpoint +* `ip_whitelist` (Pleroma extension) could be used to restrict access only to specified IPs +* `auth` sets the authentication (`false` for no auth; configurable to HTTP Basic Auth, see [prometheus-plugs](https://github.com/deadtrickster/prometheus-plugs#exporting) documentation) +* `format` sets the output format (`:text` or `:protobuf`) +* `path` sets the path to app metrics page  + +  ## `/api/pleroma/app_metrics` +  ### Exports Prometheus application metrics +  * Method: `GET` -* Authentication: not required +* Authentication: not required by default (see configuration options above)  * Params: none -* Response: JSON +* Response: text  ## Grafana +  ### Config example +  The following is a config example to use with [Grafana](https://grafana.com)  ``` diff --git a/docs/clients.md b/docs/clients.md index 1e2c14f1b..3d81763e1 100644 --- a/docs/clients.md +++ b/docs/clients.md @@ -7,97 +7,105 @@ Feel free to contact us to be added to this list!  - Homepage: <https://www.pleroma.com/#desktopApp>  - Source Code: <https://github.com/roma-apps/roma-desktop>  - Platforms: Windows, Mac, Linux -- Features: Streaming Ready +- Features: MastoAPI, Streaming Ready  ### Social  - Source Code: <https://gitlab.gnome.org/World/Social>  - Contact: [@brainblasted@social.libre.fi](https://social.libre.fi/users/brainblasted)  - Platforms: Linux (GNOME)  - Note(2019-01-28): Not at a pre-alpha stage yet +- Features: MastoAPI  ### Whalebird  - Homepage: <https://whalebird.org/>  - Source Code: <https://github.com/h3poteto/whalebird-desktop>  - Contact: [@h3poteto@pleroma.io](https://pleroma.io/users/h3poteto)  - Platforms: Windows, Mac, Linux -- Features: Streaming Ready +- Features: MastoAPI, Streaming Ready  ## Handheld +### AndStatus +- Homepage: <http://andstatus.org/> +- Source Code: <https://github.com/andstatus/andstatus/> +- Platforms: Android +- Features: MastoAPI, ActivityPub (Client-to-Server) +  ### Amaroq  - Homepage: <https://itunes.apple.com/us/app/amaroq-for-mastodon/id1214116200>  - Source Code: <https://github.com/ReticentJohn/Amaroq>  - Contact: [@eurasierboy@mastodon.social](https://mastodon.social/users/eurasierboy)  - Platforms: iOS -- Features: No Streaming +- Features: MastoAPI, No Streaming  ### Fedilab  - Homepage: <https://fedilab.app/>  - Source Code: <https://framagit.org/tom79/fedilab/>  - Contact: [@fedilab@framapiaf.org](https://framapiaf.org/users/fedilab)  - Platforms: Android -- Features: Streaming Ready, Moderation, Text Formatting +- Features: MastoAPI, Streaming Ready, Moderation, Text Formatting  ### Kyclos  - Source Code: <https://git.pleroma.social/pleroma/harbour-kyclos>  - Platforms: SailfishOS -- Features: No Streaming +- Features: MastoAPI, No Streaming  ### Husky  - Source code: <https://git.mentality.rip/FWGS/Husky>  - Contact: [@Husky@enigmatic.observer](https://enigmatic.observer/users/Husky)  - Platforms: Android -- Features: No Streaming, Emoji Reactions, Text Formatting, FE Stickers +- Features: MastoAPI, No Streaming, Emoji Reactions, Text Formatting, FE Stickers  ### Fedi  - Homepage: <https://www.fediapp.com/>  - Source Code: Proprietary, but gratis  - Platforms: iOS, Android -- Features: Pleroma-specific features like Reactions +- Features: MastoAPI, Pleroma-specific features like Reactions  ### Tusky  - Homepage: <https://tuskyapp.github.io/>  - Source Code: <https://github.com/tuskyapp/Tusky>  - Contact: [@ConnyDuck@mastodon.social](https://mastodon.social/users/ConnyDuck)  - Platforms: Android -- Features: No Streaming +- Features: MastoAPI, No Streaming  ### Twidere  - Homepage: <https://twidere.mariotaku.org/>  - Source Code: <https://github.com/TwidereProject/Twidere-Android/>  - Contact: <me@mariotaku.org>  - Platform: Android -- Features: No Streaming +- Features: MastoAPI, No Streaming  ### Indigenous  - Homepage: <https://indigenous.realize.be/>  - Source Code: <https://github.com/swentel/indigenous-android/>  - Contact: [@swentel@realize.be](https://realize.be)  - Platforms: Android -- Features: No Streaming +- Features: MastoAPI, No Streaming  ## Alternative Web Interfaces  ### Brutaldon  - Homepage: <https://jfm.carcosa.net/projects/software/brutaldon/>  - Source Code: <https://git.carcosa.net/jmcbray/brutaldon>  - Contact: [@gcupc@glitch.social](https://glitch.social/users/gcupc) -- Features: No Streaming +- Features: MastoAPI, No Streaming  ### Halcyon  - Source Code: <https://notabug.org/halcyon-suite/halcyon>  - Contact: [@halcyon@social.csswg.org](https://social.csswg.org/users/halcyon) -- Features: Streaming Ready +- Features: MastoAPI, Streaming Ready  ### Pinafore  - Homepage: <https://pinafore.social/>  - Source Code: <https://github.com/nolanlawson/pinafore>  - Contact: [@pinafore@mastodon.technology](https://mastodon.technology/users/pinafore)  - Note: Pleroma support is a secondary goal -- Features: No Streaming +- Features: MastoAPI, No Streaming  ### Sengi  - Homepage: <https://nicolasconstant.github.io/sengi/>  - Source Code: <https://github.com/NicolasConstant/sengi>  - Contact: [@sengi_app@mastodon.social](https://mastodon.social/users/sengi_app) +- Features: MastoAPI  ### DashFE  - Source Code: <https://notabug.org/daisuke/DashboardFE> @@ -107,3 +115,4 @@ Feel free to contact us to be added to this list!  - Source Code: <https://git.freesoftwareextremist.com/bloat/>  - Contact: [@r@freesoftwareextremist.com](https://freesoftwareextremist.com/users/r)  - Features: Does not requires JavaScript +- Features: MastoAPI diff --git a/lib/pleroma/helpers/inet_helper.ex b/lib/pleroma/helpers/inet_helper.ex new file mode 100644 index 000000000..126f82381 --- /dev/null +++ b/lib/pleroma/helpers/inet_helper.ex @@ -0,0 +1,19 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Helpers.InetHelper do +  def parse_address(ip) when is_tuple(ip) do +    {:ok, ip} +  end + +  def parse_address(ip) when is_binary(ip) do +    ip +    |> String.to_charlist() +    |> parse_address() +  end + +  def parse_address(ip) do +    :inet.parse_address(ip) +  end +end diff --git a/lib/pleroma/web/endpoint.ex b/lib/pleroma/web/endpoint.ex index d0e01f3d9..f26542e88 100644 --- a/lib/pleroma/web/endpoint.ex +++ b/lib/pleroma/web/endpoint.ex @@ -7,6 +7,8 @@ defmodule Pleroma.Web.Endpoint do    require Pleroma.Constants +  alias Pleroma.Config +    socket("/socket", Pleroma.Web.UserSocket)    plug(Plug.Telemetry, event_prefix: [:phoenix, :endpoint]) @@ -88,19 +90,19 @@ defmodule Pleroma.Web.Endpoint do    plug(Plug.Parsers,      parsers: [        :urlencoded, -      {:multipart, length: {Pleroma.Config, :get, [[:instance, :upload_limit]]}}, +      {:multipart, length: {Config, :get, [[:instance, :upload_limit]]}},        :json      ],      pass: ["*/*"],      json_decoder: Jason, -    length: Pleroma.Config.get([:instance, :upload_limit]), +    length: Config.get([:instance, :upload_limit]),      body_reader: {Pleroma.Web.Plugs.DigestPlug, :read_body, []}    )    plug(Plug.MethodOverride)    plug(Plug.Head) -  secure_cookies = Pleroma.Config.get([__MODULE__, :secure_cookie_flag]) +  secure_cookies = Config.get([__MODULE__, :secure_cookie_flag])    cookie_name =      if secure_cookies, @@ -108,7 +110,7 @@ defmodule Pleroma.Web.Endpoint do        else: "pleroma_key"    extra = -    Pleroma.Config.get([__MODULE__, :extra_cookie_attrs]) +    Config.get([__MODULE__, :extra_cookie_attrs])      |> Enum.join(";")    # The session will be stored in the cookie and signed, @@ -118,7 +120,7 @@ defmodule Pleroma.Web.Endpoint do      Plug.Session,      store: :cookie,      key: cookie_name, -    signing_salt: Pleroma.Config.get([__MODULE__, :signing_salt], "CqaoopA2"), +    signing_salt: Config.get([__MODULE__, :signing_salt], "CqaoopA2"),      http_only: true,      secure: secure_cookies,      extra: extra @@ -138,8 +140,34 @@ defmodule Pleroma.Web.Endpoint do      use Prometheus.PlugExporter    end +  defmodule MetricsExporterCaller do +    @behaviour Plug + +    def init(opts), do: opts + +    def call(conn, opts) do +      prometheus_config = Application.get_env(:prometheus, MetricsExporter, []) +      ip_whitelist = List.wrap(prometheus_config[:ip_whitelist]) + +      cond do +        !prometheus_config[:enabled] -> +          conn + +        ip_whitelist != [] and +            !Enum.find(ip_whitelist, fn ip -> +              Pleroma.Helpers.InetHelper.parse_address(ip) == {:ok, conn.remote_ip} +            end) -> +          conn + +        true -> +          MetricsExporter.call(conn, opts) +      end +    end +  end +    plug(PipelineInstrumenter) -  plug(MetricsExporter) + +  plug(MetricsExporterCaller)    plug(Pleroma.Web.Router) diff --git a/test/pleroma/web/endpoint/metrics_exporter_test.exs b/test/pleroma/web/endpoint/metrics_exporter_test.exs new file mode 100644 index 000000000..875addc96 --- /dev/null +++ b/test/pleroma/web/endpoint/metrics_exporter_test.exs @@ -0,0 +1,68 @@ +# Pleroma: A lightweight social networking server +# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/> +# SPDX-License-Identifier: AGPL-3.0-only + +defmodule Pleroma.Web.Endpoint.MetricsExporterTest do +  use Pleroma.Web.ConnCase + +  alias Pleroma.Web.Endpoint.MetricsExporter + +  defp config do +    Application.get_env(:prometheus, MetricsExporter) +  end + +  describe "with default config" do +    test "does NOT expose app metrics", %{conn: conn} do +      conn +      |> get(config()[:path]) +      |> json_response(404) +    end +  end + +  describe "when enabled" do +    setup do +      initial_config = config() +      on_exit(fn -> Application.put_env(:prometheus, MetricsExporter, initial_config) end) + +      Application.put_env( +        :prometheus, +        MetricsExporter, +        Keyword.put(initial_config, :enabled, true) +      ) +    end + +    test "serves app metrics", %{conn: conn} do +      conn = get(conn, config()[:path]) +      assert response = response(conn, 200) + +      for metric <- [ +            "http_requests_total", +            "http_request_duration_microseconds", +            "phoenix_controller_call_duration", +            "telemetry_scrape_duration", +            "erlang_vm_memory_atom_bytes_total" +          ] do +        assert response =~ ~r/#{metric}/ +      end +    end + +    test "when IP whitelist configured, " <> +           "serves app metrics only if client IP is whitelisted", +         %{conn: conn} do +      Application.put_env( +        :prometheus, +        MetricsExporter, +        Keyword.put(config(), :ip_whitelist, ["127.127.127.127", {1, 1, 1, 1}, '255.255.255.255']) +      ) + +      conn +      |> get(config()[:path]) +      |> json_response(404) + +      conn +      |> Map.put(:remote_ip, {127, 127, 127, 127}) +      |> get(config()[:path]) +      |> response(200) +    end +  end +end | 
