Module: PHP

Defined in:
lib/smklib/php_serialize.rb

Overview

PHP serialize() and unserialize() workalikes

First Released: 2003-06-02 (1.0.0)
Prev Release: 2003-06-16 (1.0.1), by Thomas Hurst <[email protected]>
This Release: 2004-09-17 (1.0.2), by Thomas Hurst <[email protected]>
              Switch all {}'s to explicit Hash.new's.

These two methods should, for the most part, be functionally identical
to the respective PHP functions;
 http://www.php.net/serialize, http://www.php.net/unserialize

string = PHP.serialize(mixed var[, bool assoc])
 Returns a string representing the argument in a form PHP.unserialize
 and PHP's unserialize() should both be able to load.

 Array, Hash, Fixnum, Float, True/FalseClass, NilClass, String and Struct
 are supported; as are objects which support the to_assoc method, which
 returns an array of the form [['attr_name', 'value']..].  Anything else
 will raise a TypeError.

 If 'assoc' is specified, Array's who's first element is a two value
 array will be assumed to be an associative array, and will be serialized
 as a PHP associative array rather than a multidimensional array.

mixed = PHP.unserialize(string serialized, [hash classmap, [bool assoc]])
 Returns an object containing the reconstituted data from serialized.

 If a PHP array (associative; like an ordered hash) is encountered, it
 scans the keys; if they're all incrementing integers counting from 0,
 it's unserialized as an Array, otherwise it's unserialized as a Hash.
 Note: this will lose ordering.  To avoid this, specify assoc=true,
 and it will be unserialized as an associative array: [[key,value],...]

 If a serialized object is encountered, the hash 'classmap' is searched for
 the class name (as a symbol).  Since PHP classnames are not case-preserving,
 this *must* be a .capitalize()d representation.  The value is expected
 to be the class itself; i.e. something you could call .new on.

 If it's not found in 'classmap', the current constant namespace is searched,
 and failing that, a new Struct(classname) is generated, with the arguments
 for .new specified in the same order PHP provided; since PHP uses hashes
 to represent attributes, this should be the same order they're specified
 in PHP, but this is untested.

 each serialized attribute is sent to the new object using the respective
 {attribute}=() method; you'll get a NameError if the method doesn't exist.

 Array, Hash, Fixnum, Float, True/FalseClass, NilClass and String should
 be returned identically (i.e. foo == PHP.unserialize(PHP.serialize(foo))
 for these types); Struct should be too, provided it's in the namespace
 Module.const_get within unserialize() can see, or you gave it the same
 name in the Struct.new(<structname>), otherwise you should provide it in
 classmap.

Note: StringIO is required for unserialize(); it’s loaded as needed

Class Method Summary collapse

Class Method Details

.do_unserialize(string, classmap, assoc) ⇒ Object



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
# File 'lib/smklib/php_serialize.rb', line 163

def PHP.do_unserialize(string, classmap, assoc)
	val = nil
	# determine a type
	type = string.read(2)[0,1]
	case type
		when 'a' # associative array, a:length:{[index][value]...}
			count = string.read_until('{').to_i
			val = vals = Array.new
			count.times do |i|
				vals << [do_unserialize(string, classmap, assoc), do_unserialize(string, classmap, assoc)]
			end
			string.read(1) # skip the ending }

			unless assoc
				# now, we have an associative array, let's clean it up a bit...
				# arrays have all numeric indexes, in order; otherwise we assume a hash
				array = true
				i = 0
				vals.each do |key,value|
					if key != i # wrong index -> assume hash
						array = false
						break
					end
					i += 1
				end

				if array
					vals.collect! do |key,value|
						value
					end
				else
					val = Hash.new
					vals.each do |key,value|
						val[key] = value
					end
				end
			end

		when 'O' # object, O:length:"class":length:{[attribute][value]...}
			# class name (lowercase in PHP, grr)
			len = string.read_until(':').to_i + 3 # quotes, seperator
			klass = string.read(len)[1...-2].capitalize.intern # read it, kill useless quotes

			# read the attributes
			attrs = []
			len = string.read_until('{').to_i

			len.times do
				attr = (do_unserialize(string, classmap, assoc))
				attrs << [attr.intern, (attr << '=').intern, do_unserialize(string, classmap, assoc)]
			end
			string.read(1)

			val = nil
			# See if we need to map to a particular object
			if classmap.has_key?(klass)
				val = classmap[klass].new
			elsif Struct.const_defined?(klass) # Nope; see if there's a Struct
				classmap[klass] = val = Struct.const_get(klass)
				val = val.new
			else # Nope; see if there's a Constant
				begin
					classmap[klass] = val = Module.const_get(klass)

					val = val.new
				rescue NameError # Nope; make a new Struct
					classmap[klass] = val = Struct.new(klass.to_s, *attrs.collect { |v| v[0].to_s })
				end
			end

			attrs.each do |attr,attrassign,v|
				val.__send__(attrassign, v)
			end

		when 's' # string, s:length:"data";
			len = string.read_until(':').to_i + 3 # quotes, separator
			val = string.read(len)[1...-2] # read it, kill useless quotes

		when 'i' # integer, i:123
			val = string.read_until(';').to_i

		when 'd' # double (float), d:1.23
			val = string.read_until(';').to_f

		when 'N' # NULL, N;
			val = nil

		when 'b' # bool, b:0 or 1
			val = (string.read(2)[0] == ?1 ? true : false)

		else
			raise TypeError, "Unable to unserialize type '#{type}'"
	end

	val
end

.serialize(var, assoc = false) ⇒ Object

{{{



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
# File 'lib/smklib/php_serialize.rb', line 83

def PHP.serialize(var, assoc = false) # {{{
	s = ''
	case var
		when Array
			s << "a:#{var.size}:{"
			if assoc and var.first.is_a?(Array) and var.first.size == 2
				var.each { |k,v|
					s << PHP.serialize(k) << PHP.serialize(v)
				}
			else
				var.each_with_index { |v,i|
					s << "i:#{i};#{PHP.serialize(v)}"
				}
			end

			s << '}'

		when Hash
			s << "a:#{var.size}:{"
			var.each do |k,v|
				s << "#{PHP.serialize(k)}#{PHP.serialize(v)}"
			end
			s << '}'

		when Struct
			# encode as Object with same name
			s << "O:#{var.class.to_s.length}:\"#{var.class.to_s.downcase}\":#{var.members.length}:{"
			var.members.each do |member|
				s << "#{PHP.serialize(member)}#{PHP.serialize(var[member])}"
			end
			s << '}'

		when String
			s << "s:#{var.length}:\"#{var}\";"

		when Fixnum # PHP doesn't have bignums
			s << "i:#{var};"

		when Float
			s << "d:#{var};"

		when NilClass
			s << 'N;'

		when FalseClass, TrueClass
			s << "b:#{var ? 1 :0};"

		else
			if var.respond_to?(:to_assoc)
				v = var.to_assoc
				# encode as Object with same name
				s << "O:#{var.class.to_s.length}:\"#{var.class.to_s.downcase}\":#{v.length}:{"
				v.each do |k,v|
					s << "#{PHP.serialize(k.to_s)}#{PHP.serialize(v)}"
				end
				s << '}'
			else
				raise TypeError, "Unable to serialize type #{var.class}"
			end
	end

	s
end

.unserialize(string, classmap = nil, assoc = false) ⇒ Object

}}}



147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
# File 'lib/smklib/php_serialize.rb', line 147

def PHP.unserialize(string, classmap = nil, assoc = false) # {{{
	require 'stringio'
	string = StringIO.new(string)
	def string.read_until(char)
		val = ''
		while (c = self.read(1)) != char
			val << c
		end
		val
	end

	classmap ||= Hash.new

	do_unserialize(string, classmap, assoc)
end