Mercurial > zeropaste
annotate app/models/paste.rb @ 456:caac9bc885b4
More font size
author | nanaya <me@nanaya.pro> |
---|---|
date | Tue, 14 Nov 2017 22:49:16 +0900 |
parents | b445318de858 |
children | f11862e58af4 |
rev | line source |
---|---|
433 | 1 class Paste < ApplicationRecord |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
2 attr_accessor :is_private |
379 | 3 after_initialize :set_privacy, :if => :new_record? |
4 | |
70
8f0fb869e770
Limit pastes to 100/hour per IP address.
Edho Arief <edho@myconan.net>
parents:
51
diff
changeset
|
5 before_validation :paste_limit |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
6 before_validation :convert_newlines |
2 | 7 before_validation :set_paste_hash |
176
a9dba6a3008b
Initial work to add paste deletion.
Edho Arief <edho@myconan.net>
parents:
71
diff
changeset
|
8 before_validation :set_paste_key |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
9 before_validation :set_paste_secret |
176
a9dba6a3008b
Initial work to add paste deletion.
Edho Arief <edho@myconan.net>
parents:
71
diff
changeset
|
10 validates :paste, :paste_hash, :key, :ip, :presence => true |
267
0bf1d6f75baa
Accidentally limited pastes to 0 characters.
edogawaconan <me@myconan.net>
parents:
265
diff
changeset
|
11 validates :paste, :length => { :maximum => 1_000_000 } |
2 | 12 |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
13 def to_param |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
14 path |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
15 end |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
16 |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
17 def self.safe_find(raw_id) |
425 | 18 /\A(?<id>[0-9]+)(?:-(?<secret>[0-9a-f]+))?\z/ =~ raw_id.to_s |
19 | |
431 | 20 find_by(:secret => secret, :id => id) |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
21 end |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
22 |
373
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
23 def self.graceful_create(params) |
376 | 24 paste = new(params) |
373
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
25 fresh = true |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
26 created = true |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
27 |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
28 begin |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
29 created = paste.save |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
30 rescue ActiveRecord::RecordNotUnique |
376 | 31 paste = find_by(:ip => paste.ip, :paste_hash => paste.paste_hash) |
373
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
32 fresh = false |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
33 end |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
34 |
376 | 35 [created, paste, fresh] |
373
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
36 end |
6e3e1e7b0212
Handle unique error in model instead of controller.
nanaya <me@myconan.net>
parents:
330
diff
changeset
|
37 |
316
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
38 def paste_gzip=(paste) |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
39 self.paste = ActiveSupport::Gzip.decompress paste |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
40 end |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
41 |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
42 def paste_gzip_base64=(paste) |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
43 self.paste_gzip = Base64.decode64(paste) |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
44 end |
61f7f258a6fb
Move from-gzip paste parsing to model.
edogawaconan <me@myconan.net>
parents:
290
diff
changeset
|
45 |
265
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
46 def safe_destroy(param_key) |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
47 if key == param_key |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
48 destroy |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
49 else |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
50 errors.add(:key, "is invalid") |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
51 false |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
52 end |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
53 end |
6cca1ab53337
Infinitely better error messages and notice.
edogawaconan <me@myconan.net>
parents:
255
diff
changeset
|
54 |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
55 def path |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
56 [id, secret.presence].compact.join("-") |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
57 end |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
58 |
2 | 59 def set_paste_hash |
51
36d07f047ec2
Or maybe not. Still too long with b62. Backed out changeset ba29d6394863
Edho Arief <edho@myconan.net>
parents:
46
diff
changeset
|
60 self.paste_hash = Digest::SHA512.hexdigest("#{paste}\n") |
2 | 61 end |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
62 |
176
a9dba6a3008b
Initial work to add paste deletion.
Edho Arief <edho@myconan.net>
parents:
71
diff
changeset
|
63 def set_paste_key |
a9dba6a3008b
Initial work to add paste deletion.
Edho Arief <edho@myconan.net>
parents:
71
diff
changeset
|
64 self.key ||= SecureRandom.hex(4) |
a9dba6a3008b
Initial work to add paste deletion.
Edho Arief <edho@myconan.net>
parents:
71
diff
changeset
|
65 end |
a9dba6a3008b
Initial work to add paste deletion.
Edho Arief <edho@myconan.net>
parents:
71
diff
changeset
|
66 |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
67 def set_paste_secret |
330 | 68 self.secret = SecureRandom.hex(4) if is_private? |
230
1c750d3cde1b
Correct way to test is_private flag.
Edho Arief <edho@myconan.net>
parents:
229
diff
changeset
|
69 end |
1c750d3cde1b
Correct way to test is_private flag.
Edho Arief <edho@myconan.net>
parents:
229
diff
changeset
|
70 |
1c750d3cde1b
Correct way to test is_private flag.
Edho Arief <edho@myconan.net>
parents:
229
diff
changeset
|
71 def is_private? |
330 | 72 is_private == "1" |
210
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
73 end |
d59731c3c7bf
Add support for is_private flag
Edho Arief <edho@myconan.net>
parents:
197
diff
changeset
|
74 |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
75 def convert_newlines |
431 | 76 self.paste = paste.to_s.gsub("\r\n", "\n").tr("\r", "\n") |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
77 end |
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
78 |
70
8f0fb869e770
Limit pastes to 100/hour per IP address.
Edho Arief <edho@myconan.net>
parents:
51
diff
changeset
|
79 def paste_limit |
377
4e6afc8140fb
Use Time.zone as per rubocop recommendation.
nanaya <me@myconan.net>
parents:
376
diff
changeset
|
80 ip_post_recent_count = self.class.where(:ip => ip).where("created_at > ?", Time.zone.now - 1.hour).count |
70
8f0fb869e770
Limit pastes to 100/hour per IP address.
Edho Arief <edho@myconan.net>
parents:
51
diff
changeset
|
81 errors.add :base, :limit if ip_post_recent_count > 100 |
8f0fb869e770
Limit pastes to 100/hour per IP address.
Edho Arief <edho@myconan.net>
parents:
51
diff
changeset
|
82 end |
8f0fb869e770
Limit pastes to 100/hour per IP address.
Edho Arief <edho@myconan.net>
parents:
51
diff
changeset
|
83 |
379 | 84 def set_privacy |
85 self.is_private ||= "0" | |
86 end | |
87 | |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
88 def self.fix_all |
255
a894d7696b7e
More useful return of Paste.fix_all.
edogawaconan <me@myconan.net>
parents:
247
diff
changeset
|
89 stats = Hash.new(0) |
376 | 90 all.find_each do |p| |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
91 p.save |
255
a894d7696b7e
More useful return of Paste.fix_all.
edogawaconan <me@myconan.net>
parents:
247
diff
changeset
|
92 stats[:count] += 1 |
a894d7696b7e
More useful return of Paste.fix_all.
edogawaconan <me@myconan.net>
parents:
247
diff
changeset
|
93 stats[:private] += 1 if p.secret |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
94 end |
330 | 95 stats |
22
032686a0c995
Added newline converter, integrity fixups.
Edho Arief <edho@myconan.net>
parents:
2
diff
changeset
|
96 end |
2 | 97 end |