Ruby Создать рекурсивное дерево каталогов - PullRequest
12 голосов
/ 02 апреля 2011

Мне нужно рекурсивно пройти по каталогу и создать дерево для использования с элементом управления jsTree .Элемент управления принимает формат JSON примерно так .Мне нужно немного рубиновой магии, чтобы сделать это чисто и быстро.

Любая помощь приветствуется.

Ответы [ 4 ]

25 голосов
/ 02 апреля 2011

Возможно, вы хотите что-то вроде этого (не проверено):

def directory_hash(path, name=nil)
  data = {:data => (name || path)}
  data[:children] = children = []
  Dir.foreach(path) do |entry|
    next if (entry == '..' || entry == '.')
    full_path = File.join(path, entry)
    if File.directory?(full_path)
      children << directory_hash(full_path, entry)
    else
      children << entry
    end
  end
  return data
end

Рекурсивно ходить по дереву, создавая хэш.Превратите его в json с вашей любимой библиотекой сериализации.

7 голосов
/ 02 апреля 2011

Сначала возьмите свое дерево, преобразуйте его в список путей к листьям, аналогично:

def leaves_paths tree
  if tree[:children]
    tree[:children].inject([]){|acc, c|
      leaves_paths(c).each{|p|
        acc += [[tree[:name]] + p]
      }
      acc
    }
  else
    [[tree[:name]]]
  end
end

(Не уверен, что выше точно соответствует вашей структуре jsTree, но принцип тот же.)

Вот пример ввода и вывода:

tree = {name: 'foo', children: [
      {name: 'bar'},
      {name: 'baz', children: [
        {name: 'boo'}, 
        {name: 'zoo', children: [
          {name: 'goo'}
        ]}
      ]}
    ]}

p leaves_paths tree
#=> [["foo", "bar"], ["foo", "baz", "boo"], ["foo", "baz", "zoo", "goo"]]

Затем для каждого пути вызовите FileUtils#mkdir_p:

paths = leaves_paths tree
paths.each do |path|
  FileUtils.mkdir_p(File.join(*path))
end

И с тобой должно быть все в порядке.

Редактировать: Более простая версия:

Вам не нужно создавать список листьев, достаточно пройти по всему дереву и создать каталог для каждого узла:

# executes block on each tree node, recursively, passing the path to the block as argument
def traverse_with_path tree, path = [], &block
  path += [tree[:name]]
  yield path
  tree[:children].each{|c| traverse_with_path c, path, &block} if tree[:children]
end

traverse_with_path tree do |path|
  FileUtils.mkdir(File.join(*path))
end

Edit2:

Ой, прости, я неправильно понял. Итак, вот способ создать хэш на основе дерева каталогов на диске:

Dir.glob('**/*'). # get all files below current dir
  select{|f|
    File.directory?(f) # only directories we need
  }.map{|path|
    path.split '/' # split to parts
  }.inject({}){|acc, path| # start with empty hash
    path.inject(acc) do |acc2,dir| # for each path part, create a child of current node
      acc2[dir] ||= {} # and pass it as new current node
    end
    acc
  }

Итак, для следующей структуры:

#$ mkdir -p foo/bar
#$ mkdir -p baz/boo/bee
#$ mkdir -p baz/goo

Код выше возвращает этот хеш:

{
  "baz"=>{
    "boo"=>{
      "bee"=>{}},
    "goo"=>{}},
  "foo"=>{
    "bar"=>{}}}

Надеюсь, вам удастся удовлетворить это вашим потребностям.

1 голос
/ 30 июня 2015

Принятый ответ не работал по состоянию на июнь 2015 г. Я изменил ключ :data на 'text'. Я также обобщил код для исключения каталогов и файлов.

def directory_hash(path, name=nil, exclude = [])                                
  exclude.concat(['..', '.', '.git', '__MACOSX', '.DS_Store'])                  
  data = {'text' => (name || path)}                                             
  data[:children] = children = []                                               
  Dir.foreach(path) do |entry|                                                  
    next if exclude.include?(entry)                                             
    full_path = File.join(path, entry)                                          
    if File.directory?(full_path)                                               
      children << directory_hash(full_path, entry)                              
    else                                                                        
      children << {'icon' => 'jstree-file', 'text' => entry}                    
    end                                                                         
  end                                                                           
  return data                                                                   
end  
1 голос
/ 02 апреля 2011

Модуль поиска Ruby (require 'find') минималистичен, но хорошо обрабатывает рекурсию каталога: http://www.ruby -doc.org / stdlib / libdoc / find / rdoc / classes / Find.html

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...