defmodule PhilomenaWeb.StatsUpdater do alias Philomena.Elasticsearch alias Philomena.Servers.Config alias Philomena.Images.Image alias Philomena.Comments.Comment alias Philomena.Topics.Topic alias Philomena.Forums.Forum alias Philomena.Posts.Post alias Philomena.Users.User alias Philomena.Galleries.Gallery alias Philomena.Galleries.Interaction alias Philomena.Commissions.Commission alias Philomena.Commissions.Item alias Philomena.Reports.Report alias Philomena.StaticPages.StaticPage alias Philomena.Repo import Ecto.Query def child_spec([]) do %{ id: PhilomenaWeb.StatsUpdater, start: {PhilomenaWeb.StatsUpdater, :start_link, [[]]} } end def start_link([]) do {:ok, spawn_link(&run/0)} end defp run do :timer.sleep(:timer.seconds(300)) {gallery_count, gallery_size, distinct_creators, images_in_galleries} = galleries() {open_reports, report_count, response_time} = moderation() {open_commissions, commission_items} = commissions() {image_aggs, comment_aggs} = aggregations() {forums, topics, posts} = forums() {users, users_24h} = users() result = Phoenix.View.render( PhilomenaWeb.StatView, "index.html", image_aggs: image_aggs, comment_aggs: comment_aggs, forums_count: forums, topics_count: topics, posts_count: posts, users_count: users, users_24h: users_24h, open_commissions: open_commissions, commission_items: commission_items, open_reports: open_reports, report_stat_count: report_count, response_time: response_time, gallery_count: gallery_count, gallery_size: gallery_size, distinct_creators: distinct_creators, images_in_galleries: images_in_galleries ) now = NaiveDateTime.utc_now() |> NaiveDateTime.truncate(:second) static_page = %{ title: "Statistics", slug: "stats", body: Phoenix.HTML.safe_to_string(result), created_at: now, updated_at: now } Repo.insert_all(StaticPage, [static_page], on_conflict: {:replace, [:body, :updated_at]}, conflict_target: :slug) run() end defp aggregations do data = Config.get(:aggregation) { Elasticsearch.search(Image, data["images"]), Elasticsearch.search(Comment, data["comments"]) } end defp forums do forums = Forum |> where(access_level: "normal") |> Repo.aggregate(:count, :id) first_topic = Repo.one(first(Topic)) last_topic = Repo.one(last(Topic)) first_post = Repo.one(first(Post)) last_post = Repo.one(last(Post)) {forums, diff(last_topic, first_topic), diff(last_post, first_post)} end defp users do first_user = Repo.one(first(User)) last_user = Repo.one(last(User)) time = DateTime.utc_now() |> DateTime.add(-86400, :second) last_24h = User |> where([u], u.created_at > ^time) |> Repo.aggregate(:count, :id) {diff(last_user, first_user), last_24h} end defp galleries do gallery_count = Repo.aggregate(Gallery, :count, :id) gallery_size = Repo.aggregate(Gallery, :avg, :image_count) |> Kernel.||(Decimal.new(0)) |> Decimal.to_float() |> trunc() distinct_creators = Gallery |> distinct(:creator_id) |> Repo.aggregate(:count, :id) first_gi = Repo.one(first(Interaction)) last_gi = Repo.one(last(Interaction)) {gallery_count, gallery_size, distinct_creators, diff(last_gi, first_gi)} end defp commissions do open_commissions = Repo.aggregate(where(Commission, open: true), :count, :id) commission_items = Repo.aggregate(Item, :count, :id) {open_commissions, commission_items} end defp moderation do open_reports = Repo.aggregate(where(Report, open: true), :count, :id) first_report = Repo.one(first(Report)) last_report = Repo.one(last(Report)) closed_reports = Report |> where(open: false) |> order_by(desc: :created_at) |> limit(250) |> Repo.all() response_time = closed_reports |> Enum.reduce(0, &(&2 + NaiveDateTime.diff(&1.updated_at, &1.created_at, :second))) |> Kernel./(safe_length(closed_reports) * 3600) |> trunc() {open_reports, diff(last_report, first_report), response_time} end defp diff(nil, nil), do: 0 defp diff(%{id: id2}, %{id: id1}), do: id2 - id1 defp safe_length([]), do: 1 defp safe_length(list), do: length(list) end