org.apache.camel.component.box.api.BoxFilesManager Maven / Gradle / Ivy
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.camel.component.box.api;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URL;
import java.text.DateFormat;
import java.util.Collection;
import java.util.Date;
import java.util.List;
import com.box.sdk.BoxAPIConnection;
import com.box.sdk.BoxAPIException;
import com.box.sdk.BoxAPIResponseException;
import com.box.sdk.BoxFile;
import com.box.sdk.BoxFileVersion;
import com.box.sdk.BoxFolder;
import com.box.sdk.BoxItem;
import com.box.sdk.BoxSharedLink;
import com.box.sdk.FileUploadParams;
import com.box.sdk.Metadata;
import com.box.sdk.ProgressListener;
import com.box.sdk.sharedlink.BoxSharedLinkRequest;
import org.apache.camel.RuntimeCamelException;
import org.apache.camel.util.StopWatch;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Provides operations to manage Box files.
*/
public class BoxFilesManager {
private static final Logger LOG = LoggerFactory.getLogger(BoxFilesManager.class);
/**
* Box connection to authenticated user account.
*/
private BoxAPIConnection boxConnection;
/**
* Create files manager to manage the files of Box connection's authenticated user.
*
* @param boxConnection - Box connection to authenticated user account.
*/
public BoxFilesManager(BoxAPIConnection boxConnection) {
this.boxConnection = boxConnection;
}
/**
* Get file information.
*
* @param fileId - the id of file.
* @param fields - the information fields to retrieve; if null
all information fields are retrieved.
* @return The file information.
*/
public BoxFile.Info getFileInfo(String fileId, String... fields) {
try {
LOG.debug("Getting info for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
if (fields == null || fields.length == 0) {
return file.getInfo();
} else {
return file.getInfo(fields);
}
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Update file information.
*
* @param fileId - the id of file to update.
* @param info - the updated information
* @return The updated file.
*/
public BoxFile updateFileInfo(String fileId, BoxFile.Info info) {
try {
LOG.debug("Updating info for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(info, BoxHelper.INFO);
BoxFile file = new BoxFile(boxConnection, fileId);
file.updateInfo(info);
return file;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(
BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Upload a new file to parent folder.
*
* @param parentFolderId - the id of parent folder.
* @param content - a stream containing contents of the file to upload.
* @param fileName the name to give the uploaded file.
* @param created - the content created date that will be given to the uploaded file.
* @param modified - the content modified date that will be given to the uploaded file.
* @param size - the size of the file's content used for monitoring the upload's progress.
* @param check - if the file name is already used, call the uploadNewVersion instead.
* @param listener - a listener for monitoring the upload's progress.
* @return The uploaded file.
*/
public BoxFile uploadFile(
String parentFolderId, InputStream content, String fileName, Date created, Date modified,
Long size, Boolean check, ProgressListener listener) {
try {
LOG.debug("Uploading file with name '{}}' to parent_folder(id={}})", fileName, parentFolderId);
BoxHelper.notNull(parentFolderId, BoxHelper.PARENT_FOLDER_ID);
BoxHelper.notNull(content, BoxHelper.CONTENT);
BoxHelper.notNull(fileName, BoxHelper.FILE_NAME);
BoxFile boxFile = null;
boolean uploadNewFile = true;
if (check != null && check) {
BoxFolder folder = null;
try {
folder = new BoxFolder(boxConnection, parentFolderId);
// check if the file can be uploaded
// otherwise upload a new revision of the existing file
folder.canUpload(fileName, 0);
} catch (BoxAPIResponseException boxException) {
// https://developer.box.com/en/reference/options-files-content/
// error 409, filename exists, get the file info to upload a new revision
if (409 == boxException.getResponseCode()) {
// the box search api relies on a folder index to search for items, however our tests
// noticed the folder (box service) can take 11 minutes to reindex
// for a recently uploaded file, the folder search will return empty as the file was not indexed yet.
// see https://community.box.com/t5/Platform-and-Development-Forum/Box-Search-Delay/m-p/40072
// this check operation uses the folder list item as fallback mechanism to check if the file really exists
// it is slower than the search api, but reliable
// for faster results, we recommend the folder to contain no more than 500 items
// otherwise it can take more time to iterate over all items to check if the filename exists
// display a WARN if the delay is higher than 5s
final StopWatch watch = new StopWatch();
int delayLimit = 5;
boolean exists = false;
BoxItem.Info existingFile = null;
if (folder != null) {
// returns only the name and type fields of each folder item
for (BoxItem.Info itemInfo : folder.getChildren("name", BoxFolder.SortDirection.ASC, "name",
"type")) {
// check if the filename exists
exists = "file".equals(itemInfo.getType()) && fileName.equals(itemInfo.getName());
if (exists) {
existingFile = itemInfo;
break;
}
}
}
long elapsed = watch.taken();
if (elapsed > delayLimit) {
LOG.warn(
"The upload operation, checks if the file exists by using the Box list folder, however it took {}"
+ " seconds to verify, try to reduce the size of the folder items for faster results.",
elapsed);
}
if (exists) {
boxFile = uploadNewFileVersion(existingFile.getID(), content, modified, size, listener);
uploadNewFile = false;
}
} else {
throw boxException;
}
}
}
if (uploadNewFile) {
BoxFolder parentFolder = new BoxFolder(boxConnection, parentFolderId);
FileUploadParams uploadParams = new FileUploadParams();
uploadParams.setName(fileName);
uploadParams.setContent(content);
if (created != null) {
uploadParams.setCreated(created);
}
if (modified != null) {
uploadParams.setModified(modified);
}
if (size != null) {
uploadParams.setSize(size);
}
if (listener != null) {
uploadParams.setProgressListener(listener);
}
boxFile = parentFolder.uploadFile(uploadParams).getResource();
}
return boxFile;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(
BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Upload a new version of file.
*
* @param fileId - the id of file.
* @param fileContent - a stream containing contents of the file to upload.
* @param modified - the content modified date that will be given to the uploaded file.
* @param fileSize - the size of the file's content used for monitoring the upload's progress.
* @param listener - a listener for monitoring the upload's progress.
* @return The uploaded file.
*/
public BoxFile uploadNewFileVersion(
String fileId, InputStream fileContent, Date modified, Long fileSize,
ProgressListener listener) {
try {
LOG.debug("Uploading new version of file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(fileContent, BoxHelper.FILE_CONTENT);
BoxFile file = new BoxFile(boxConnection, fileId);
if (modified != null) {
if (fileSize != null) {
file.uploadNewVersion(fileContent, modified, fileSize, listener);
} else {
file.uploadNewVersion(fileContent, modified, 0, listener);
}
} else {
file.uploadNewVersion(fileContent);
}
return file;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Get any previous versions of file.
*
* @param fileId - the id of file.
* @return The list of previous file versions.
*/
public Collection getFileVersions(String fileId) {
try {
LOG.debug("Getting versions of file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
return file.getVersions();
} catch (BoxAPIException e) {
throw new RuntimeCamelException(
BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Download a file.
*
* @param fileId - the id of file.
* @param output - the stream to which the file contents will be written.
* @param rangeStart - the byte offset in file at which to start the download; if null
the entire
* contents of file will be downloaded.
* @param rangeEnd - the byte offset in file at which to stop the download; if null
the entire
* contents of file will be downloaded.
* @param listener - a listener for monitoring the download's progress; if null
the download's
* progress will not be monitored.
* @return The stream containing the contents of the downloaded file.
*/
public OutputStream downloadFile(
String fileId, OutputStream output, Long rangeStart, Long rangeEnd,
ProgressListener listener) {
try {
LOG.debug("Downloading file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(output, "output");
BoxFile file = new BoxFile(boxConnection, fileId);
if (listener != null) {
if (rangeStart != null && rangeEnd != null) {
file.downloadRange(output, rangeStart, rangeEnd, listener);
} else {
file.download(output, listener);
}
} else {
if (rangeStart != null && rangeEnd != null) {
file.downloadRange(output, rangeStart, rangeEnd);
} else {
file.download(output);
}
}
return output;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Download a previous version of file.
*
* @param fileId - the id of file.
* @param version - the version of file to download; initial version of file has value of 0
, second
* version of file is 1
and so on.
* @param output - the stream to which the version contents will be written.
* @param listener - a listener for monitoring the download's progress; if null
the download's
* progress will not be monitored.
* @return The stream containing the contents of the downloaded file version.
*/
public OutputStream downloadPreviousFileVersion(
String fileId, Integer version, OutputStream output,
ProgressListener listener) {
try {
LOG.debug("Downloading file(id={}, version={})", fileId, version);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(version, BoxHelper.VERSION);
BoxHelper.notNull(output, "output");
BoxFile file = new BoxFile(boxConnection, fileId);
List fileVersions = (List) file.getVersions();
BoxFileVersion fileVersion = fileVersions.get(version);
if (listener != null) {
fileVersion.download(output, listener);
} else {
fileVersion.download(output);
}
return output;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Promote a previous version of file.
*
* @param fileId - the id of file.
* @param version - the version of file to promote; initial version of file has value of 0
, second
* version of file is 1
and so on.
* @return The promoted version of file.
*/
public BoxFileVersion promoteFileVersion(String fileId, Integer version) {
try {
LOG.debug("Promoting file(id={}, version={})", fileId, version);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(version, BoxHelper.VERSION);
BoxFile file = new BoxFile(boxConnection, fileId);
List fileVersions = (List) file.getVersions();
BoxFileVersion fileVersion = fileVersions.get(version);
fileVersion.promote();
return fileVersion;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Copy file to destination folder while optionally giving it a new name.
*
* @param fileId - the id of file to copy.
* @param destinationFolderId - the id of the destination folder.
* @param newName - the new name for copied file; if newName
is null
, the
* copied file has same name as the original.
* @return The copied file.
*/
public BoxFile copyFile(String fileId, String destinationFolderId, String newName) {
try {
LOG.debug("Copying file(id={}) to destination_folder(id={}) {}",
fileId, destinationFolderId,
newName == null ? "" : " with new name '" + newName + "'");
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(destinationFolderId, BoxHelper.VERSION);
BoxFile fileToCopy = new BoxFile(boxConnection, fileId);
BoxFolder destinationFolder = new BoxFolder(boxConnection, destinationFolderId);
if (newName == null) {
return fileToCopy.copy(destinationFolder).getResource();
} else {
return fileToCopy.copy(destinationFolder, newName).getResource();
}
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Move file to destination folder while optionally giving it a new name.
*
* @param fileId - the id of file to move.
* @param destinationFolderId - the id of the destination folder.
* @param newName - the new name of moved file; if newName
is null
, the moved
* file has same name as the original.
* @return The moved file.
*/
public BoxFile moveFile(String fileId, String destinationFolderId, String newName) {
try {
LOG.debug("Moving file(id={}) to destination_folder(id={}) {}",
fileId, destinationFolderId,
newName == null ? "" : " with new name '" + newName + "'");
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(destinationFolderId, BoxHelper.VERSION);
BoxFile fileToMove = new BoxFile(boxConnection, fileId);
BoxFolder destinationFolder = new BoxFolder(boxConnection, destinationFolderId);
if (newName == null) {
return (BoxFile) fileToMove.move(destinationFolder).getResource();
} else {
return (BoxFile) fileToMove.move(destinationFolder, newName).getResource();
}
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Rename file giving it the name newName
*
* @param fileId - the id of file to rename.
* @param newFileName - the new name of file.
* @return The renamed file.
*/
public BoxFile renameFile(String fileId, String newFileName) {
try {
LOG.debug("Renaming file(id={}) to '{}'", fileId, newFileName);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(newFileName, BoxHelper.VERSION);
BoxFile fileToRename = new BoxFile(boxConnection, fileId);
fileToRename.rename(newFileName);
return fileToRename;
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Delete the file.
*
* @param fileId - the id of file to delete.
*/
public void deleteFile(String fileId) {
try {
LOG.debug("Deleting file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
file.delete();
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Delete a file version.
*
* @param fileId - the id of file with version to delete.
* @param version - the version of file to delete; initial version of file has value of 0
, second
* version of file is 1
and so on.
*/
public void deleteFileVersion(String fileId, Integer version) {
try {
LOG.debug("Deleting file(id={}, version={})", fileId, version);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(version, BoxHelper.VERSION);
BoxFile file = new BoxFile(boxConnection, fileId);
List versions = (List) file.getVersions();
BoxFileVersion fileVersion = versions.get(version);
fileVersion.delete();
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Create a shared link to file.
*
* @param fileId - the id of the file to create shared link on.
* @param access - the access level of the shared link.
* @param unshareDate - the date and time at which time the created shared link will expire; if
* unsharedDate
is null
then a non-expiring link is created.
* @param permissions - the permissions of the created link; if permissions
is null
then
* the created shared link is created with default permissions.
* @return The created shared link.
*/
public BoxSharedLink createFileSharedLink(
String fileId, BoxSharedLink.Access access, Date unshareDate,
BoxSharedLink.Permissions permissions) {
try {
LOG.debug("Creating shared link for file(id={}) with access={} {}",
fileId, access, unshareDate == null
? ""
: " unsharedDate=" + DateFormat.getDateTimeInstance().format(unshareDate)
+ " permissions=" + permissions);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(access, "access");
BoxFile file = new BoxFile(boxConnection, fileId);
BoxSharedLinkRequest request = new BoxSharedLinkRequest();
request.access(access).unsharedDate(unshareDate)
.permissions(permissions.getCanDownload(), permissions.getCanPreview(), permissions.getCanEdit());
return file.createSharedLink(request);
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Get an expiring URL for downloading a file directly from Box. This can be user, for example, for sending as a
* redirect to a browser to cause the browser to download the file directly from Box.
*
* @param fileId - the id of file.
* @return The temporary download URL
*/
public URL getDownloadURL(String fileId) {
try {
LOG.debug("Getting download URL for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
return file.getDownloadURL();
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Get an expiring URL for creating an embedded preview session. The URL will expire after 60 seconds and the
* preview session will expire after 60 minutes.
*
* @param fileId - the id of the file to get preview link on.
* @return The preview link.
*/
public URL getFilePreviewLink(String fileId) {
try {
LOG.debug("Getting preview link for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
return file.getPreviewLink();
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Create metadata for file in either the global properties template or the specified template type.
*
* @param fileId - the id of the file to create metadata for.
* @param metadata - the new metadata values.
* @param typeName - the metadata template type name; if null
the global properties template type is
* used.
* @return The metadata returned from the server.
*/
public Metadata createFileMetadata(String fileId, Metadata metadata, String typeName) {
try {
LOG.debug("Creating metadata for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(metadata, BoxHelper.METADATA);
BoxFile file = new BoxFile(boxConnection, fileId);
if (typeName != null) {
return file.createMetadata(typeName, metadata);
} else {
return file.createMetadata(metadata);
}
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Gets the file properties metadata.
*
* @param fileId - the id of the file to retrieve metadata for.
* @param typeName - the metadata template type name; if null
the global properties template type is
* used.
* @return The metadata returned from the server.
*/
public Metadata getFileMetadata(String fileId, String typeName) {
try {
LOG.debug("Get metadata for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
if (typeName != null) {
return file.getMetadata(typeName);
} else {
return file.getMetadata();
}
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Update the file properties metadata.
*
* @param fileId - the id of file to delete.
* @param metadata - the new metadata values.
* @return The metadata returned from the server.
*/
public Metadata updateFileMetadata(String fileId, Metadata metadata) {
try {
LOG.debug("Updating metadata for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxHelper.notNull(metadata, BoxHelper.METADATA);
BoxFile file = new BoxFile(boxConnection, fileId);
return file.updateMetadata(metadata);
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Delete the file properties metadata.
*
* @param fileId - the id of file to delete.
*/
public void deleteFileMetadata(String fileId) {
try {
LOG.debug("Deleting metadata for file(id={})", fileId);
BoxHelper.notNull(fileId, BoxHelper.FILE_ID);
BoxFile file = new BoxFile(boxConnection, fileId);
file.deleteMetadata();
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
/**
* Does a pre-verification before upload, to check if the filename already exists or if there is permission to
* upload. It will throw a BoxAPIResponseException if there is any problem in uploading the given file.
*
* @param parentFolderId - the id of parent folder.
* @param fileName the name to give the uploaded file.
* @param size - the size of the file's content used for monitoring the upload's progress.
*
*/
public void checkUpload(String fileName, String parentFolderId, Long size) {
try {
LOG.debug("Preflight check file with name '{}' to parent_folder(id={})", fileName, parentFolderId);
BoxHelper.notNull(parentFolderId, BoxHelper.PARENT_FOLDER_ID);
BoxHelper.notNull(fileName, BoxHelper.FILE_NAME);
BoxHelper.notNull(size, BoxHelper.SIZE);
BoxFolder parentFolder = new BoxFolder(boxConnection, parentFolderId);
parentFolder.canUpload(fileName, size);
} catch (BoxAPIException e) {
throw new RuntimeCamelException(BoxHelper.buildBoxApiErrorMessage(e), e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy