logo

pleroma

My custom branche(s) on git.pleroma.social/pleroma/pleroma git clone https://hacktivis.me/git/pleroma.git

config_db_test.exs (17777B)


  1. # Pleroma: A lightweight social networking server
  2. # Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
  3. # SPDX-License-Identifier: AGPL-3.0-only
  4. defmodule Pleroma.ConfigDBTest do
  5. use Pleroma.DataCase, async: true
  6. import Pleroma.Factory
  7. alias Pleroma.ConfigDB
  8. test "get_by_params/1" do
  9. config = insert(:config)
  10. insert(:config)
  11. assert config == ConfigDB.get_by_params(%{group: config.group, key: config.key})
  12. end
  13. test "get_all_as_keyword/0" do
  14. saved = insert(:config)
  15. insert(:config, group: ":goose", key: ":level", value: :info)
  16. insert(:config, group: ":goose", key: ":meta", value: [:none])
  17. insert(:config,
  18. group: ":goose",
  19. key: ":webhook_url",
  20. value: "https://gander.com/"
  21. )
  22. config = ConfigDB.get_all_as_keyword()
  23. assert config[:pleroma] == [
  24. {saved.key, saved.value}
  25. ]
  26. assert config[:goose][:level] == :info
  27. assert config[:goose][:meta] == [:none]
  28. assert config[:goose][:webhook_url] == "https://gander.com/"
  29. end
  30. describe "update_or_create/1" do
  31. test "common" do
  32. config = insert(:config)
  33. key2 = :another_key
  34. params = [
  35. %{group: :pleroma, key: key2, value: "another_value"},
  36. %{group: :pleroma, key: config.key, value: [a: 1, b: 2, c: "new_value"]}
  37. ]
  38. assert Repo.all(ConfigDB) |> length() == 1
  39. Enum.each(params, &ConfigDB.update_or_create(&1))
  40. assert Repo.all(ConfigDB) |> length() == 2
  41. config1 = ConfigDB.get_by_params(%{group: config.group, key: config.key})
  42. config2 = ConfigDB.get_by_params(%{group: :pleroma, key: key2})
  43. assert config1.value == [a: 1, b: 2, c: "new_value"]
  44. assert config2.value == "another_value"
  45. end
  46. test "partial update" do
  47. config = insert(:config, value: [key1: "val1", key2: :val2])
  48. {:ok, config} =
  49. ConfigDB.update_or_create(%{
  50. group: config.group,
  51. key: config.key,
  52. value: [key1: :val1, key3: :val3]
  53. })
  54. updated = ConfigDB.get_by_params(%{group: config.group, key: config.key})
  55. assert config.value == updated.value
  56. assert updated.value[:key1] == :val1
  57. assert updated.value[:key2] == :val2
  58. assert updated.value[:key3] == :val3
  59. end
  60. test "deep merge" do
  61. config = insert(:config, value: [key1: "val1", key2: [k1: :v1, k2: "v2"]])
  62. {:ok, config} =
  63. ConfigDB.update_or_create(%{
  64. group: config.group,
  65. key: config.key,
  66. value: [key1: :val1, key2: [k2: :v2, k3: :v3], key3: :val3]
  67. })
  68. updated = ConfigDB.get_by_params(%{group: config.group, key: config.key})
  69. assert config.value == updated.value
  70. assert updated.value[:key1] == :val1
  71. assert updated.value[:key2] == [k1: :v1, k2: :v2, k3: :v3]
  72. assert updated.value[:key3] == :val3
  73. end
  74. test "only full update for some keys" do
  75. config1 = insert(:config, key: :ecto_repos, value: [repo: Pleroma.Repo])
  76. config2 = insert(:config, group: :cors_plug, key: :max_age, value: 18)
  77. {:ok, _config} =
  78. ConfigDB.update_or_create(%{
  79. group: config1.group,
  80. key: config1.key,
  81. value: [another_repo: [Pleroma.Repo]]
  82. })
  83. {:ok, _config} =
  84. ConfigDB.update_or_create(%{
  85. group: config2.group,
  86. key: config2.key,
  87. value: 777
  88. })
  89. updated1 = ConfigDB.get_by_params(%{group: config1.group, key: config1.key})
  90. updated2 = ConfigDB.get_by_params(%{group: config2.group, key: config2.key})
  91. assert updated1.value == [another_repo: [Pleroma.Repo]]
  92. assert updated2.value == 777
  93. end
  94. test "full update if value is not keyword" do
  95. config =
  96. insert(:config,
  97. group: ":tesla",
  98. key: ":adapter",
  99. value: Tesla.Adapter.Hackney
  100. )
  101. {:ok, _config} =
  102. ConfigDB.update_or_create(%{
  103. group: config.group,
  104. key: config.key,
  105. value: Tesla.Adapter.Httpc
  106. })
  107. updated = ConfigDB.get_by_params(%{group: config.group, key: config.key})
  108. assert updated.value == Tesla.Adapter.Httpc
  109. end
  110. test "only full update for some subkeys" do
  111. config1 =
  112. insert(:config,
  113. key: ":emoji",
  114. value: [groups: [a: 1, b: 2], key: [a: 1]]
  115. )
  116. config2 =
  117. insert(:config,
  118. key: ":assets",
  119. value: [mascots: [a: 1, b: 2], key: [a: 1]]
  120. )
  121. {:ok, _config} =
  122. ConfigDB.update_or_create(%{
  123. group: config1.group,
  124. key: config1.key,
  125. value: [groups: [c: 3, d: 4], key: [b: 2]]
  126. })
  127. {:ok, _config} =
  128. ConfigDB.update_or_create(%{
  129. group: config2.group,
  130. key: config2.key,
  131. value: [mascots: [c: 3, d: 4], key: [b: 2]]
  132. })
  133. updated1 = ConfigDB.get_by_params(%{group: config1.group, key: config1.key})
  134. updated2 = ConfigDB.get_by_params(%{group: config2.group, key: config2.key})
  135. assert updated1.value == [groups: [c: 3, d: 4], key: [a: 1, b: 2]]
  136. assert updated2.value == [mascots: [c: 3, d: 4], key: [a: 1, b: 2]]
  137. end
  138. end
  139. describe "delete/1" do
  140. test "error on deleting non existing setting" do
  141. {:error, error} = ConfigDB.delete(%{group: ":pleroma", key: ":key"})
  142. assert error =~ "Config with params %{group: \":pleroma\", key: \":key\"} not found"
  143. end
  144. test "full delete" do
  145. config = insert(:config)
  146. {:ok, deleted} = ConfigDB.delete(%{group: config.group, key: config.key})
  147. assert Ecto.get_meta(deleted, :state) == :deleted
  148. refute ConfigDB.get_by_params(%{group: config.group, key: config.key})
  149. end
  150. test "partial subkeys delete" do
  151. config = insert(:config, value: [groups: [a: 1, b: 2], key: [a: 1]])
  152. {:ok, deleted} =
  153. ConfigDB.delete(%{group: config.group, key: config.key, subkeys: [":groups"]})
  154. assert Ecto.get_meta(deleted, :state) == :loaded
  155. assert deleted.value == [key: [a: 1]]
  156. updated = ConfigDB.get_by_params(%{group: config.group, key: config.key})
  157. assert updated.value == deleted.value
  158. end
  159. test "full delete if remaining value after subkeys deletion is empty list" do
  160. config = insert(:config, value: [groups: [a: 1, b: 2]])
  161. {:ok, deleted} =
  162. ConfigDB.delete(%{group: config.group, key: config.key, subkeys: [":groups"]})
  163. assert Ecto.get_meta(deleted, :state) == :deleted
  164. refute ConfigDB.get_by_params(%{group: config.group, key: config.key})
  165. end
  166. end
  167. describe "to_elixir_types/1" do
  168. test "string" do
  169. assert ConfigDB.to_elixir_types("value as string") == "value as string"
  170. end
  171. test "boolean" do
  172. assert ConfigDB.to_elixir_types(false) == false
  173. end
  174. test "nil" do
  175. assert ConfigDB.to_elixir_types(nil) == nil
  176. end
  177. test "integer" do
  178. assert ConfigDB.to_elixir_types(150) == 150
  179. end
  180. test "atom" do
  181. assert ConfigDB.to_elixir_types(":atom") == :atom
  182. end
  183. test "ssl options" do
  184. assert ConfigDB.to_elixir_types([":tlsv1", ":tlsv1.1", ":tlsv1.2", ":tlsv1.3"]) == [
  185. :tlsv1,
  186. :"tlsv1.1",
  187. :"tlsv1.2",
  188. :"tlsv1.3"
  189. ]
  190. end
  191. test "pleroma module" do
  192. assert ConfigDB.to_elixir_types("Pleroma.Bookmark") == Pleroma.Bookmark
  193. end
  194. test "pleroma string" do
  195. assert ConfigDB.to_elixir_types("Pleroma") == "Pleroma"
  196. end
  197. test "phoenix module" do
  198. assert ConfigDB.to_elixir_types("Phoenix.Socket.V1.JSONSerializer") ==
  199. Phoenix.Socket.V1.JSONSerializer
  200. end
  201. test "tesla module" do
  202. assert ConfigDB.to_elixir_types("Tesla.Adapter.Hackney") == Tesla.Adapter.Hackney
  203. end
  204. test "ExSyslogger module" do
  205. assert ConfigDB.to_elixir_types("ExSyslogger") == ExSyslogger
  206. end
  207. test "Swoosh.Adapters modules" do
  208. assert ConfigDB.to_elixir_types("Swoosh.Adapters.SMTP") == Swoosh.Adapters.SMTP
  209. assert ConfigDB.to_elixir_types("Swoosh.Adapters.AmazonSES") == Swoosh.Adapters.AmazonSES
  210. end
  211. test "sigil" do
  212. assert ConfigDB.to_elixir_types("~r[comp[lL][aA][iI][nN]er]") == ~r/comp[lL][aA][iI][nN]er/
  213. end
  214. test "link sigil" do
  215. assert ConfigDB.to_elixir_types("~r/https:\/\/example.com/") == ~r/https:\/\/example.com/
  216. end
  217. test "link sigil with um modifiers" do
  218. assert ConfigDB.to_elixir_types("~r/https:\/\/example.com/um") ==
  219. ~r/https:\/\/example.com/um
  220. end
  221. test "link sigil with i modifier" do
  222. assert ConfigDB.to_elixir_types("~r/https:\/\/example.com/i") == ~r/https:\/\/example.com/i
  223. end
  224. test "link sigil with s modifier" do
  225. assert ConfigDB.to_elixir_types("~r/https:\/\/example.com/s") == ~r/https:\/\/example.com/s
  226. end
  227. test "raise if valid delimiter not found" do
  228. assert_raise ArgumentError, "valid delimiter for Regex expression not found", fn ->
  229. ConfigDB.to_elixir_types("~r/https://[]{}<>\"'()|example.com/s")
  230. end
  231. end
  232. test "2 child tuple" do
  233. assert ConfigDB.to_elixir_types(%{"tuple" => ["v1", ":v2"]}) == {"v1", :v2}
  234. end
  235. test "proxy tuple with localhost" do
  236. assert ConfigDB.to_elixir_types(%{
  237. "tuple" => [":proxy_url", %{"tuple" => [":socks5", "localhost", 1234]}]
  238. }) == {:proxy_url, {:socks5, :localhost, 1234}}
  239. end
  240. test "proxy tuple with domain" do
  241. assert ConfigDB.to_elixir_types(%{
  242. "tuple" => [":proxy_url", %{"tuple" => [":socks5", "domain.com", 1234]}]
  243. }) == {:proxy_url, {:socks5, 'domain.com', 1234}}
  244. end
  245. test "proxy tuple with ip" do
  246. assert ConfigDB.to_elixir_types(%{
  247. "tuple" => [":proxy_url", %{"tuple" => [":socks5", "127.0.0.1", 1234]}]
  248. }) == {:proxy_url, {:socks5, {127, 0, 0, 1}, 1234}}
  249. end
  250. test "tuple with n children" do
  251. assert ConfigDB.to_elixir_types(%{
  252. "tuple" => [
  253. "v1",
  254. ":v2",
  255. "Pleroma.Bookmark",
  256. 150,
  257. false,
  258. "Phoenix.Socket.V1.JSONSerializer"
  259. ]
  260. }) == {"v1", :v2, Pleroma.Bookmark, 150, false, Phoenix.Socket.V1.JSONSerializer}
  261. end
  262. test "map with string key" do
  263. assert ConfigDB.to_elixir_types(%{"key" => "value"}) == %{"key" => "value"}
  264. end
  265. test "map with atom key" do
  266. assert ConfigDB.to_elixir_types(%{":key" => "value"}) == %{key: "value"}
  267. end
  268. test "list of strings" do
  269. assert ConfigDB.to_elixir_types(["v1", "v2", "v3"]) == ["v1", "v2", "v3"]
  270. end
  271. test "list of modules" do
  272. assert ConfigDB.to_elixir_types(["Pleroma.Repo", "Pleroma.Activity"]) == [
  273. Pleroma.Repo,
  274. Pleroma.Activity
  275. ]
  276. end
  277. test "list of atoms" do
  278. assert ConfigDB.to_elixir_types([":v1", ":v2", ":v3"]) == [:v1, :v2, :v3]
  279. end
  280. test "list of mixed values" do
  281. assert ConfigDB.to_elixir_types([
  282. "v1",
  283. ":v2",
  284. "Pleroma.Repo",
  285. "Phoenix.Socket.V1.JSONSerializer",
  286. 15,
  287. false
  288. ]) == [
  289. "v1",
  290. :v2,
  291. Pleroma.Repo,
  292. Phoenix.Socket.V1.JSONSerializer,
  293. 15,
  294. false
  295. ]
  296. end
  297. test "simple keyword" do
  298. assert ConfigDB.to_elixir_types([%{"tuple" => [":key", "value"]}]) == [key: "value"]
  299. end
  300. test "keyword" do
  301. assert ConfigDB.to_elixir_types([
  302. %{"tuple" => [":types", "Pleroma.PostgresTypes"]},
  303. %{"tuple" => [":telemetry_event", ["Pleroma.Repo.Instrumenter"]]},
  304. %{"tuple" => [":migration_lock", nil]},
  305. %{"tuple" => [":key1", 150]},
  306. %{"tuple" => [":key2", "string"]}
  307. ]) == [
  308. types: Pleroma.PostgresTypes,
  309. telemetry_event: [Pleroma.Repo.Instrumenter],
  310. migration_lock: nil,
  311. key1: 150,
  312. key2: "string"
  313. ]
  314. end
  315. test "trandformed keyword" do
  316. assert ConfigDB.to_elixir_types(a: 1, b: 2, c: "string") == [a: 1, b: 2, c: "string"]
  317. end
  318. test "complex keyword with nested mixed children" do
  319. assert ConfigDB.to_elixir_types([
  320. %{"tuple" => [":uploader", "Pleroma.Uploaders.Local"]},
  321. %{"tuple" => [":filters", ["Pleroma.Upload.Filter.Dedupe"]]},
  322. %{"tuple" => [":link_name", true]},
  323. %{"tuple" => [":proxy_remote", false]},
  324. %{"tuple" => [":common_map", %{":key" => "value"}]},
  325. %{
  326. "tuple" => [
  327. ":proxy_opts",
  328. [
  329. %{"tuple" => [":redirect_on_failure", false]},
  330. %{"tuple" => [":max_body_length", 1_048_576]},
  331. %{
  332. "tuple" => [
  333. ":http",
  334. [
  335. %{"tuple" => [":follow_redirect", true]},
  336. %{"tuple" => [":pool", ":upload"]}
  337. ]
  338. ]
  339. }
  340. ]
  341. ]
  342. }
  343. ]) == [
  344. uploader: Pleroma.Uploaders.Local,
  345. filters: [Pleroma.Upload.Filter.Dedupe],
  346. link_name: true,
  347. proxy_remote: false,
  348. common_map: %{key: "value"},
  349. proxy_opts: [
  350. redirect_on_failure: false,
  351. max_body_length: 1_048_576,
  352. http: [
  353. follow_redirect: true,
  354. pool: :upload
  355. ]
  356. ]
  357. ]
  358. end
  359. test "common keyword" do
  360. assert ConfigDB.to_elixir_types([
  361. %{"tuple" => [":level", ":warning"]},
  362. %{"tuple" => [":meta", [":all"]]},
  363. %{"tuple" => [":path", ""]},
  364. %{"tuple" => [":val", nil]},
  365. %{"tuple" => [":webhook_url", "https://hooks.slack.com/services/YOUR-KEY-HERE"]}
  366. ]) == [
  367. level: :warning,
  368. meta: [:all],
  369. path: "",
  370. val: nil,
  371. webhook_url: "https://hooks.slack.com/services/YOUR-KEY-HERE"
  372. ]
  373. end
  374. test "complex keyword with sigil" do
  375. assert ConfigDB.to_elixir_types([
  376. %{"tuple" => [":federated_timeline_removal", []]},
  377. %{"tuple" => [":reject", ["~r/comp[lL][aA][iI][nN]er/"]]},
  378. %{"tuple" => [":replace", []]}
  379. ]) == [
  380. federated_timeline_removal: [],
  381. reject: [~r/comp[lL][aA][iI][nN]er/],
  382. replace: []
  383. ]
  384. end
  385. test "complex keyword with tuples with more than 2 values" do
  386. assert ConfigDB.to_elixir_types([
  387. %{
  388. "tuple" => [
  389. ":http",
  390. [
  391. %{
  392. "tuple" => [
  393. ":key1",
  394. [
  395. %{
  396. "tuple" => [
  397. ":_",
  398. [
  399. %{
  400. "tuple" => [
  401. "/api/v1/streaming",
  402. "Pleroma.Web.MastodonAPI.WebsocketHandler",
  403. []
  404. ]
  405. },
  406. %{
  407. "tuple" => [
  408. "/websocket",
  409. "Phoenix.Endpoint.CowboyWebSocket",
  410. %{
  411. "tuple" => [
  412. "Phoenix.Transports.WebSocket",
  413. %{
  414. "tuple" => [
  415. "Pleroma.Web.Endpoint",
  416. "Pleroma.Web.UserSocket",
  417. []
  418. ]
  419. }
  420. ]
  421. }
  422. ]
  423. },
  424. %{
  425. "tuple" => [
  426. ":_",
  427. "Phoenix.Endpoint.Cowboy2Handler",
  428. %{"tuple" => ["Pleroma.Web.Endpoint", []]}
  429. ]
  430. }
  431. ]
  432. ]
  433. }
  434. ]
  435. ]
  436. }
  437. ]
  438. ]
  439. }
  440. ]) == [
  441. http: [
  442. key1: [
  443. {:_,
  444. [
  445. {"/api/v1/streaming", Pleroma.Web.MastodonAPI.WebsocketHandler, []},
  446. {"/websocket", Phoenix.Endpoint.CowboyWebSocket,
  447. {Phoenix.Transports.WebSocket,
  448. {Pleroma.Web.Endpoint, Pleroma.Web.UserSocket, []}}},
  449. {:_, Phoenix.Endpoint.Cowboy2Handler, {Pleroma.Web.Endpoint, []}}
  450. ]}
  451. ]
  452. ]
  453. ]
  454. end
  455. end
  456. end