PHP 8.2.31
Preview: tsort.rb Size: 14.14 KB
//proc/thread-self/root/opt/alt/ruby21/lib64/ruby/2.1.0/tsort.rb

#--
# tsort.rb - provides a module for topological sorting and strongly connected components.
#++
#

#
# TSort implements topological sorting using Tarjan's algorithm for
# strongly connected components.
#
# TSort is designed to be able to be used with any object which can be
# interpreted as a directed graph.
#
# TSort requires two methods to interpret an object as a graph,
# tsort_each_node and tsort_each_child.
#
# * tsort_each_node is used to iterate for all nodes over a graph.
# * tsort_each_child is used to iterate for child nodes of a given node.
#
# The equality of nodes are defined by eql? and hash since
# TSort uses Hash internally.
#
# == A Simple Example
#
# The following example demonstrates how to mix the TSort module into an
# existing class (in this case, Hash). Here, we're treating each key in
# the hash as a node in the graph, and so we simply alias the required
# #tsort_each_node method to Hash's #each_key method. For each key in the
# hash, the associated value is an array of the node's child nodes. This
# choice in turn leads to our implementation of the required #tsort_each_child
# method, which fetches the array of child nodes and then iterates over that
# array using the user-supplied block.
#
#   require 'tsort'
#
#   class Hash
#     include TSort
#     alias tsort_each_node each_key
#     def tsort_each_child(node, &block)
#       fetch(node).each(&block)
#     end
#   end
#
#   {1=>[2, 3], 2=>[3], 3=>[], 4=>[]}.tsort
#   #=> [3, 2, 1, 4]
#
#   {1=>[2], 2=>[3, 4], 3=>[2], 4=>[]}.strongly_connected_components
#   #=> [[4], [2, 3], [1]]
#
# == A More Realistic Example
#
# A very simple `make' like tool can be implemented as follows:
#
#   require 'tsort'
#
#   class Make
#     def initialize
#       @dep = {}
#       @dep.default = []
#     end
#
#     def rule(outputs, inputs=[], &block)
#       triple = [outputs, inputs, block]
#       outputs.each {|f| @dep[f] = [triple]}
#       @dep[triple] = inputs
#     end
#
#     def build(target)
#       each_strongly_connected_component_from(target) {|ns|
#         if ns.length != 1
#           fs = ns.delete_if {|n| Array === n}
#           raise TSort::Cyclic.new("cyclic dependencies: #{fs.join ', '}")
#         end
#         n = ns.first
#         if Array === n
#           outputs, inputs, block = n
#           inputs_time = inputs.map {|f| File.mtime f}.max
#           begin
#             outputs_time = outputs.map {|f| File.mtime f}.min
#           rescue Errno::ENOENT
#             outputs_time = nil
#           end
#           if outputs_time == nil ||
#              inputs_time != nil && outputs_time <= inputs_time
#             sleep 1 if inputs_time != nil && inputs_time.to_i == Time.now.to_i
#             block.call
#           end
#         end
#       }
#     end
#
#     def tsort_each_child(node, &block)
#       @dep[node].each(&block)
#     end
#     include TSort
#   end
#
#   def command(arg)
#     print arg, "\n"
#     system arg
#   end
#
#   m = Make.new
#   m.rule(%w[t1]) { command 'date > t1' }
#   m.rule(%w[t2]) { command 'date > t2' }
#   m.rule(%w[t3]) { command 'date > t3' }
#   m.rule(%w[t4], %w[t1 t3]) { command 'cat t1 t3 > t4' }
#   m.rule(%w[t5], %w[t4 t2]) { command 'cat t4 t2 > t5' }
#   m.build('t5')
#
# == Bugs
#
# * 'tsort.rb' is wrong name because this library uses
#   Tarjan's algorithm for strongly connected components.
#   Although 'strongly_connected_components.rb' is correct but too long.
#
# == References
#
# R. E. Tarjan, "Depth First Search and Linear Graph Algorithms",
# <em>SIAM Journal on Computing</em>, Vol. 1, No. 2, pp. 146-160, June 1972.
#

module TSort
  class Cyclic < StandardError
  end

  # Returns a topologically sorted array of nodes.
  # The array is sorted from children to parents, i.e.
  # the first element has no child and the last node has no parent.
  #
  # If there is a cycle, TSort::Cyclic is raised.
  #
  #   class G
  #     include TSort
  #     def initialize(g)
  #       @g = g
  #     end
  #     def tsort_each_child(n, &b) @g[n].each(&b) end
  #     def tsort_each_node(&b) @g.each_key(&b) end
  #   end
  #
  #   graph = G.new({1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]})
  #   p graph.tsort #=> [4, 2, 3, 1]
  #
  #   graph = G.new({1=>[2], 2=>[3, 4], 3=>[2], 4=>[]})
  #   p graph.tsort # raises TSort::Cyclic
  #
  def tsort
    each_node = method(:tsort_each_node)
    each_child = method(:tsort_each_child)
    TSort.tsort(each_node, each_child)
  end

  # Returns a topologically sorted array of nodes.
  # The array is sorted from children to parents, i.e.
  # the first element has no child and the last node has no parent.
  #
  # The graph is represented by _each_node_ and _each_child_.
  # _each_node_ should have +call+ method which yields for each node in the graph.
  # _each_child_ should have +call+ method which takes a node argument and yields for each child node.
  #
  # If there is a cycle, TSort::Cyclic is raised.
  #
  #   g = {1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   p TSort.tsort(each_node, each_child) #=> [4, 2, 3, 1]
  #
  #   g = {1=>[2], 2=>[3, 4], 3=>[2], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   p TSort.tsort(each_node, each_child) # raises TSort::Cyclic
  #
  def TSort.tsort(each_node, each_child)
    result = []
    TSort.tsort_each(each_node, each_child) {|element| result << element}
    result
  end

  # The iterator version of the #tsort method.
  # <tt><em>obj</em>.tsort_each</tt> is similar to <tt><em>obj</em>.tsort.each</tt>, but
  # modification of _obj_ during the iteration may lead to unexpected results.
  #
  # #tsort_each returns +nil+.
  # If there is a cycle, TSort::Cyclic is raised.
  #
  #   class G
  #     include TSort
  #     def initialize(g)
  #       @g = g
  #     end
  #     def tsort_each_child(n, &b) @g[n].each(&b) end
  #     def tsort_each_node(&b) @g.each_key(&b) end
  #   end
  #
  #   graph = G.new({1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]})
  #   graph.tsort_each {|n| p n }
  #   #=> 4
  #   #   2
  #   #   3
  #   #   1
  #
  def tsort_each(&block) # :yields: node
    each_node = method(:tsort_each_node)
    each_child = method(:tsort_each_child)
    TSort.tsort_each(each_node, each_child, &block)
  end

  # The iterator version of the TSort.tsort method.
  #
  # The graph is represented by _each_node_ and _each_child_.
  # _each_node_ should have +call+ method which yields for each node in the graph.
  # _each_child_ should have +call+ method which takes a node argument and yields for each child node.
  #
  #   g = {1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   TSort.tsort_each(each_node, each_child) {|n| p n }
  #   #=> 4
  #   #   2
  #   #   3
  #   #   1
  #
  def TSort.tsort_each(each_node, each_child) # :yields: node
    TSort.each_strongly_connected_component(each_node, each_child) {|component|
      if component.size == 1
        yield component.first
      else
        raise Cyclic.new("topological sort failed: #{component.inspect}")
      end
    }
  end

  # Returns strongly connected components as an array of arrays of nodes.
  # The array is sorted from children to parents.
  # Each elements of the array represents a strongly connected component.
  #
  #   class G
  #     include TSort
  #     def initialize(g)
  #       @g = g
  #     end
  #     def tsort_each_child(n, &b) @g[n].each(&b) end
  #     def tsort_each_node(&b) @g.each_key(&b) end
  #   end
  #
  #   graph = G.new({1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]})
  #   p graph.strongly_connected_components #=> [[4], [2], [3], [1]]
  #
  #   graph = G.new({1=>[2], 2=>[3, 4], 3=>[2], 4=>[]})
  #   p graph.strongly_connected_components #=> [[4], [2, 3], [1]]
  #
  def strongly_connected_components
    each_node = method(:tsort_each_node)
    each_child = method(:tsort_each_child)
    TSort.strongly_connected_components(each_node, each_child)
  end

  # Returns strongly connected components as an array of arrays of nodes.
  # The array is sorted from children to parents.
  # Each elements of the array represents a strongly connected component.
  #
  # The graph is represented by _each_node_ and _each_child_.
  # _each_node_ should have +call+ method which yields for each node in the graph.
  # _each_child_ should have +call+ method which takes a node argument and yields for each child node.
  #
  #   g = {1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   p TSort.strongly_connected_components(each_node, each_child)
  #   #=> [[4], [2], [3], [1]]
  #
  #   g = {1=>[2], 2=>[3, 4], 3=>[2], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   p TSort.strongly_connected_components(each_node, each_child)
  #   #=> [[4], [2, 3], [1]]
  #
  def TSort.strongly_connected_components(each_node, each_child)
    result = []
    TSort.each_strongly_connected_component(each_node, each_child) {|component| result << component}
    result
  end

  # The iterator version of the #strongly_connected_components method.
  # <tt><em>obj</em>.each_strongly_connected_component</tt> is similar to
  # <tt><em>obj</em>.strongly_connected_components.each</tt>, but
  # modification of _obj_ during the iteration may lead to unexpected results.
  #
  # #each_strongly_connected_component returns +nil+.
  #
  #   class G
  #     include TSort
  #     def initialize(g)
  #       @g = g
  #     end
  #     def tsort_each_child(n, &b) @g[n].each(&b) end
  #     def tsort_each_node(&b) @g.each_key(&b) end
  #   end
  #
  #   graph = G.new({1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]})
  #   graph.each_strongly_connected_component {|scc| p scc }
  #   #=> [4]
  #   #   [2]
  #   #   [3]
  #   #   [1]
  #
  #   graph = G.new({1=>[2], 2=>[3, 4], 3=>[2], 4=>[]})
  #   graph.each_strongly_connected_component {|scc| p scc }
  #   #=> [4]
  #   #   [2, 3]
  #   #   [1]
  #
  def each_strongly_connected_component(&block) # :yields: nodes
    each_node = method(:tsort_each_node)
    each_child = method(:tsort_each_child)
    TSort.each_strongly_connected_component(each_node, each_child, &block)
  end

  # The iterator version of the TSort.strongly_connected_components method.
  #
  # The graph is represented by _each_node_ and _each_child_.
  # _each_node_ should have +call+ method which yields for each node in the graph.
  # _each_child_ should have +call+ method which takes a node argument and yields for each child node.
  #
  #   g = {1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   TSort.each_strongly_connected_component(each_node, each_child) {|scc| p scc }
  #   #=> [4]
  #   #   [2]
  #   #   [3]
  #   #   [1]
  #
  #   g = {1=>[2], 2=>[3, 4], 3=>[2], 4=>[]}
  #   each_node = lambda {|&b| g.each_key(&b) }
  #   each_child = lambda {|n, &b| g[n].each(&b) }
  #   TSort.each_strongly_connected_component(each_node, each_child) {|scc| p scc }
  #   #=> [4]
  #   #   [2, 3]
  #   #   [1]
  #
  def TSort.each_strongly_connected_component(each_node, each_child) # :yields: nodes
    id_map = {}
    stack = []
    each_node.call {|node|
      unless id_map.include? node
        TSort.each_strongly_connected_component_from(node, each_child, id_map, stack) {|c|
          yield c
        }
      end
    }
    nil
  end

  # Iterates over strongly connected component in the subgraph reachable from
  # _node_.
  #
  # Return value is unspecified.
  #
  # #each_strongly_connected_component_from doesn't call #tsort_each_node.
  #
  #   class G
  #     include TSort
  #     def initialize(g)
  #       @g = g
  #     end
  #     def tsort_each_child(n, &b) @g[n].each(&b) end
  #     def tsort_each_node(&b) @g.each_key(&b) end
  #   end
  #
  #   graph = G.new({1=>[2, 3], 2=>[4], 3=>[2, 4], 4=>[]})
  #   graph.each_strongly_connected_component_from(2) {|scc| p scc }
  #   #=> [4]
  #   #   [2]
  #
  #   graph = G.new({1=>[2], 2=>[3, 4], 3=>[2], 4=>[]})
  #   graph.each_strongly_connected_component_from(2) {|scc| p scc }
  #   #=> [4]
  #   #   [2, 3]
  #
  def each_strongly_connected_component_from(node, id_map={}, stack=[], &block) # :yields: nodes
    TSort.each_strongly_connected_component_from(node, method(:tsort_each_child), id_map, stack, &block)
  end

  # Iterates over strongly connected components in a graph.
  # The graph is represented by _node_ and _each_child_.
  #
  # _node_ is the first node.
  # _each_child_ should have +call+ method which takes a node argument
  # and yields for each child node.
  #
  # Return value is unspecified.
  #
  # #TSort.each_strongly_connected_component_from is a class method and
  # it doesn't need a class to represent a graph which includes TSort.
  #
  #   graph = {1=>[2], 2=>[3, 4], 3=>[2], 4=>[]}
  #   each_child = lambda {|n, &b| graph[n].each(&b) }
  #   TSort.each_strongly_connected_component_from(1, each_child) {|scc|
  #     p scc
  #   }
  #   #=> [4]
  #   #   [2, 3]
  #   #   [1]
  #
  def TSort.each_strongly_connected_component_from(node, each_child, id_map={}, stack=[]) # :yields: nodes
    minimum_id = node_id = id_map[node] = id_map.size
    stack_length = stack.length
    stack << node

    each_child.call(node) {|child|
      if id_map.include? child
        child_id = id_map[child]
        minimum_id = child_id if child_id && child_id < minimum_id
      else
        sub_minimum_id =
          TSort.each_strongly_connected_component_from(child, each_child, id_map, stack) {|c|
            yield c
          }
        minimum_id = sub_minimum_id if sub_minimum_id < minimum_id
      end
    }

    if node_id == minimum_id
      component = stack.slice!(stack_length .. -1)
      component.each {|n| id_map[n] = nil}
      yield component
    end

    minimum_id
  end

  # Should be implemented by a extended class.
  #
  # #tsort_each_node is used to iterate for all nodes over a graph.
  #
  def tsort_each_node # :yields: node
    raise NotImplementedError.new
  end

  # Should be implemented by a extended class.
  #
  # #tsort_each_child is used to iterate for child nodes of _node_.
  #
  def tsort_each_child(node) # :yields: child
    raise NotImplementedError.new
  end
end

Directory Contents

Dirs: 31 × Files: 77

Name Size Perms Modified Actions
cgi DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
date DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
digest DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
dl DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
drb DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
fiddle DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
io DIR
- drwxr-xr-x 2024-03-03 22:43:07
Edit Download
irb DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
json DIR
- drwxr-xr-x 2024-03-03 22:43:07
Edit Download
matrix DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
net DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
openssl DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
optparse DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
psych DIR
- drwxr-xr-x 2024-03-03 22:43:08
Edit Download
racc DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
rake DIR
- drwxr-xr-x 2024-03-03 22:53:20
Edit Download
rbconfig DIR
- drwxr-xr-x 2024-03-03 22:43:11
Edit Download
rdoc DIR
- drwxr-xr-x 2024-03-03 22:43:10
Edit Download
rexml DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
rinda DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
ripper DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
rss DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
rubygems DIR
- drwxr-xr-x 2024-03-03 22:43:11
Edit Download
shell DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
syslog DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
test DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
uri DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
webrick DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
- drwxr-xr-x 2024-03-03 22:43:08
Edit Download
xmlrpc DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
yaml DIR
- drwxr-xr-x 2024-03-03 22:43:05
Edit Download
3.31 KB lrw-r--r-- 2013-02-21 17:35:32
Edit Download
2.63 KB lrw-r--r-- 2013-05-19 03:10:21
Edit Download
17.95 KB lrw-r--r-- 2013-09-20 16:05:48
Edit Download
9.84 KB lrw-r--r-- 2013-07-18 13:50:32
Edit Download
8.93 KB lrw-r--r-- 2015-07-03 09:12:53
Edit Download
380 B lrw-r--r-- 2009-08-16 15:34:35
Edit Download
81.68 KB lrw-r--r-- 2014-09-14 15:25:34
Edit Download
946 B lrw-r--r-- 2023-07-26 14:22:11
Edit Download
29.08 KB lrw-r--r-- 2013-12-14 14:48:36
Edit Download
11.13 KB lrw-r--r-- 2014-02-02 13:46:46
Edit Download
2.34 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
280 B lrw-r--r-- 2023-07-26 14:22:11
Edit Download
19 B lrw-r--r-- 2009-10-02 10:45:39
Edit Download
3.89 KB lrw-r--r-- 2013-12-17 11:20:10
Edit Download
6.44 KB lrw-r--r-- 2013-02-04 02:50:20
Edit Download
26.07 KB lrw-r--r-- 2014-08-11 14:55:58
Edit Download
2.14 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
1.65 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
47.17 KB lrw-r--r-- 2015-04-28 05:16:46
Edit Download
2.36 KB lrw-r--r-- 2014-10-15 15:31:03
Edit Download
7.86 KB lrw-r--r-- 2013-05-23 21:46:43
Edit Download
15.38 KB lrw-r--r-- 2013-05-19 14:50:47
Edit Download
8.86 KB lrw-r--r-- 2014-07-06 13:57:45
Edit Download
17.05 KB lrw-r--r-- 2015-10-29 14:36:05
Edit Download
20.03 KB lrw-r--r-- 2016-03-25 08:22:04
Edit Download
1.74 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
5.74 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
22.37 KB lrw-r--r-- 2016-02-25 11:04:33
Edit Download
6.52 KB lrw-r--r-- 2011-08-26 22:22:37
Edit Download
45.43 KB lrw-r--r-- 2014-07-11 14:08:51
Edit Download
80.50 KB lrw-r--r-- 2023-07-26 14:19:40
Edit Download
6.93 KB lrw-r--r-- 2012-11-16 16:55:29
Edit Download
2.00 KB lrw-r--r-- 2013-02-20 02:51:51
Edit Download
5.81 KB lrw-r--r-- 2013-11-21 04:44:45
Edit Download
24.27 KB lrw-r--r-- 2014-02-22 08:54:43
Edit Download
20.37 KB lrw-r--r-- 2013-12-01 03:13:11
Edit Download
528 B lrw-r--r-- 2023-07-26 14:22:11
Edit Download
50.80 KB lrw-r--r-- 2013-11-28 07:34:15
Edit Download
7.70 KB lrw-r--r-- 2013-10-23 15:14:17
Edit Download
15.30 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
14.10 KB lrw-r--r-- 2013-10-22 09:29:53
Edit Download
16.33 KB lrw-r--r-- 2013-05-19 03:10:21
Edit Download
13.20 KB lrw-r--r-- 2013-07-15 04:21:34
Edit Download
205 B lrw-r--r-- 2009-10-02 10:45:39
Edit Download
4.51 KB lrw-r--r-- 2013-05-19 23:04:36
Edit Download
14.85 KB lrw-r--r-- 2012-11-11 04:23:04
Edit Download
14.88 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
2.12 KB lrw-r--r-- 2013-10-11 21:35:01
Edit Download
308 B lrw-r--r-- 2009-09-24 00:42:23
Edit Download
4.92 KB lrw-r--r-- 2013-12-22 10:25:26
Edit Download
1.73 KB lrw-r--r-- 2013-03-11 13:47:04
Edit Download
72.31 KB lrw-r--r-- 2015-05-13 05:33:45
Edit Download
2.53 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
2.84 KB lrw-r--r-- 2011-05-11 10:22:16
Edit Download
30.97 KB lrw-r--r-- 2023-07-26 14:19:40
Edit Download
23.54 KB lrw-r--r-- 2013-12-14 02:55:33
Edit Download
8.49 KB lrw-r--r-- 2013-11-07 18:04:05
Edit Download
18.70 KB lrw-r--r-- 2013-11-22 23:50:06
Edit Download
10.30 KB lrw-r--r-- 2013-05-19 03:10:21
Edit Download
5.94 KB lrw-r--r-- 2013-12-14 18:26:59
Edit Download
4.02 KB lrw-r--r-- 2011-05-18 14:09:38
Edit Download
25.60 KB lrw-r--r-- 2023-07-26 14:22:11
Edit Download
7.25 KB lrw-r--r-- 2013-05-19 03:10:21
Edit Download
11.40 KB lrw-r--r-- 2013-11-21 09:28:43
Edit Download
3.38 KB lrw-r--r-- 2013-11-07 17:02:56
Edit Download
21.32 KB lrw-r--r-- 2015-08-17 07:41:58
Edit Download
3.70 KB lrw-r--r-- 2015-08-17 07:55:26
Edit Download
4.15 KB lrw-r--r-- 2013-05-19 03:10:21
Edit Download
6.40 KB lrw-r--r-- 2013-07-18 13:50:32
Edit Download
14.14 KB lrw-r--r-- 2013-10-17 15:59:40
Edit Download
268 B lrw-r--r-- 2009-10-02 10:45:39
Edit Download
8.87 KB lrw-r--r-- 2013-07-05 13:43:25
Edit Download
3.07 KB lrw-r--r-- 2011-05-13 20:03:21
Edit Download
3.23 KB lrw-r--r-- 2013-11-09 22:42:20
Edit Download
6.69 KB lrw-r--r-- 2013-10-05 23:39:32
Edit Download
8.49 KB lrw-r--r-- 2013-12-12 03:09:19
Edit Download
2.31 KB lrw-r--r-- 2013-08-12 03:49:50
Edit Download

If ZipArchive is unavailable, a .tar will be created (no compression).