-
Notifications
You must be signed in to change notification settings - Fork 54
/
danbooru_default_config.rb
712 lines (581 loc) · 17 KB
/
danbooru_default_config.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
# frozen_string_literal: true
module Danbooru
class Configuration
# The version of this Danbooru.
def version
"2.1.0"
end
# The name of this Danbooru.
def app_name
"e621"
end
def description
"Find good furry art, fast"
end
def domain
"e621.net"
end
# Force rating:s on this version of the site.
def safe_mode?
false
end
# The canonical hostname of the site.
def hostname
Socket.gethostname
end
# Contact email address of the admin.
def contact_email
"management@#{domain}"
end
def takedown_email
"management@#{domain}"
end
# System actions, such as sending automated dmails, will be performed with
# this account. This account must have Moderator privileges.
#
# Run `rake db:seed` to create this account if it doesn't already exist in your install.
def system_user
"auto_moderator"
end
def source_code_url
"https://github.com/e621ng/e621ng"
end
# Stripped of any special characters.
def safe_app_name
app_name.gsub(/[^a-zA-Z0-9_-]/, "_")
end
# The default name to use for anyone who isn't logged in.
def default_guest_name
"Anonymous"
end
def levels
{
"Anonymous" => 0,
"Blocked" => 10,
"Member" => 20,
"Privileged" => 30,
"Former Staff" => 34,
"Janitor" => 35,
"Moderator" => 40,
"Admin" => 50
}
end
# Prevent new users from going above 80k while allowing those currently above
# it to continue adding new favorites with the old limit.
# { 123 => 200_000 }
def legacy_favorite_limit
{}
end
# Set the default level, permissions, and other settings for new users here.
def customize_new_user(user)
user.blacklisted_tags = default_blacklist.join("\n")
user.comment_threshold = -10
user.enable_auto_complete = true
user.enable_keyboard_navigation = true
user.per_page = records_per_page
user.show_post_statistics = true
user.style_usernames = true
end
def default_blacklist
[]
end
def safeblocked_tags
[]
end
# This allows using statically linked copies of ffmpeg in non default locations. Not universally supported across
# the codebase at this time.
def ffmpeg_path
"/usr/bin/ffmpeg"
end
# Thumbnail size
def small_image_width
150
end
# Large resize image width. Set to nil to disable.
def large_image_width
850
end
def large_image_prefix
"sample-"
end
def protected_path_prefix
"deleted"
end
def protected_file_secret
"abc123"
end
def replacement_path_prefix
"replacements"
end
def replacement_file_secret
"abc123"
end
def deleted_preview_url
"/images/deleted-preview.png"
end
# When calculating statistics based on the posts table, gather this many posts to sample from.
def post_sample_size
300
end
# List of memcached servers
def memcached_servers
%w(127.0.0.1:11211)
end
def alias_implication_forum_category
1
end
# After a post receives this many comments, new comments will no longer bump the post in comment/index.
def comment_threshold
40
end
def disable_throttles?
false
end
def disable_age_checks?
false
end
def disable_cache_store?
false
end
# Members cannot post more than X comments in an hour.
def member_comment_limit
15
end
def comment_vote_limit
10
end
def post_vote_limit
3_000
end
def dmail_minute_limit
1
end
def dmail_limit
10
end
def dmail_day_limit
50
end
def tag_suggestion_limit
15
end
def forum_vote_limit
50
end
# Blips created in the last hour
def blip_limit
25
end
# Artists creator or edited in the last hour
def artist_edit_limit
25
end
# Wiki pages created or edited in the last hour
def wiki_edit_limit
60
end
# Notes applied to posts edited or created in the last hour
def note_edit_limit
50
end
# Pools created in the last hour
def pool_limit
2
end
# Pools created or edited in the last hour
def pool_edit_limit
10
end
# Pools that you can edit the posts for in the last hour
def pool_post_edit_limit
30
end
# Members cannot create more than X post versions in an hour.
def post_edit_limit
150
end
def post_flag_limit
20
end
# Flat limit that applies to all users, regardless of level
def hourly_upload_limit
30
end
def ticket_limit
30
end
# Members cannot change the category of pools with more than this many posts.
def pool_category_change_limit
30
end
def post_replacement_per_day_limit
2
end
def post_replacement_per_post_limit
5
end
def remember_key
"abc123"
end
def tag_type_change_cutoff
100
end
# Users cannot search for more than X regular tags at a time.
def tag_query_limit
40
end
# Return true if the given tag shouldn't count against the user's tag search limit.
def is_unlimited_tag?(tag)
!!(tag =~ /\A(-?status:deleted|rating:s.*|limit:.+)\z/i)
end
# After this many pages, the paginator will switch to sequential mode.
def max_numbered_pages
750
end
def blip_max_size
1_000
end
def comment_max_size
10_000
end
def dmail_max_size
50_000
end
def forum_post_max_size
50_000
end
def note_max_size
1_000
end
def pool_descr_max_size
10_000
end
def post_descr_max_size
50_000
end
def ticket_max_size
5_000
end
def user_about_max_size
50_000
end
def wiki_page_max_size
250_000
end
def user_feedback_max_size
20_000
end
def discord_site
end
def discord_secret
end
# Maximum size of an upload. If you change this, you must also change
# `client_max_body_size` in your nginx.conf.
def max_file_size
100.megabytes
end
def max_file_sizes
{
'jpg' => 100.megabytes,
'png' => 100.megabytes,
'gif' => 20.megabytes,
'webm' => 100.megabytes
}
end
def max_apng_file_size
20.megabytes
end
# Measured in seconds
def max_video_duration
3600
end
# Maximum resolution (width * height) of an upload. Default: 441 megapixels (21000x21000 pixels).
def max_image_resolution
15000 * 15000
end
# Maximum width of an upload.
def max_image_width
15000
end
# Maximum height of an upload.
def max_image_height
15000
end
def max_tags_per_post
2000
end
# Permanently redirect all HTTP requests to HTTPS.
#
# https://en.wikipedia.org/wiki/HTTP_Strict_Transport_Security
# http://api.rubyonrails.org/classes/ActionDispatch/SSL.html
def ssl_options
{
redirect: { exclude: ->(request) { request.subdomain == "insecure" } },
hsts: {
expires: 1.year,
preload: true,
subdomains: false,
},
}
end
# The method to use for storing image files.
def storage_manager
# Store files on the local filesystem.
# base_dir - where to store files (default: under public/data)
# base_url - where to serve files from (default: http://#{hostname}/data)
# hierarchical: false - store files in a single directory
# hierarchical: true - store files in a hierarchical directory structure, based on the MD5 hash
StorageManager::Local.new(base_dir: "#{Rails.root}/public/data", hierarchical: true)
# Select the storage method based on the post's id and type (preview, large, or original).
# StorageManager::Hybrid.new do |id, md5, file_ext, type|
# if type.in?([:large, :original]) && id.in?(0..850_000)
# StorageManager::Local.new(base_dir: "/path/to/files", hierarchical: true)
# else
# StorageManager::Local.new(base_dir: "/path/to/files", hierarchical: true)
# end
# end
end
# The method to use for backing up image files.
def backup_storage_manager
# Don't perform any backups.
StorageManager::Null.new
# Backup files to /mnt/backup on the local filesystem.
# StorageManager::Local.new(base_dir: "/mnt/backup", hierarchical: false)
end
# If enabled, users must verify their email addresses.
def enable_email_verification?
false
end
def enable_signups?
true
end
def flag_reasons
[
{
name: "uploading_guidelines",
reason: "Does not meet the [[uploading_guidelines|uploading guidelines]]",
text: "This post fails to meet the site's standards, be it for artistic worth, image quality, relevancy, or something else.\nKeep in mind that your personal preferences have no bearing on this. If you find the content of a post objectionable, simply [[e621:blacklist|blacklist]] it."
},
{
name: 'dnp_artist',
reason: "The artist of this post is on the [[avoid_posting|avoid posting list]]",
text: "Certain artists have requested that their work is not to be published on this site, and were granted [[avoid_posting|Do Not Post]] status.\nSometimes, that status comes with conditions; see [[conditional_dnp]] for more information"
},
{
name: 'pay_content',
reason: "Paysite, commercial, or subscription content",
text: "We do not host paysite or commercial content of any kind. This includes Patreon leaks, reposts from piracy websites, and so on."
},
{
name: 'trace',
reason: "Trace of another artist's work",
text: "Images traced from other artists' artwork are not accepted on this site. Referencing from something is fine, but outright copying someone else's work is not.\nPlease, leave more information in the comments, or simply add the original artwork as the posts's parent if it's hosted on this site."
},
{
name: 'previously_deleted',
reason: "Previously deleted",
text: "Posts usually get removed for a good reason, and reuploading of deleted content is not acceptable.\nPlease, leave more information in the comments, or simply add the original post as this post's parent."
},
{
name: 'real_porn',
reason: "Real-life pornography",
text: "Posts featuring real-life pornography are not acceptable on this site. No exceptions.\nNote that images featuring non-erotic photographs are acceptable."
},
{
name: 'corrupt',
reason: "File is either corrupted, broken, or otherwise does not work",
text: "Something about this post does not work quite right. This may be a broken video, or a corrupted image.\nEither way, in order to avoid confusion, please explain the situation in the comments."
},
{
name: 'inferior',
reason: "Duplicate or inferior version of another post",
text: "A superior version of this post already exists on the site.\nThis may include images with better visual quality (larger, less compressed), but may also feature \"fixed\" versions, with visual mistakes accounted for by the artist.\nNote that edits and alternate versions do not fall under this category.",
parent: true
},
]
end
def deletion_reasons
[
"Inferior version/duplicate of post #%PARENT_ID%",
"Previously deleted (post #%PARENT_ID%)",
"Excessive same base image set",
"Colored base",
"Advertisement",
"Underage artist",
"",
"Does not meet minimum quality standards (Artistic)",
"Does not meet minimum quality standards (Resolution)",
"Does not meet minimum quality standards (Compression)",
"Does not meet minimum quality standards (Trivial or low quality edit)",
"Does not meet minimum quality standards (Bad digitization of traditional media)",
"Does not meet minimum quality standards (Photo)",
"Does not meet minimum quality standards (%OTHER_ID%)",
"Broken/corrupted file",
"JPG resaved as PNG",
"",
"Irrelevant to site (Human only)",
"Irrelevant to site (Screencap)",
"Irrelevant to site (Zero pictured)",
"Irrelevant to site (AI assisted/generated)",
"Irrelevant to site (%OTHER_ID%)",
"",
"Paysite/commercial content",
"Traced artwork",
"Traced artwork (post #%PARENT_ID%)",
"Takedown #%OTHER_ID%",
"The artist of this post is on the [[avoid_posting|avoid posting list]]",
"[[conditional_dnp|Conditional DNP]] (Only the artist is allowed to post)",
"[[conditional_dnp|Conditional DNP]] (%OTHER_ID%)",
]
end
# Any custom code you want to insert into the default layout without
# having to modify the templates.
def custom_html_header_content
nil
end
def flag_notice_wiki_page
"help:flag_notice"
end
def replacement_notice_wiki_page
"help:replacement_notice"
end
# The number of records displayed per page. Posts use `user.per_page` which is configurable by the user
def records_per_page
75
end
def is_post_restricted?(post)
false
end
# TODO: Investigate what this does and where it is used.
def is_user_restricted?(user)
!user.is_privileged?
end
def can_user_see_post?(user, post)
return false if post.is_deleted? && !user.is_janitor?
if is_user_restricted?(user) && is_post_restricted?(post)
false
else
true
end
end
def user_needs_login_for_post?(post)
false
end
def select_posts_visible_to_user(user, posts)
posts.select {|x| can_user_see_post?(user, x)}
end
def enable_dimension_autotagging?
true
end
# The default headers to be sent with outgoing http requests. Some external
# services will fail if you don't set a valid User-Agent.
def http_headers
{
"User-Agent" => "#{Danbooru.config.safe_app_name}/#{Danbooru.config.version}",
}
end
def httparty_options
# proxy example:
# {http_proxyaddr: "", http_proxyport: "", http_proxyuser: nil, http_proxypass: nil}
{
timeout: 10,
open_timout: 5,
headers: Danbooru.config.http_headers,
}
end
# you should override this
def email_key
"zDMSATq0W3hmA5p3rKTgD"
end
def mailgun_api_key
''
end
def mailgun_domain
''
end
def mail_from_addr
'noreply@localhost'
end
# disable this for tests
def enable_sock_puppet_validation?
true
end
def iqdb_server
end
def opensearch_host
end
# Use a recaptcha on the signup page to protect against spambots creating new accounts.
# https://developers.google.com/recaptcha/intro
def enable_recaptcha?
Rails.env.production? && Danbooru.config.recaptcha_site_key.present? && Danbooru.config.recaptcha_secret_key.present?
end
def recaptcha_site_key
end
def recaptcha_secret_key
end
def enable_image_cropping?
true
end
def redis_url
end
def bypass_upload_whitelist?(user)
user.is_admin?
end
def ads_enabled?
false
end
# These tags will be sent to the revive server to do filtering on
def ads_keyword_tags
[]
end
def ads_zone_desktop
{zone: nil, revive_id: nil, checksum: nil}
end
def ads_zone_mobile
{zone: nil, revive_id: nil, checksum: nil}
end
# Additional video samples will be generated in these dimensions if it makes sense to do so
# They will be available as additional scale options on applicable posts in the order they appear here
def video_rescales
{'720p' => [1280, 720], '480p' => [640, 480]}
end
def image_rescales
[]
end
def enable_visitor_metrics?
false
end
def janitor_reports_discord_webhook_url
nil
end
def moderator_stats_discord_webhook_url
nil
end
def aibur_stats_discord_webhook_url
nil
end
end
class EnvironmentConfiguration
def custom_configuration
@custom_configuration ||= CustomConfiguration.new
end
def env_to_boolean(method, var)
is_boolean = method.to_s.end_with? "?"
return true if is_boolean && var.truthy?
return false if is_boolean && var.falsy?
var
end
def method_missing(method, *)
var = ENV["DANBOORU_#{method.to_s.upcase.chomp("?")}"]
if var.present?
env_to_boolean(method, var)
else
custom_configuration.send(method, *)
end
end
end
def config
@configuration ||= EnvironmentConfiguration.new
end
module_function :config
end