如何使用尽可能少的API调用为给定文件夹的子树中的每个文件和文件夹获取基本信息(id,title,mime-type)?即。不要调用api来下载每个子文件夹的详细信息吗?
我发现解决方法是读取具有一些非分层特征(例如所有者)的所有文件,并在客户端脚本中构建树结构。不幸的是,我的文件全部来自一个所有者(应用程序),所以我不能这样做。
好的,这是recursion-multiple-api-calls方式的示例代码,对于某些用例来说已经足够了。但我想找到更好的概念(不是讨论这个实现,而是另一种方式,如何不为每个文件夹调用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;
}
}
答案 0 :(得分:0)
首先,我建议你不要获取所有文件和文件夹。对于在其云端硬盘上上传了许多文件的用户来说,需要花费太多时间。此外,您的应用程序密钥中存在查询限制。实际上,每次用户请求子文件夹时,许多具有自定义文件选择器的应用程序都会进行查询。
其次,如果这是webapp,最好使用Google Picker。 Google Picker从Drive中选择文件的速度更快,效率更高。有很多选项和过滤器,你可以控制文件。
第三,您无法在树形结构中完全表示Drive的文件和文件夹。正如您在查询中看到的那样,每个文件都有父级,这意味着每个文件/文件夹可以有多个父级。您需要考虑一些解决方法,例如只为每个文件选择一个父项。
如果您仍想获取所有文件/文件夹信息,就性能而言,最佳实现将递归调用Children.list()。仅供参考,文件ID“root”是您可以轻松入门的保留ID。一旦你得到了孩子的id,你可以用multipart批量查询Files.get()。这是遍历Google云端硬盘文件系统的最快方式。
同样,除非您有充分的理由,否则请不要尝试一次遍历云端硬盘中的所有文件。有些用户在他们的云端硬盘中有很多文件,无论你做了多么优化,你都会让他们永远等待。此外,您将轻松达到查询限制。