Простая работа с Google Drive API

Сегодня рассмотрим простой пример работы с Google Drive API. Почему именно данное API, потому что оно популярно. Как-то не так давно, мне пришлось разрабатывать кое-что подобное для нескольких студентов. Там использовались не только сервисы от Google, но и от других не менее известных компаний. Для упрощения статьи я решил опустить некоторые моменты по настройке, о которых можно прочитать на других сайтах. Но хватит вступления, приступим к рассмотрению моего примера.

Подготовка к работе

Первым делом Вам необходимо иметь учетную запись в Google. Затем надо будет подключить нужное нам API и получить секретный ключ и идентификатор клиента. Подключить Google Drive API можно в Google API Console. Не забывайте, что после включения API нужно будет создать новый проект. Здесь я лишь приведу примеры изображений, как настроено у меня.

В своей работе я буду использовать использовать библиотеку Google API Client Library for PHP. Если перейти по данной ссылке, то Вы найдете описание как установить зависимости через composer.

Для удобства всю логику приложения оставил в файле index.php.

Листинг файла index.php

<?php
/**
 * @Doc https://developers.google.com/drive/api/v3/about-sdk
 */
// include your composer dependencies
require_once 'vendor/autoload.php';
include 'BuildTree.php';

session_start();
ob_start();

$redirect_uri = "http://{$_SERVER ['HTTP_HOST']}/";

$client = new Google_Client();
$client->setClientId('797383941645-12og1l10jlhbmkpp4e71rpe4akqcu562.apps.googleusercontent.com');
$client->setClientSecret('qPzJuo0wRApxXnCO9PryjKgE');
$client->setRedirectUri($redirect_uri);
$client->addScope(Google_Service_Drive::DRIVE);

// Запрос на подтверждение работы с Google-диском
if (isset($_REQUEST['code'])) {
  $token = $client->authenticate($_REQUEST['code']);
  $_SESSION['accessToken'] = $token;
  header('Location:' . filter_var($redirect_uri, FILTER_SANITIZE_URL));
} elseif (!isset($_SESSION['accessToken'])) {
  header('Location:' . filter_var($client->createAuthUrl(), FILTER_SANITIZE_URL));
}

// Присваиваем защитный токен для работы с Google-диском
$client->setAccessToken($_SESSION['accessToken']);

$driveService = new Google_Service_Drive($client);

$listFiles = $driveService->files->listFiles([
  'fields' => 'nextPageToken, files(id, name, parents, fileExtension, mimeType, size, iconLink, thumbnailLink, webContentLink, webViewLink, createdTime)'
]);

// Пересобираем массив для добавления ключа parentId
$files = [];

foreach ($listFiles['modelData']['files'] as $k => $item) {
  $files[$k] = $item;
  $files[$k]['parentId'] = $item['parents'][0];
  unset($files[$k]['parents']);
}

// Строим дерево элементов
$buildTree = new buildTree();
$tree = $buildTree->makeTree($files, '0AHCqnAQsA3IDUk9PVA');

echo '<style type="text/css">
  ul > li {
    list-style: none;
  }
  ul > li > img {
    margin-right: 3px;
  }
  
  .icon-new {
    display: inline-block;
    background: url("");
    width: 16px;
    height: 16px;
  }
  .icon-upload{
    display: inline-block;
    background: url("");
    width: 16px;
    height: 16px;
  }
  .icon-delete {
    display: inline-block;
    background: url("");
    width: 16px;
    height: 16px;
  }
</style>';

// Вывод дерева
printTree($tree);

if (isset($_GET['action'])) {
    switch ($_GET['action']) {
        // Создание новой папки
        case 'newdir':

            if (isset($_GET['pid'])) {
                include 'form-create-dir.php';

                if ($_POST) {
                    if (!empty($_POST['newFolder'])) {
                        $fileMetadata = new Google_Service_Drive_DriveFile([
                            'name' => htmlspecialchars($_POST['newFolder']),
                            'parents' => [htmlspecialchars($_GET['pid'])],
                            'mimeType' => 'application/vnd.google-apps.folder'
                        ]);
                        $driveService->files->create($fileMetadata, ['fields' => 'id']);
                        header('Location:' . filter_var($redirect_uri, FILTER_SANITIZE_URL));

                    } else {
                        die('Заполните название папки');
                    }
                }
            }

            break;

        // Загрузка файлов
        case 'upload':

            if (isset($_GET['fid'])) {
                include 'form-upload-file.php';

                if (isset($_FILES['Files']) && is_uploaded_file($_FILES['Files']['tmp_name'][0]) && ($_FILES['Files']['error'][0] == 0)) {
                    $files = normalizeFilesArray($_FILES);

                    $fileMetadata = new Google_Service_Drive_DriveFile([
                        'parents' => [htmlspecialchars($_GET['fid'])]
                    ]);

                    foreach ($files as $file) {
                        $filePath = $file['tmp_name'];
                        $mimeType = $file['type'];

                        $fileMetadata->setName($file['name']);
                        $fileMetadata->setDescription('Это документ ' . $mimeType);
                        $fileMetadata->setMimeType($mimeType);

                        $driveService->files->create($fileMetadata, [
                            'data' => file_get_contents($filePath),
                            'mimeType' => $mimeType,
                            'uploadType' => 'multipart',
                            'fields' => 'id'
                        ]);
                    }

                    header('Location:' . filter_var($redirect_uri, FILTER_SANITIZE_URL));
                }
            }

            break;

        case 'delete':

            if (isset($_GET['fid'])) {
                $driveService->files->delete(htmlspecialchars($_GET['fid']));
                header('Location:' . filter_var($redirect_uri, FILTER_SANITIZE_URL));
            }

            break;
    }
}
ob_end_flush();

/** Функции */

function printTree($tree)
{
  if (!is_null($tree) && count($tree) > 0) {
    echo '<ul>';
    foreach ($tree as $kn => $node) {
      echo '<li>';
      echo '<img src="'.$node['iconLink'].'" alt="">';
      echo '<a href="'.$node['webViewLink'].'">';

      if ($node['mimeType'] == 'application/vnd.google-apps.folder'){
        echo '<b>' . $node['name'] . '</b>';
      } else {
        echo $node['name'];
      }

      echo '</a> ';
      echo date('d.m.Y H:i:s', strtotime($node['createdTime']));

      if ($node['mimeType'] == 'application/vnd.google-apps.folder'){
        echo ' <a href="?action=newdir&pid='.urlencode($node['id']).'" title="Новая папка"><i class="icon-new"></i></a>';
      }
      echo ' <a href="?action=upload&fid=' . urlencode($node['id']) . '" title="Загрузить файл(ы)"><i class="icon-upload"></i></a>';
      echo ' <a href="?action=delete&fid=' . urlencode($node['id']) . '" title="Удалить файл(ы)" onclick="return confirm(\'Вы действительно хотите удалить данный файл?\');"><i class="icon-delete"></i></a>';

      printTree($node['children']);
      echo '</li>';
    }
    echo '</ul>';
  }
}

/**
 * Приводим к нормальному виду глобальный массив $_FILES
 *
 * @param array $files
 * @return array
 */
function normalizeFilesArray($files = [])
{
  $result = [];

  foreach ($files as $file) {
    if (!is_array($file['name'])) {
      $result[] = $file;
      continue;
    }

    foreach ($file['name'] as $idx => $name) {
      $result[$idx] = [
        'name' 		=> $name,
        'type' 		=> $file['type'][$idx],
        'tmp_name' 	=> $file['tmp_name'][$idx],
        'error' 	=> $file['error'][$idx],
        'size' 		=> $file['size'][$idx]
      ];
    }
  }
  return $result;
}

/**
 * Вспомогательная функция для отладки
 *
 * @param $data
 */
function varDumper($data){
  echo '<pre>';
  print_r($data);
  echo '</pre>';
}

Внимание!

Необходимо настроить в настройка переадресацию, например, у меня http://localhost.

Для методов $client->setClientId() и $client->setClientSecret() нужно установить значения, которые Вы возьмете из настроек своего проекта в Google API Console.

Дополнительно вынес рекурсивную функцию в отдельный класс, который расположен в файле BuildTree.php. По факту функцию можно было бы оставить в файле index.php, но планировалось немного большее, однако как всегда не хватило времени на реализацию.

Листинг файла BuildTree.php

<?php

class BuildTree
{
  /**
   * Рекурсивное создание дерева элементов
   *
   * @param array  $elements Массив элементов
   * @param string $parentId Идентификатор корневого элемента
   * @return array
   */
  public function makeTree(Array &$elements, $parentId = '') : array
  {
    $branch = [];
    foreach ($elements as &$element) {
      if ($element['parentId'] == $parentId) {
        $element['children'] = $this->makeTree($elements, $element['id']);
        $branch[$element['id']] = $element;
        unset($element);
      }
    }
    return $branch;
  }
}

Листинг файла для отображения создания новой папки.

<h3>Новая папка</h3>
<form method="post">
  <label>Название папки<br>
    <input type="text" name="newFolder" value="">
  </label>
  <input type="submit" value="Создать">
</form>

Листинг файла для отображения загрузки файлов.

<h3>Загрузить файлы</h3>
<form method="post" enctype="multipart/form-data">
  <label>Выберите файлы:<br>
    <input type="file" name="Files[]" multiple>
  </label><br>
  <p>
    <input type="submit" value="Загрузить файлы">
  </p>
</form>

Вот и все основные файлы. Если все сделали правильно, то должно работать. В конце статьи добавлю ссылку на архив (если кто-то не хочет заморачиваться с composer).

А вот и результаты мое работы. Как Вы заметили, в коде нет особых проверок, т.к. это всего лишь пример, а не реальный рабочий проект.

Если появились вопросы, пожалуйста, задавайте, а я постараюсь на них ответить. Спасибо и до новых встреч.

Полезные ссылки

  1. Getting Started
  2. Google APIs Client Library for PHP
  3. Google Drive REST API
  4. Пример работы с Google Drive API (исходники на github)
  5. Sample Google Drive API (исходники в архиве zip)

 

Если вы нашли ошибку, пожалуйста, выделите фрагмент текста и нажмите Ctrl+Enter.

Добавить комментарий

Сообщить об опечатке

Текст, который будет отправлен нашим редакторам: