util.array: Add support for + operator to create a new array from two arrays joined
[prosody.git] / util / array.lua
1 local array = {};
2
3 local array_mt = { __index = array, __tostring = function (array) return array:concat(", "); end };
4 local function new_array(_, t)
5         return setmetatable(t or {}, array_mt);
6 end
7
8 function array_mt.__add(a1, a2)
9         local res = new_array();
10         return res:append(a1):append(a2);
11 end
12
13 setmetatable(array, { __call = new_array });
14
15 function array:map(func, t2)
16         local t2 = t2 or array{};
17         for k,v in ipairs(self) do
18                 t2[k] = func(v);
19         end
20         return t2;
21 end
22
23 function array:filter(func, t2)
24         local t2 = t2 or array{};
25         for k,v in ipairs(self) do
26                 if func(v) then
27                         t2:push(v);
28                 end
29         end
30         return t2;
31 end
32
33
34 array.push = table.insert;
35 array.pop = table.remove;
36 array.sort = table.sort;
37 array.concat = table.concat;
38 array.length = function (t) return #t; end
39
40 function array:random()
41         return self[math.random(1,#self)];
42 end
43
44 function array:shuffle()
45         local len = #self;
46         for i=1,#self do
47                 local r = math.random(i,len);
48                 self[i], self[r] = self[r], self[i];
49         end
50         return self;
51 end
52
53 function array:reverse()
54         local len = #self-1;
55         for i=len,1,-1 do
56                 self:push(self[i]);
57                 self:pop(i);
58         end
59         return self;
60 end
61
62 function array:append(array)
63         local len,len2  = #self, #array;
64         for i=1,len2 do
65                 self[len+i] = array[i];
66         end
67         return self;
68 end
69
70 function array.collect(f, s, var)
71         local t, var = {};
72         while true do
73                 var = f(s, var);
74                 if var == nil then break; end
75                 table.insert(t, var);
76         end
77         return setmetatable(t, array_mt);
78 end
79
80 _G.array = array;
81 module("array");
82
83 return array;