jdunphy-sequel_revisioned 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/History.txt +4 -0
- data/Manifest.txt +5 -0
- data/README.rdoc +62 -0
- data/lib/sequel_revisioned.rb +12 -0
- data/lib/sequel_revisioned/sequel_revisioned.rb +116 -0
- metadata +87 -0
data/History.txt
ADDED
data/Manifest.txt
ADDED
data/README.rdoc
ADDED
@@ -0,0 +1,62 @@
|
|
1
|
+
= sequel_revisioned
|
2
|
+
|
3
|
+
== DESCRIPTION:
|
4
|
+
|
5
|
+
Sequel plugin designed to maintain a revision history of an object.
|
6
|
+
|
7
|
+
Sequel_revisioned will create a revision table and model specific to the class that calls
|
8
|
+
`is :revisioned`. It's set up this way to allow multiple tables to maintain revisions
|
9
|
+
without interfering with each other.
|
10
|
+
|
11
|
+
== USAGE
|
12
|
+
|
13
|
+
class Post < Sequel::Model
|
14
|
+
is :revisioned, :watch => [:title, :body, :synopsis]
|
15
|
+
|
16
|
+
end
|
17
|
+
|
18
|
+
$ p = Post[n]
|
19
|
+
$ p.revisions
|
20
|
+
=> Array of post's revisions
|
21
|
+
$ p.roll_back(x)
|
22
|
+
=> Rolls watched fields back to the sate of revision x
|
23
|
+
|
24
|
+
$ revision = p.revisions.first
|
25
|
+
$ revision.class.name
|
26
|
+
=> PostRevision
|
27
|
+
$ revision.table_name
|
28
|
+
=> post_revisions
|
29
|
+
|
30
|
+
|
31
|
+
== REQUIREMENTS:
|
32
|
+
|
33
|
+
Sequel >= 2.9.0
|
34
|
+
|
35
|
+
== INSTALL:
|
36
|
+
|
37
|
+
sudo gem install jdunphy-sequel_revisioned --source http://gems.github.com
|
38
|
+
|
39
|
+
== LICENSE:
|
40
|
+
|
41
|
+
(The MIT License)
|
42
|
+
|
43
|
+
Copyright (c) 2009 Jacob Dunphy
|
44
|
+
|
45
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
46
|
+
a copy of this software and associated documentation files (the
|
47
|
+
'Software'), to deal in the Software without restriction, including
|
48
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
49
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
50
|
+
permit persons to whom the Software is furnished to do so, subject to
|
51
|
+
the following conditions:
|
52
|
+
|
53
|
+
The above copyright notice and this permission notice shall be
|
54
|
+
included in all copies or substantial portions of the Software.
|
55
|
+
|
56
|
+
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
|
57
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
58
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
59
|
+
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
60
|
+
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
61
|
+
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
62
|
+
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
@@ -0,0 +1,12 @@
|
|
1
|
+
$:.unshift(File.dirname(__FILE__)) unless
|
2
|
+
$:.include?(File.dirname(__FILE__)) || $:.include?(File.expand_path(File.dirname(__FILE__)))
|
3
|
+
|
4
|
+
module SequelRevisioned
|
5
|
+
VERSION = '0.0.1'
|
6
|
+
end
|
7
|
+
|
8
|
+
module Sequel
|
9
|
+
class InvalidRevisionError < StandardError; end
|
10
|
+
end
|
11
|
+
|
12
|
+
require 'sequel_revisioned/sequel_revisioned'
|
@@ -0,0 +1,116 @@
|
|
1
|
+
module Sequel
|
2
|
+
module Plugins
|
3
|
+
module Revisioned
|
4
|
+
|
5
|
+
def self.apply(model, options = {})
|
6
|
+
revision_model_name = "::#{model.name}Revision"
|
7
|
+
eval "
|
8
|
+
class #{revision_model_name} < Sequel::Model
|
9
|
+
before_create :set_created_at
|
10
|
+
before_create :set_version
|
11
|
+
many_to_one :#{model.name.underscore}
|
12
|
+
|
13
|
+
def next_revision
|
14
|
+
if version && post_id
|
15
|
+
self.class.find(:post_id => post_id, :version => version + 1)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
def previous_revision
|
20
|
+
if version && post_id && version > 1
|
21
|
+
self.class.find(:post_id => post_id, :version => version - 1)
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
private
|
26
|
+
|
27
|
+
def set_version
|
28
|
+
if last_revision = #{model.name.underscore}.revisions.first
|
29
|
+
self.version = last_revision.version + 1
|
30
|
+
else
|
31
|
+
self.version = 1
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def set_created_at
|
36
|
+
self.created_at ||= Time.now
|
37
|
+
end
|
38
|
+
end
|
39
|
+
"
|
40
|
+
revision_model = revision_model_name.constantize
|
41
|
+
unless revision_model.table_exists?
|
42
|
+
migration = "
|
43
|
+
class CreateRevisions < Sequel::Migration
|
44
|
+
def up
|
45
|
+
create_table :#{revision_model.table_name} do
|
46
|
+
primary_key :id
|
47
|
+
integer :#{model.name.underscore}_id
|
48
|
+
integer :version
|
49
|
+
timestamp :created_at
|
50
|
+
#{additional_migration_columns(model, options)}
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
CreateRevisions.apply(DB, :up)
|
55
|
+
#{revision_model_name}.columns"
|
56
|
+
# TODO - This last line feels really hacky.
|
57
|
+
# The #generate_revision code below doesn't work without it.
|
58
|
+
# It seems like I have to load the schema into the model
|
59
|
+
eval migration
|
60
|
+
end
|
61
|
+
|
62
|
+
model.class_eval "
|
63
|
+
one_to_many :revisions, :class => '#{revision_model}', :order => :version.desc
|
64
|
+
after_save :generate_revision
|
65
|
+
@@watched_columns = [#{Array(options[:watch]).map {|w| ":#{w.to_s}"}.join(",")}]
|
66
|
+
|
67
|
+
def self.revision_class; #{revision_model}; end
|
68
|
+
"
|
69
|
+
end
|
70
|
+
|
71
|
+
def self.additional_migration_columns(model, options = {})
|
72
|
+
if options[:watch]
|
73
|
+
Array(options[:watch]).map do |column|
|
74
|
+
"#{model.db_schema[column.to_sym][:type]} :#{column}"
|
75
|
+
end.join("\n")
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
module InstanceMethods
|
80
|
+
|
81
|
+
def roll_back(version)
|
82
|
+
revision = revisions.detect {|rev| rev.version == version.to_i }
|
83
|
+
raise Sequel::InvalidRevisionError unless revision
|
84
|
+
self.class.watched_columns.each do |col|
|
85
|
+
send("#{col}=", revision.send(col))
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
def watched_data
|
90
|
+
data = {}
|
91
|
+
self.class.watched_columns.each do |col|
|
92
|
+
data[col] = send(col)
|
93
|
+
end
|
94
|
+
data
|
95
|
+
end
|
96
|
+
private :watched_data
|
97
|
+
|
98
|
+
def generate_revision
|
99
|
+
revision = self.class.revision_class.new(watched_data)
|
100
|
+
add_revision(revision)
|
101
|
+
revision.save
|
102
|
+
end
|
103
|
+
private :generate_revision
|
104
|
+
end
|
105
|
+
|
106
|
+
module ClassMethods
|
107
|
+
|
108
|
+
def watched_columns
|
109
|
+
class_variable_get(:@@watched_columns)
|
110
|
+
end
|
111
|
+
end
|
112
|
+
|
113
|
+
end
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
metadata
ADDED
@@ -0,0 +1,87 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: jdunphy-sequel_revisioned
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Jacob Dunphy
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
|
12
|
+
date: 2009-01-19 00:00:00 -08:00
|
13
|
+
default_executable:
|
14
|
+
dependencies:
|
15
|
+
- !ruby/object:Gem::Dependency
|
16
|
+
name: sequel
|
17
|
+
version_requirement:
|
18
|
+
version_requirements: !ruby/object:Gem::Requirement
|
19
|
+
requirements:
|
20
|
+
- - ">="
|
21
|
+
- !ruby/object:Gem::Version
|
22
|
+
version: 2.9.0
|
23
|
+
version:
|
24
|
+
- !ruby/object:Gem::Dependency
|
25
|
+
name: newgem
|
26
|
+
version_requirement:
|
27
|
+
version_requirements: !ruby/object:Gem::Requirement
|
28
|
+
requirements:
|
29
|
+
- - ">="
|
30
|
+
- !ruby/object:Gem::Version
|
31
|
+
version: 1.2.3
|
32
|
+
version:
|
33
|
+
- !ruby/object:Gem::Dependency
|
34
|
+
name: hoe
|
35
|
+
version_requirement:
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - ">="
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: 1.8.0
|
41
|
+
version:
|
42
|
+
description: Sequel plugin designed to maintain a revision history of an object. Sequel_revisioned will create a revision table and model specific to the class that calls `is :revisioned`. It's set up this way to allow multiple tables to maintain revisions without interfering with each other.
|
43
|
+
email:
|
44
|
+
- jacob.dunphy@gmail.com
|
45
|
+
executables: []
|
46
|
+
|
47
|
+
extensions: []
|
48
|
+
|
49
|
+
extra_rdoc_files:
|
50
|
+
- History.txt
|
51
|
+
- Manifest.txt
|
52
|
+
- README.rdoc
|
53
|
+
files:
|
54
|
+
- History.txt
|
55
|
+
- Manifest.txt
|
56
|
+
- README.rdoc
|
57
|
+
- lib/sequel_revisioned.rb
|
58
|
+
- lib/sequel_revisioned/sequel_revisioned.rb
|
59
|
+
has_rdoc: true
|
60
|
+
homepage:
|
61
|
+
post_install_message:
|
62
|
+
rdoc_options:
|
63
|
+
- --main
|
64
|
+
- README.rdoc
|
65
|
+
require_paths:
|
66
|
+
- lib
|
67
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
68
|
+
requirements:
|
69
|
+
- - ">="
|
70
|
+
- !ruby/object:Gem::Version
|
71
|
+
version: "0"
|
72
|
+
version:
|
73
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
74
|
+
requirements:
|
75
|
+
- - ">="
|
76
|
+
- !ruby/object:Gem::Version
|
77
|
+
version: "0"
|
78
|
+
version:
|
79
|
+
requirements: []
|
80
|
+
|
81
|
+
rubyforge_project: sequel_revisioned
|
82
|
+
rubygems_version: 1.2.0
|
83
|
+
signing_key:
|
84
|
+
specification_version: 2
|
85
|
+
summary: Sequel plugin designed to maintain a revision history of an object
|
86
|
+
test_files: []
|
87
|
+
|