google-drive-picker.ts 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. // Google Drive Picker API configuration
  2. const API_KEY = import.meta.env.VITE_GOOGLE_API_KEY;
  3. const CLIENT_ID = import.meta.env.VITE_GOOGLE_CLIENT_ID;
  4. const SCOPE = ['https://www.googleapis.com/auth/drive.readonly', 'https://www.googleapis.com/auth/drive.file'];
  5. // Validate required credentials
  6. const validateCredentials = () => {
  7. if (!API_KEY || !CLIENT_ID) {
  8. throw new Error('Google Drive API credentials not configured');
  9. }
  10. if (API_KEY === 'your-api-key' || CLIENT_ID === 'your-client-id') {
  11. throw new Error('Please configure valid Google Drive API credentials');
  12. }
  13. };
  14. let pickerApiLoaded = false;
  15. let oauthToken: string | null = null;
  16. let initialized = false;
  17. export const loadGoogleDriveApi = () => {
  18. return new Promise((resolve, reject) => {
  19. if (typeof gapi === 'undefined') {
  20. const script = document.createElement('script');
  21. script.src = 'https://apis.google.com/js/api.js';
  22. script.onload = () => {
  23. gapi.load('picker', () => {
  24. pickerApiLoaded = true;
  25. resolve(true);
  26. });
  27. };
  28. script.onerror = reject;
  29. document.body.appendChild(script);
  30. } else {
  31. gapi.load('picker', () => {
  32. pickerApiLoaded = true;
  33. resolve(true);
  34. });
  35. }
  36. });
  37. };
  38. export const loadGoogleAuthApi = () => {
  39. return new Promise((resolve, reject) => {
  40. if (typeof google === 'undefined') {
  41. const script = document.createElement('script');
  42. script.src = 'https://accounts.google.com/gsi/client';
  43. script.onload = resolve;
  44. script.onerror = reject;
  45. document.body.appendChild(script);
  46. } else {
  47. resolve(true);
  48. }
  49. });
  50. };
  51. export const getAuthToken = async () => {
  52. if (!oauthToken) {
  53. return new Promise((resolve, reject) => {
  54. const tokenClient = google.accounts.oauth2.initTokenClient({
  55. client_id: CLIENT_ID,
  56. scope: SCOPE.join(' '),
  57. callback: (response: any) => {
  58. if (response.access_token) {
  59. oauthToken = response.access_token;
  60. resolve(oauthToken);
  61. } else {
  62. reject(new Error('Failed to get access token'));
  63. }
  64. },
  65. error_callback: (error: any) => {
  66. reject(new Error(error.message || 'OAuth error occurred'));
  67. }
  68. });
  69. tokenClient.requestAccessToken();
  70. });
  71. }
  72. return oauthToken;
  73. };
  74. const initialize = async () => {
  75. if (!initialized) {
  76. validateCredentials();
  77. await Promise.all([loadGoogleDriveApi(), loadGoogleAuthApi()]);
  78. initialized = true;
  79. }
  80. };
  81. export const createPicker = () => {
  82. return new Promise(async (resolve, reject) => {
  83. try {
  84. console.log('Initializing Google Drive Picker...');
  85. await initialize();
  86. console.log('Getting auth token...');
  87. const token = await getAuthToken();
  88. if (!token) {
  89. console.error('Failed to get OAuth token');
  90. throw new Error('Unable to get OAuth token');
  91. }
  92. console.log('Auth token obtained successfully');
  93. const picker = new google.picker.PickerBuilder()
  94. .enableFeature(google.picker.Feature.NAV_HIDDEN)
  95. .enableFeature(google.picker.Feature.MULTISELECT_ENABLED)
  96. .addView(new google.picker.DocsView()
  97. .setIncludeFolders(false)
  98. .setSelectFolderEnabled(false)
  99. .setMimeTypes('application/pdf,text/plain,application/vnd.openxmlformats-officedocument.wordprocessingml.document,application/vnd.google-apps.document,application/vnd.google-apps.spreadsheet'))
  100. .setOAuthToken(token)
  101. .setDeveloperKey(API_KEY)
  102. // Remove app ID setting as it's not needed and can cause 404 errors
  103. .setCallback(async (data: any) => {
  104. if (data[google.picker.Response.ACTION] === google.picker.Action.PICKED) {
  105. try {
  106. const doc = data[google.picker.Response.DOCUMENTS][0];
  107. const fileId = doc[google.picker.Document.ID];
  108. const fileName = doc[google.picker.Document.NAME];
  109. const fileUrl = doc[google.picker.Document.URL];
  110. if (!fileId || !fileName) {
  111. throw new Error('Required file details missing');
  112. }
  113. // Construct download URL based on MIME type
  114. const mimeType = doc[google.picker.Document.MIME_TYPE];
  115. let downloadUrl;
  116. let exportFormat;
  117. if (mimeType.includes('google-apps')) {
  118. // Handle Google Workspace files
  119. if (mimeType.includes('document')) {
  120. exportFormat = 'application/vnd.openxmlformats-officedocument.wordprocessingml.document';
  121. } else if (mimeType.includes('spreadsheet')) {
  122. exportFormat = 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet';
  123. } else if (mimeType.includes('presentation')) {
  124. exportFormat = 'application/vnd.openxmlformats-officedocument.presentationml.presentation';
  125. } else {
  126. exportFormat = 'application/pdf';
  127. }
  128. downloadUrl = `https://www.googleapis.com/drive/v3/files/${fileId}/export?mimeType=${encodeURIComponent(exportFormat)}`;
  129. } else {
  130. // Regular files use direct download URL
  131. downloadUrl = `https://www.googleapis.com/drive/v3/files/${fileId}?alt=media`;
  132. }
  133. // Create a Blob from the file download
  134. const response = await fetch(downloadUrl, {
  135. headers: {
  136. 'Authorization': `Bearer ${token}`,
  137. 'Accept': '*/*'
  138. }
  139. });
  140. if (!response.ok) {
  141. const errorText = await response.text();
  142. console.error('Download failed:', {
  143. status: response.status,
  144. statusText: response.statusText,
  145. error: errorText
  146. });
  147. throw new Error(`Failed to download file (${response.status}): ${errorText}`);
  148. }
  149. const blob = await response.blob();
  150. const result = {
  151. id: fileId,
  152. name: fileName,
  153. url: downloadUrl,
  154. blob: blob,
  155. headers: {
  156. 'Authorization': `Bearer ${token}`,
  157. 'Accept': '*/*'
  158. }
  159. };
  160. resolve(result);
  161. } catch (error) {
  162. reject(error);
  163. }
  164. } else if (data[google.picker.Response.ACTION] === google.picker.Action.CANCEL) {
  165. resolve(null);
  166. }
  167. })
  168. .build();
  169. picker.setVisible(true);
  170. } catch (error) {
  171. console.error('Google Drive Picker error:', error);
  172. reject(error);
  173. }
  174. });
  175. };