This file is indexed.

/usr/share/lua/5.1/apr/serialize.lua is in lua-apr 0.23.2.dfsg-4.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  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
--[[

 Serialization function for the Lua/APR binding.

 Last Change: November 20, 2011
 Homepage: http://peterodding.com/code/lua/apr/
 License: MIT

 Based on the table-to-source serializer included in Metalua, which is
 copyright (c) 2008-2009, Fabien Fleutot <metalua@gmail.com>. Retrieved
 from https://github.com/fab13n/metalua/blob/master/src/lib/serialize.lua.

 Minor changes by Peter Odding <peter@peterodding.com> to serialize
 function upvalues and userdata objects created by the Lua/APR binding.

 Inline documentation can be found in "serialize.c" because I'm too lazy to
 change my documentation generator...

]]

local no_identity = {
  ['nil'] = true,
  ['boolean'] = true,
  ['number'] = true,
  ['string'] = true,
}

local function serialize(x)

  local gensym_max = 0  -- index of the gensym() symbol generator
  local seen_once = {} -- element->true set of elements seen exactly once in the table
  local multiple = {} -- element->varname set of elements seen more than once
  local nested = {} -- transient, set of elements currently being traversed
  local nest_points = {}
  local nest_patches = {}

  -- Generate fresh indexes to store new sub-tables:
  local function gensym()
    gensym_max = gensym_max + 1
    return gensym_max
  end

  -----------------------------------------------------------------------------
  -- `nest_points' are places where a (recursive) table appears within
  -- itself, directly or not.  for instance, all of these chunks
  -- create nest points in table `x':
  --
  -- "x = {}; x[x] = 1"
  -- "x = {}; x[1] = x"
  -- "x = {}; x[1] = { y = { x } }".
  --
  -- To handle those, two tables are created by `mark_nest_point()':
  --
  -- * `nest_points[parent]' associates all keys and values in table
  --   parent which create a nest_point with boolean `true'
  --
  -- * `nest_patches' contains a list of `{ parent, key, value }'
  --   tuples creating a nest point. They're all dumped after all the
  --   other table operations have been performed.
  --
  -- `mark_nest_point(p, k, v)' fills tables `nest_points' and
  -- `nest_patches' with information required to remember that
  -- key/value `(k,v)' creates a nest point in parent table `p'.
  -- It also marks `p' as occurring multiple times, since several
  -- references to it will be required in order to patch the nest
  -- points.
  -----------------------------------------------------------------------------
  local function mark_nest_point(parent, k, v)
    local nk, nv = nested[k], nested[v]
    assert(not nk or seen_once[k] or multiple[k])
    assert(not nv or seen_once[v] or multiple[v])
    local parent_np = nest_points[parent]
    if not parent_np then
      parent_np = {}
      nest_points[parent] = parent_np
    end
    if nk then parent_np[k] = true end
    if nv then parent_np[v] = true end
    table.insert(nest_patches, { parent, k, v })
    seen_once[parent] = nil
    multiple[parent] = true
  end

  -----------------------------------------------------------------------------
  -- 1st pass, list the tables and functions which appear more than once in `x'
  -----------------------------------------------------------------------------
  local function mark_multiple_occurrences(x)
    local t = type(x)
    if no_identity[t] then
      return
    elseif seen_once[x] then
      seen_once[x] = nil
      multiple[x] = true
    elseif not multiple[x] then
      seen_once[x] = true
    end
    if t == 'table' then
      nested[x] = true
      for k, v in pairs(x) do
        if nested[k] or nested[v] then
          mark_nest_point(x, k, v)
        else
          mark_multiple_occurrences(k)
          mark_multiple_occurrences(v)
        end
      end
      nested[x] = nil
    elseif t == 'function' then
      for i = 1, math.huge do
        local n, v = debug.getupvalue(x, i)
        if n then mark_multiple_occurrences(v) else break end
      end
    end
  end

  local dumped = {} -- multiply occurring values already dumped in localdefs
  local localdefs = {} -- already dumped local definitions as source code lines

  -- mutually recursive functions:
  local dump_val, dump_or_ref_val

  ------------------------------------------------------------------------------
  -- if `x' occurs multiple times, dump the local var rather than the
  -- value. If it's the first time it's dumped, also dump the content
  -- in localdefs.
  ------------------------------------------------------------------------------
  function dump_or_ref_val(x)
    if nested[x] then
      -- placeholder for recursive reference
      return 'false'
    elseif not multiple[x] then
      -- value referenced only once, dump directly
      return dump_val(x)
    else
      local var = dumped[x]
      if var then
        -- already referenced
        return '_[' .. var .. ']'
      else
        -- first occutrence, create and register reference
        local val = dump_val(x)
        var = gensym()
        table.insert(localdefs, '_[' .. var .. '] = ' .. val)
        dumped[x] = var
        return '_[' .. var .. ']'
      end
    end
  end

  -----------------------------------------------------------------------------
  -- 2nd pass, dump the object; subparts occurring multiple times are dumped
  -- in local variables, which can then be referenced multiple times;
  -- care is taken to dump local vars in an order which respect dependencies.
  -----------------------------------------------------------------------------
  function dump_val(x)
    local t = type(x)
    if x == nil then
      return 'nil'
    elseif t == 'number' then
      return x == math.huge and 'math.huge' or string.format('%.16f', x)
    elseif t == 'string' then
      return string.format('%q', x)
    elseif t == 'boolean' then
      return x and 'true' or 'false'
    elseif t == 'function' then
      local body = string.format("loadstring(%q, '@serialized')", string.dump(x))
      if not debug.getupvalue(x, 1) then return body end
      local acc = {}
      -- FIXME This doesn't actually need an anonymous function, I'm just lazy.
      table.insert(acc, '(function()')
      table.insert(acc, 'local f = ' .. body)
      for i = 1, math.huge do
        local n, v = debug.getupvalue(x, i)
        if not n then break end
        table.insert(acc, string.format('debug.setupvalue(f, %i, %s)', i, dump_or_ref_val(v)))
      end
      table.insert(acc, 'return f')
      table.insert(acc, 'end)()')
      return table.concat(acc, '\n')
    elseif t == 'table' then
      local acc = {}
      local idx_dumped = {}
      local np = nest_points[x]
      for i, v in ipairs(x) do
        if np and np[v] then
          table.insert(acc, 'false') -- placeholder
        else
          table.insert(acc, dump_or_ref_val(v))
        end
        idx_dumped[i] = true
      end
      for k, v in pairs(x) do
        if np and (np[k] or np[v]) then
          --check_multiple(k); check_multiple(v) -- force dumps in localdefs
        elseif not idx_dumped[k] then
          table.insert(acc, '[' .. dump_or_ref_val(k) .. '] = ' .. dump_or_ref_val(v))
        end
      end
      return '{ ' .. table.concat(acc, ', ') .. ' }'
    elseif t == 'userdata' then
      local apr = require 'apr'
      return string.format("require('apr').deref(%q)", apr.ref(x))
    else
      error("Can't serialize data of type " .. t)
    end
  end

  -- Patch the recursive table entries:
  local function dump_nest_patches()
    for _, entry in ipairs(nest_patches) do
      local p, k, v = unpack(entry)
      assert(multiple[p])
      table.insert(localdefs, dump_or_ref_val(p)
          .. '[' .. dump_or_ref_val(k) .. '] = '
          .. dump_or_ref_val(v) .. ' -- rec')
    end
  end

  mark_multiple_occurrences(x)
  local toplevel = dump_or_ref_val(x)
  dump_nest_patches()

  if next(localdefs) then
    -- Dump local vars containing shared or recursive parts,
    -- then the main table using them.
    return 'local _={}\n' ..
      table.concat(localdefs, '\n') ..
      '\nreturn ' .. toplevel
  else
    -- No shared part, straightforward dump:
    return 'return ' .. toplevel
  end

end

return serialize

-- vim: ts=2 sw=2 et