4

如何使用尽可能少的 API 调用获取给定文件夹子树中每个文件和文件夹的基本信息(至少 ID、标题、mime 类型)?IE。不调用 api 下载每个子文件夹的详细信息?

我找到了一种解决方法来读取具有一些非分层特征(例如所有者)的所有文件并在客户端脚本中构建树结构。不幸的是,我的文件都来自一个所有者(应用程序),所以我不能这样做。


好的,这里是递归多 API 调用方式的示例代码,这对于某些用例来说已经足够了。但我想找到更好的概念(不是讨论这个实现,而是另一种方式,如何不为每个文件夹调用 API):

class Foo {

const FOLDER_MIME_TYPE = 'application/vnd.google-apps.folder';


public function getSubtreeForFolder($parentId, $sort=true)
{
    $service = $this->createCrmGService();

    // A. folder info
    $file = $service->files->get($parentId);
    $ret = array(
        'id' => $parentId,
        'name' => $file->getTitle(),
        'description' => $file->getDescription(),
        'mimetype' => $file->getMimeType(),
        'is_folder' => true,
        'children' => array(),
        'node' => $file,
    );

    if ($ret['mimetype'] != self::FOLDER_MIME_TYPE) {
        throw new Exception(_t("{$ret['name']} is not a folder."));
    }

    $items = $this->findAllFiles($queryString='trashed = false', $parentId, $fieldsFilter='items(alternateLink,description,fileSize,id,mimeType,title)', $service); 

    foreach ($items as $child)
    {
        if ($this->isFolder($child)) 
        {
            $ret['children'][] = $this->getSubtreeForFolder($child->id, $sort);
        }

        else
        {
            // B. file info
            $a['id'] = $child->id;
            $a['name'] = $child->title;
            $a['description'] = $child->description;
            $a['is_folder'] = false;
            $a['url'] = $file->getDownloadUrl();
            $a['url_detail'] = $child->getAlternateLink();
            $a['versionLabel'] = false; //FIXME
            $a['node'] = $child;

            if (!$a['versionLabel']) {
                $a['versionLabel'] = '1.0'; //old files compatibility hack
            }
            $ret['children'][] = $a;
        }
    }

    if ($sort && isset($ret['children'])) 
    {
        if ($sort === true) {
            $sort = create_function('$a, $b', 'if ($a[\'name\'] == $b[\'name\']) return 0; return strcasecmp($a[\'name\'], $b[\'name\']);');
        }
        usort($ret['children'], $sort);
    }

    return $ret;
}


public function findAllFiles($queryString, $parentId=false, $fieldsFilter='items(id,title)', $service = false) 
{
    if (!$service)  $service = $this->createCrmGService();

    $result = array();
    $pageToken = NULL;

    if ($parentId) {
        $queryString .= ($queryString ? ' AND ' : '') . "'{$parentId}' in parents";
    }

    do {
        try {

            $parameters = array('q' => $queryString);

            if ($fieldsFilter) $parameters['fields'] = $fieldsFilter;

            if ($pageToken) {
                $parameters['pageToken'] = $pageToken;
            }

            $files = $service->files->listFiles($parameters);

            $result = array_merge($result, $files->getItems());

            $pageToken = $files->getNextPageToken();

        } catch (Exception $e) {
            print "An error occurred: " . $e->getMessage();
            $pageToken = NULL;
        }
    } while ($pageToken);

    return $result;
}


    /**
     * @param Google_DriveFile $file
     * @return boolean, jestli je $file slozka.
     */
    protected function isFolder($file)
    {
        return $file->getMimeType() == self::FOLDER_MIME_TYPE;
    }

}
4

1 回答 1

3

首先,我建议您不要获取所有文件和文件夹。对于一些在云端硬盘上上传了很多文件的用户来说,这需要太多时间。此外,您的应用程序密钥中存在查询限制。事实上,许多具有自定义文件选择器的应用程序在每次用户请求子文件夹时都会进行查询。

其次,如果这是 webapp,最好使用Google Picker。Google Picker 从云端硬盘中选择文件的速度更快、效率更高。有许多选项和过滤器,您可以很好地控制文件。

第三,您不能完全以树状结构表示 Drive 的文件和文件夹。正如您在查询中看到的那样,每个文件都有parents,这意味着每个文件/文件夹可以有多个父级。您需要考虑一些解决方法,例如为每个文件只选择一个父文件。

如果您仍想获取所有文件/文件夹信息,就性能而言,最好的实现是递归调用Children.list()。仅供参考,文件 id 'root' 是您可以轻松开始的保留 id。一旦你获得了孩子的 id,你就可以使用 multipart对Files.get()进行批量查询。这是遍历 Google Drive afaik 文件系统的最快方式。

同样,除非您有充分的理由,否则请不要尝试一次遍历云端硬盘中的所有文件。有些用户的云端硬盘中有很多文件,无论您进行了多么出色的优化,您都会让他们永远等待。此外,您将轻松达到查询限制。

于 2013-06-21T02:25:58.833 回答