mirror of
https://github.com/open-webui/open-webui
synced 2025-04-28 10:11:34 +00:00
150 lines
5.7 KiB
TypeScript
150 lines
5.7 KiB
TypeScript
// Google Drive Picker API configuration
|
|
const API_KEY = import.meta.env.VITE_GOOGLE_API_KEY;
|
|
const CLIENT_ID = import.meta.env.VITE_GOOGLE_CLIENT_ID;
|
|
const SCOPE = ['https://www.googleapis.com/auth/drive.readonly', 'https://www.googleapis.com/auth/drive.file'];
|
|
|
|
// Validate required credentials
|
|
const validateCredentials = () => {
|
|
if (!API_KEY || !CLIENT_ID) {
|
|
throw new Error('Google Drive API credentials not configured');
|
|
}
|
|
if (API_KEY === 'your-api-key' || CLIENT_ID === 'your-client-id') {
|
|
throw new Error('Please configure valid Google Drive API credentials');
|
|
}
|
|
};
|
|
|
|
let pickerApiLoaded = false;
|
|
let oauthToken: string | null = null;
|
|
let initialized = false;
|
|
|
|
export const loadGoogleDriveApi = () => {
|
|
return new Promise((resolve, reject) => {
|
|
if (typeof gapi === 'undefined') {
|
|
const script = document.createElement('script');
|
|
script.src = 'https://apis.google.com/js/api.js';
|
|
script.onload = () => {
|
|
gapi.load('picker', () => {
|
|
pickerApiLoaded = true;
|
|
resolve(true);
|
|
});
|
|
};
|
|
script.onerror = reject;
|
|
document.body.appendChild(script);
|
|
} else {
|
|
gapi.load('picker', () => {
|
|
pickerApiLoaded = true;
|
|
resolve(true);
|
|
});
|
|
}
|
|
});
|
|
};
|
|
|
|
export const loadGoogleAuthApi = () => {
|
|
return new Promise((resolve, reject) => {
|
|
if (typeof google === 'undefined') {
|
|
const script = document.createElement('script');
|
|
script.src = 'https://accounts.google.com/gsi/client';
|
|
script.onload = resolve;
|
|
script.onerror = reject;
|
|
document.body.appendChild(script);
|
|
} else {
|
|
resolve(true);
|
|
}
|
|
});
|
|
};
|
|
|
|
export const getAuthToken = async () => {
|
|
if (!oauthToken) {
|
|
return new Promise((resolve, reject) => {
|
|
const tokenClient = google.accounts.oauth2.initTokenClient({
|
|
client_id: CLIENT_ID,
|
|
scope: SCOPE.join(' '),
|
|
callback: (response: any) => {
|
|
if (response.access_token) {
|
|
oauthToken = response.access_token;
|
|
resolve(oauthToken);
|
|
} else {
|
|
reject(new Error('Failed to get access token'));
|
|
}
|
|
},
|
|
error_callback: (error: any) => {
|
|
reject(new Error(error.message || 'OAuth error occurred'));
|
|
}
|
|
});
|
|
tokenClient.requestAccessToken();
|
|
});
|
|
}
|
|
return oauthToken;
|
|
};
|
|
|
|
const initialize = async () => {
|
|
if (!initialized) {
|
|
validateCredentials();
|
|
await Promise.all([loadGoogleDriveApi(), loadGoogleAuthApi()]);
|
|
initialized = true;
|
|
}
|
|
};
|
|
|
|
export const createPicker = () => {
|
|
return new Promise(async (resolve, reject) => {
|
|
try {
|
|
console.log('Initializing Google Drive Picker...');
|
|
await initialize();
|
|
console.log('Getting auth token...');
|
|
const token = await getAuthToken();
|
|
if (!token) {
|
|
console.error('Failed to get OAuth token');
|
|
throw new Error('Unable to get OAuth token');
|
|
}
|
|
console.log('Auth token obtained successfully');
|
|
|
|
const picker = new google.picker.PickerBuilder()
|
|
.enableFeature(google.picker.Feature.NAV_HIDDEN)
|
|
.enableFeature(google.picker.Feature.MULTISELECT_ENABLED)
|
|
.addView(new google.picker.DocsView()
|
|
.setIncludeFolders(true)
|
|
.setSelectFolderEnabled(true))
|
|
.setOAuthToken(token)
|
|
.setDeveloperKey(API_KEY)
|
|
// Remove app ID setting as it's not needed and can cause 404 errors
|
|
.setCallback((data: any) => {
|
|
console.log('Picker callback received:', data);
|
|
if (data[google.picker.Response.ACTION] === google.picker.Action.PICKED) {
|
|
console.log('File picked from Google Drive');
|
|
const doc = data[google.picker.Response.DOCUMENTS][0];
|
|
const fileId = doc[google.picker.Document.ID];
|
|
const fileName = doc[google.picker.Document.NAME];
|
|
const fileUrl = doc[google.picker.Document.URL];
|
|
|
|
console.log('Selected file details:', {
|
|
id: fileId,
|
|
name: fileName,
|
|
url: fileUrl
|
|
});
|
|
|
|
// Construct download URL without embedding token
|
|
const downloadUrl = `https://www.googleapis.com/drive/v3/files/${fileId}?alt=media`;
|
|
const result = {
|
|
id: fileId,
|
|
name: fileName,
|
|
url: downloadUrl,
|
|
headers: {
|
|
'Authorization': `Bearer ${oauthToken}`,
|
|
'Content-Type': 'application/json'
|
|
}
|
|
};
|
|
console.log('Resolving picker with:', result);
|
|
resolve(result);
|
|
} else if (data[google.picker.Response.ACTION] === google.picker.Action.CANCEL) {
|
|
resolve(null);
|
|
}
|
|
})
|
|
.build();
|
|
picker.setVisible(true);
|
|
} catch (error) {
|
|
console.error('Google Drive Picker error:', error);
|
|
reject(error);
|
|
}
|
|
});
|
|
};
|