Skip to content

Commit

Permalink
Merge pull request #138 from owncloud/1556_access_to_avatars
Browse files Browse the repository at this point in the history
Add GetRemoteUserAvatarOperation to get data and metadata of current user avatar.
  • Loading branch information
davivel authored Oct 18, 2016
2 parents dc14474 + 0b945f8 commit a13b5ec
Show file tree
Hide file tree
Showing 5 changed files with 298 additions and 8 deletions.
6 changes: 0 additions & 6 deletions .travis.yml
Original file line number Diff line number Diff line change
Expand Up @@ -35,12 +35,6 @@ env:
# The next declaration is the encrypted COVERITY_SCAN_TOKEN, created
# via the "travis encrypt" command using the project repo's public key
- secure: epTZ0zZGDbHL3o6vSC9uNkZsi5j5SA6O/tvQBH7QW/dluuzIJxIjfhNbZHDyBReYDleirLzUFQpdWAUdvulCMLs/qZdIzFGlYXZSpxEnvPYMGQcilwADdJcxLw8L+3+ET5hSexxhjrTGw427IljkqGUpqQTxaLwFdFu98lDWSbc=
# The next 3 declarations are the variables needed to access the test server,
# added via "travis encrypt", using the repo public key
- secure: gPCBnpGLA2sdSMtfhT+/InThmXNEU8XrrS54uuIP8iXBLvVe0yZrNl76GbMosV0ry3YtDngsmsbHwRjPPb0+3mTTdAqZ60HHzGaNPgEm6b5t0t4bpJ3LW9osLZsuf9jRsI2LD66zxblaMrK2+8hN/dUrj707ijsZHp3SPSQJ6g8=
- secure: AnxLVarfwM7IhJ7Sca35USyRlFHFvlcBhWTt2TVDcyQ+ldDyb+U6IWXFK0Yy82QP0ZH/RCLu7FnmHK/rKG0BHNRt1Ymco1VkTQql0MZcHXP+4IKgEvgJyUn1TqYj+hSVmM6lgTA+QUjZYGSfwU8mhUFiU7644ZTdTe6ALdqa+v8=
- secure: ezKyZbb3q1Phcv/vJntuJe0C2req+Hp4/C+tFZIWZ3o8wRO9jVI3bnED9TWQyQOOT0SoRYjJ5zqp0UcEOGCzPeWFO6bA7RWp+zA/R9sziLNcVWMVv3WXnuClQjPBHJeXRnP7YmNjxDmSfV97a14dk5d9LgJZYliTDepH4dLsxro=
- secure: HV8REF7bB1i6cBEe4YCZvjgTf6tGiptAPzp9qZXGUH9Lxh6yUjYMXXP8flDaIFcE2JuAakW6P4SPr7/SQ+X4f8/84cNhmU1I+6Y1nG37WvelRHQVslOUDbl1TNUnynYZ+ybkb3BtrQvggCGMdQKvGZXYiggFhN/TWuDo0c7Q2Ro=
matrix:
- ANDROID_TARGET=android-24 ANDROID_ABI=armeabi-v7a
addons:
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,188 @@
/* ownCloud Android Library is available under MIT license
*
* @author David A. Velasco
* Copyright (C) 2016 ownCloud GmbH.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
*/

package com.owncloud.android.lib.resources.users;

import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.util.ArrayList;

import org.apache.commons.httpclient.Header;
import org.apache.commons.httpclient.HttpStatus;
import org.apache.commons.httpclient.methods.GetMethod;

import com.owncloud.android.lib.common.OwnCloudClient;
import com.owncloud.android.lib.common.network.WebdavUtils;
import com.owncloud.android.lib.common.operations.RemoteOperation;
import com.owncloud.android.lib.common.operations.RemoteOperationResult;
import com.owncloud.android.lib.common.utils.Log_OC;


/**
* Gets avatar about the user logged in, if available
*/

public class GetRemoteUserAvatarOperation extends RemoteOperation {

private static final String TAG = GetRemoteUserAvatarOperation.class.getSimpleName();

private static final String NON_OFFICIAL_AVATAR_PATH = "/index.php/avatar/";
private static final String IF_NONE_MATCH_HEADER = "If-None-Match";

/** Desired size in pixels of the squared image */
private int mDimension;

/**
* Etag of current local copy of the avatar; if not null, remote avatar will be downloaded only
* if its Etag changed.
*/
private String mCurrentEtag;


public GetRemoteUserAvatarOperation(int dimension, String currentEtag) {
mDimension = dimension;
mCurrentEtag = currentEtag;
}

@Override
protected RemoteOperationResult run(OwnCloudClient client) {
RemoteOperationResult result = null;
GetMethod get = null;
InputStream inputStream = null;

try {
String uri =
client.getBaseUri() + NON_OFFICIAL_AVATAR_PATH +
client.getCredentials().getUsername() + "/" + mDimension;
;
Log_OC.d(TAG, "avatar URI: " + uri);
get = new GetMethod(uri);
if (mCurrentEtag != null && mCurrentEtag.length() > 0) {
get.addRequestHeader(IF_NONE_MATCH_HEADER, "\"" + mCurrentEtag + "\"");
}
//get.addRequestHeader(OCS_API_HEADER, OCS_API_HEADER_VALUE);
int status = client.executeMethod(get);
if (isSuccess(status)) {

// find out size of file to read
int totalToTransfer = 0;
Header contentLength = get.getResponseHeader("Content-Length");
if (contentLength != null && contentLength.getValue().length() > 0) {
totalToTransfer = Integer.parseInt(contentLength.getValue());
}

// find out MIME-type!
String mimeType;
Header contentType = get.getResponseHeader("Content-Type");
if (contentType == null || !contentType.getValue().startsWith("image")) {
Log_OC.e(
TAG, "Not an image, failing with no avatar"
);
result = new RemoteOperationResult(
RemoteOperationResult.ResultCode.FILE_NOT_FOUND
);
return result;
}
mimeType = contentType.getValue();

/// download will be performed to a buffer
inputStream = get.getResponseBodyAsStream();
BufferedInputStream bis = new BufferedInputStream(inputStream);
ByteArrayOutputStream bos = new ByteArrayOutputStream(totalToTransfer);

long transferred = 0;
byte[] bytes = new byte[4096];
int readResult = 0;
while ((readResult = bis.read(bytes)) != -1) {
bos.write(bytes, 0, readResult);
transferred += readResult;
}
// TODO check total bytes transferred?

// find out etag
String etag = WebdavUtils.getEtagFromResponse(get);
if (etag.length() == 0) {
Log_OC.w(TAG, "Could not read Etag from avatar");
}

// Result
result = new RemoteOperationResult(true, status, get.getResponseHeaders());
ResultData resultData = new ResultData(bos.toByteArray(), mimeType, etag);
ArrayList<Object> data = new ArrayList<Object>();
data.add(resultData);
result.setData(data);

} else {
client.exhaustResponse(get.getResponseBodyAsStream());
result = new RemoteOperationResult(false, status, get.getResponseHeaders());
}

} catch (Exception e) {
result = new RemoteOperationResult(e);
Log_OC.e(TAG, "Exception while getting OC user avatar", e);

} finally {
if (get != null) {
if (inputStream != null) {
client.exhaustResponse(inputStream);
}
get.releaseConnection();
}
}

return result;
}

private boolean isSuccess(int status) {
return (status == HttpStatus.SC_OK);
}

public static class ResultData {
private String mEtag;
private String mMimeType;
private byte[] mAvatarData;

ResultData(byte[] avatarData, String mimeType, String etag) {
mAvatarData = avatarData;
mMimeType = (mimeType == null) ? "" : mimeType;
mEtag = (etag == null) ? "" : etag;
}

public String getEtag() {
return mEtag;
}

public String getMimeType() {
return mMimeType;
}

public byte[] getAvatarData() {
return mAvatarData;
}
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ protected RemoteOperationResult run(OwnCloudClient client) {
status = client.executeMethod(get);
if (isSuccess(status)) {
String response = get.getResponseBodyAsString();
Log_OC.d(TAG, "Successful response: " + response);
Log_OC.d(TAG, "Successful response");

// Parse the response
JSONObject respJSON = new JSONObject(response);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@
import com.owncloud.android.lib.resources.shares.RemoveRemoteShareOperation;
import com.owncloud.android.lib.resources.shares.ShareType;
import com.owncloud.android.lib.resources.users.GetRemoteUserQuotaOperation;
import com.owncloud.android.lib.resources.users.GetRemoteUserAvatarOperation;

/**
* Activity to test OC framework
Expand Down Expand Up @@ -337,7 +338,12 @@ public RemoteOperationResult getQuota() {
GetRemoteUserQuotaOperation getUserQuotaOperation = new GetRemoteUserQuotaOperation();
return getUserQuotaOperation.execute(mClient);
}



public RemoteOperationResult getUserAvatar(int dimension, String etag) {
GetRemoteUserAvatarOperation getUserAvatarOperation = new GetRemoteUserAvatarOperation(dimension, etag);
return getUserAvatarOperation.execute(mClient);
}

/**
* Extracts file from AssetManager to cache folder.
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
/* ownCloud Android Library is available under MIT license
*
* Copyright (C) 2016 ownCloud GmbH.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
*/

package com.owncloud.android.lib.test_project.test;

import java.util.*;

import com.owncloud.android.lib.common.operations.RemoteOperationResult;
import com.owncloud.android.lib.resources.users.GetRemoteUserAvatarOperation.ResultData;
import com.owncloud.android.lib.test_project.TestActivity;

import org.apache.commons.httpclient.HttpStatus;

/**
* Class to test {@link GetRemoteUserAvatarOperation}
*
* @author David A. Velasco
*/
public class GetUserAvatarTest extends RemoteTest {


private static final String LOG_TAG = GetUserAvatarTest.class.getCanonicalName();

private static final int AVATAR_DIMENSION = 256;

private TestActivity mActivity;


@Override
protected void setUp() throws Exception {
super.setUp();
mActivity = getActivity();
}

/**
* Test get user avatar (succesful)
*/
public void testGetUserAvatar() {
RemoteOperationResult result = mActivity.getUserAvatar(AVATAR_DIMENSION, null);
assertTrue(result.isSuccess());
ResultData userAvatar = (ResultData) result.getData().get(0);
assertTrue(userAvatar.getEtag() != null && userAvatar.getEtag().length() > 0);
assertTrue(userAvatar.getMimeType() != null && userAvatar.getMimeType().startsWith("image"));
assertTrue(userAvatar.getAvatarData() != null && userAvatar.getAvatarData().length > 0);
}

/**
* Test get user avatar only if changed, but wasn't changed
*/
public void testGetUserAvatarOnlyIfChangedAfterUnchanged() {
RemoteOperationResult result = mActivity.getUserAvatar(AVATAR_DIMENSION, null);
ResultData userAvatar = (ResultData) result.getData().get(0);
String etag = userAvatar.getEtag();

// request again, with the just received etag
result = mActivity.getUserAvatar(AVATAR_DIMENSION, etag);
assertFalse(result.isSuccess());
assertTrue(result.getHttpCode() == HttpStatus.SC_NOT_MODIFIED);
}

/**
* Test get user avatar only if changed, and was changed
*/
public void testGetUserAvatarOnlyIfChangedAfterChanged() {
// TODO can't test this without provisioning API or mocking the server
}

/**
* Test get user avatar not found
*/
public void testGetUserAvatarNofFound() {
// TODO can't test this without provisioning API, mocking the server or another set of credentials
}

@Override
protected void tearDown() throws Exception {
super.tearDown();
}

}

0 comments on commit a13b5ec

Please sign in to comment.