/* * Nextcloud SingleSignOn * * @author David Luhmer * Copyright (C) 2019 David Luhmer * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * More information here: https://github.com/abeluck/android-streams-ipc */ package com.nextcloud.android.sso; import android.accounts.Account; import android.accounts.AuthenticatorException; import android.accounts.OperationCanceledException; import android.content.Context; import android.content.SharedPreferences; import android.net.Uri; import android.os.Binder; import android.os.ParcelFileDescriptor; import android.text.TextUtils; import android.util.Log; import com.nextcloud.android.sso.aidl.IInputStreamService; import com.nextcloud.android.sso.aidl.NextcloudRequest; import com.nextcloud.android.sso.aidl.ParcelFileDescriptorUtil; import com.nextcloud.client.account.UserAccountManager; import com.owncloud.android.lib.common.OwnCloudAccount; import com.owncloud.android.lib.common.OwnCloudClient; import com.owncloud.android.lib.common.OwnCloudClientManager; import com.owncloud.android.lib.common.OwnCloudClientManagerFactory; import com.owncloud.android.lib.common.utils.Log_OC; import com.owncloud.android.utils.EncryptionUtils; import org.apache.commons.httpclient.HttpConnection; import org.apache.commons.httpclient.HttpMethodBase; import org.apache.commons.httpclient.HttpState; import org.apache.commons.httpclient.NameValuePair; import org.apache.commons.httpclient.methods.DeleteMethod; import org.apache.commons.httpclient.methods.GetMethod; import org.apache.commons.httpclient.methods.HeadMethod; import org.apache.commons.httpclient.methods.InputStreamRequestEntity; import org.apache.commons.httpclient.methods.PostMethod; import org.apache.commons.httpclient.methods.PutMethod; import org.apache.commons.httpclient.methods.RequestEntity; import org.apache.commons.httpclient.methods.StringRequestEntity; import org.apache.jackrabbit.webdav.DavConstants; import org.apache.jackrabbit.webdav.client.methods.MkColMethod; import org.apache.jackrabbit.webdav.client.methods.PropFindMethod; import org.apache.jackrabbit.webdav.property.DavPropertyNameSet; import java.io.BufferedReader; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable; import java.util.Collection; import java.util.List; import java.util.Map; import androidx.annotation.VisibleForTesting; import static com.nextcloud.android.sso.Constants.DELIMITER; import static com.nextcloud.android.sso.Constants.EXCEPTION_ACCOUNT_NOT_FOUND; import static com.nextcloud.android.sso.Constants.EXCEPTION_HTTP_REQUEST_FAILED; import static com.nextcloud.android.sso.Constants.EXCEPTION_INVALID_REQUEST_URL; import static com.nextcloud.android.sso.Constants.EXCEPTION_INVALID_TOKEN; import static com.nextcloud.android.sso.Constants.EXCEPTION_UNSUPPORTED_METHOD; import static com.nextcloud.android.sso.Constants.SSO_SHARED_PREFERENCE; /** * Stream binder to pass usable InputStreams across the process boundary in Android. */ public class InputStreamBinder extends IInputStreamService.Stub { private final static String TAG = "InputStreamBinder"; private static final String CONTENT_TYPE_APPLICATION_JSON = "application/json"; private static final String CHARSET_UTF8 = "UTF-8"; private static final int HTTP_STATUS_CODE_OK = 200; private static final int HTTP_STATUS_CODE_MULTIPLE_CHOICES = 300; private static final char PATH_SEPARATOR = '/'; private static final int ZERO_LENGTH = 0; private Context context; private UserAccountManager accountManager; public InputStreamBinder(Context context, UserAccountManager accountManager) { this.context = context; this.accountManager = accountManager; } public ParcelFileDescriptor performNextcloudRequestV2(ParcelFileDescriptor input) { return performNextcloudRequestAndBodyStreamV2(input, null); } public ParcelFileDescriptor performNextcloudRequestAndBodyStreamV2( ParcelFileDescriptor input, ParcelFileDescriptor requestBodyParcelFileDescriptor) { // read the input final InputStream is = new ParcelFileDescriptor.AutoCloseInputStream(input); final InputStream requestBodyInputStream = requestBodyParcelFileDescriptor != null ? new ParcelFileDescriptor.AutoCloseInputStream(requestBodyParcelFileDescriptor) : null; Exception exception = null; Response response = new Response(); try { // Start request and catch exceptions NextcloudRequest request = deserializeObjectAndCloseStream(is); response = processRequestV2(request, requestBodyInputStream); } catch (Exception e) { Log_OC.e(TAG, "Error during Nextcloud request", e); exception = e; } try { // Write exception to the stream followed by the actual network stream InputStream exceptionStream = serializeObjectToInputStreamV2(exception, response.getPlainHeadersString()); InputStream resultStream = new java.io.SequenceInputStream(exceptionStream, response.getBody()); return ParcelFileDescriptorUtil.pipeFrom(resultStream, thread -> Log.d(TAG, "Done sending result"), response.getMethod()); } catch (IOException e) { Log_OC.e(TAG, "Error while sending response back to client app", e); } return null; } public ParcelFileDescriptor performNextcloudRequest(ParcelFileDescriptor input) { return performNextcloudRequestAndBodyStream(input, null); } public ParcelFileDescriptor performNextcloudRequestAndBodyStream( ParcelFileDescriptor input, ParcelFileDescriptor requestBodyParcelFileDescriptor) { // read the input final InputStream is = new ParcelFileDescriptor.AutoCloseInputStream(input); final InputStream requestBodyInputStream = requestBodyParcelFileDescriptor != null ? new ParcelFileDescriptor.AutoCloseInputStream(requestBodyParcelFileDescriptor) : null; Exception exception = null; HttpMethodBase httpMethod = null; InputStream httpStream = new InputStream() { @Override public int read() { return ZERO_LENGTH; } }; try { // Start request and catch exceptions NextcloudRequest request = deserializeObjectAndCloseStream(is); httpMethod = processRequest(request, requestBodyInputStream); httpStream = httpMethod.getResponseBodyAsStream(); } catch (Exception e) { Log_OC.e(TAG, "Error during Nextcloud request", e); exception = e; } try { // Write exception to the stream followed by the actual network stream InputStream exceptionStream = serializeObjectToInputStream(exception); InputStream resultStream; if (httpStream != null) { resultStream = new java.io.SequenceInputStream(exceptionStream, httpStream); } else { resultStream = exceptionStream; } return ParcelFileDescriptorUtil.pipeFrom(resultStream, thread -> Log.d(TAG, "Done sending result"), httpMethod); } catch (IOException e) { Log_OC.e(TAG, "Error while sending response back to client app", e); } return null; } private ByteArrayInputStream serializeObjectToInputStreamV2(Exception exception, String headers) { byte[] baosByteArray = new byte[0]; try (ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos)) { oos.writeObject(exception); oos.writeObject(headers); oos.flush(); oos.close(); baosByteArray = baos.toByteArray(); } catch (IOException e) { Log_OC.e(TAG, "Error while sending response back to client app", e); } return new ByteArrayInputStream(baosByteArray); } private ByteArrayInputStream serializeObjectToInputStream(T obj) throws IOException { ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(obj); oos.flush(); oos.close(); return new ByteArrayInputStream(baos.toByteArray()); } private T deserializeObjectAndCloseStream(InputStream is) throws IOException, ClassNotFoundException { ObjectInputStream ois = new ObjectInputStream(is); T result = (T) ois.readObject(); is.close(); ois.close(); return result; } public class NCPropFindMethod extends PropFindMethod { NCPropFindMethod(String uri, int propfindType, int depth) throws IOException { super(uri, propfindType, new DavPropertyNameSet(), depth); } @Override protected void processResponseBody(HttpState httpState, HttpConnection httpConnection) { // Do not process the response body here. Instead pass it on to client app. } } private HttpMethodBase buildMethod(NextcloudRequest request, Uri baseUri, InputStream requestBodyInputStream) throws IOException { String requestUrl = baseUri + request.getUrl(); HttpMethodBase method; switch (request.getMethod()) { case "GET": method = new GetMethod(requestUrl); break; case "POST": method = new PostMethod(requestUrl); if (requestBodyInputStream != null) { RequestEntity requestEntity = new InputStreamRequestEntity(requestBodyInputStream); ((PostMethod) method).setRequestEntity(requestEntity); } else if (request.getRequestBody() != null) { StringRequestEntity requestEntity = new StringRequestEntity( request.getRequestBody(), CONTENT_TYPE_APPLICATION_JSON, CHARSET_UTF8); ((PostMethod) method).setRequestEntity(requestEntity); } break; case "PATCH": method = new PatchMethod(requestUrl); if (requestBodyInputStream != null) { RequestEntity requestEntity = new InputStreamRequestEntity(requestBodyInputStream); ((PatchMethod) method).setRequestEntity(requestEntity); } else if (request.getRequestBody() != null) { StringRequestEntity requestEntity = new StringRequestEntity( request.getRequestBody(), CONTENT_TYPE_APPLICATION_JSON, CHARSET_UTF8); ((PatchMethod) method).setRequestEntity(requestEntity); } break; case "PUT": method = new PutMethod(requestUrl); if (requestBodyInputStream != null) { RequestEntity requestEntity = new InputStreamRequestEntity(requestBodyInputStream); ((PutMethod) method).setRequestEntity(requestEntity); } else if (request.getRequestBody() != null) { StringRequestEntity requestEntity = new StringRequestEntity( request.getRequestBody(), CONTENT_TYPE_APPLICATION_JSON, CHARSET_UTF8); ((PutMethod) method).setRequestEntity(requestEntity); } break; case "DELETE": method = new DeleteMethod(requestUrl); break; case "PROPFIND": method = new NCPropFindMethod(requestUrl, DavConstants.PROPFIND_ALL_PROP, DavConstants.DEPTH_1); if (request.getRequestBody() != null) { //text/xml; charset=UTF-8 is taken from XmlRequestEntity... Should be application/xml StringRequestEntity requestEntity = new StringRequestEntity( request.getRequestBody(), "text/xml; charset=UTF-8", CHARSET_UTF8); ((PropFindMethod) method).setRequestEntity(requestEntity); } break; case "MKCOL": method = new MkColMethod(requestUrl); break; case "HEAD": method = new HeadMethod(requestUrl); break; default: throw new UnsupportedOperationException(EXCEPTION_UNSUPPORTED_METHOD); } return method; } private HttpMethodBase processRequest(final NextcloudRequest request, final InputStream requestBodyInputStream) throws UnsupportedOperationException, com.owncloud.android.lib.common.accounts.AccountUtils.AccountNotFoundException, OperationCanceledException, AuthenticatorException, IOException { Account account = accountManager.getAccountByName(request.getAccountName()); if (account == null) { throw new IllegalStateException(EXCEPTION_ACCOUNT_NOT_FOUND); } // Validate token if (!isValid(request)) { throw new IllegalStateException(EXCEPTION_INVALID_TOKEN); } // Validate URL if (request.getUrl().length() == 0 || request.getUrl().charAt(0) != PATH_SEPARATOR) { throw new IllegalStateException(EXCEPTION_INVALID_REQUEST_URL, new IllegalStateException("URL need to start with a /")); } OwnCloudClientManager ownCloudClientManager = OwnCloudClientManagerFactory.getDefaultSingleton(); OwnCloudAccount ocAccount = new OwnCloudAccount(account, context); OwnCloudClient client = ownCloudClientManager.getClientFor(ocAccount, context); HttpMethodBase method = buildMethod(request, client.getBaseUri(), requestBodyInputStream); if (request.getParameterV2() != null && !request.getParameterV2().isEmpty()) { method.setQueryString(convertListToNVP(request.getParameterV2())); } else { method.setQueryString(convertMapToNVP(request.getParameter())); } method.addRequestHeader("OCS-APIREQUEST", "true"); for (Map.Entry> header : request.getHeader().entrySet()) { // https://stackoverflow.com/a/3097052 method.addRequestHeader(header.getKey(), TextUtils.join(",", header.getValue())); if ("OCS-APIREQUEST".equalsIgnoreCase(header.getKey())) { throw new IllegalStateException( "The 'OCS-APIREQUEST' header will be automatically added by the Nextcloud SSO Library. " + "Please remove the header before making a request"); } } client.setFollowRedirects(request.isFollowRedirects()); int status = client.executeMethod(method); // Check if status code is 2xx --> https://en.wikipedia.org/wiki/List_of_HTTP_status_codes#2xx_Success if (status >= HTTP_STATUS_CODE_OK && status < HTTP_STATUS_CODE_MULTIPLE_CHOICES) { return method; } else { InputStream inputStream = method.getResponseBodyAsStream(); String total = "No response body"; // If response body is available if (inputStream != null) { total = inputStreamToString(inputStream); Log_OC.e(TAG, total); } method.releaseConnection(); throw new IllegalStateException(EXCEPTION_HTTP_REQUEST_FAILED, new IllegalStateException(String.valueOf(status), new IllegalStateException(total))); } } private Response processRequestV2(final NextcloudRequest request, final InputStream requestBodyInputStream) throws UnsupportedOperationException, com.owncloud.android.lib.common.accounts.AccountUtils.AccountNotFoundException, OperationCanceledException, AuthenticatorException, IOException { Account account = accountManager.getAccountByName(request.getAccountName()); if (account == null) { throw new IllegalStateException(EXCEPTION_ACCOUNT_NOT_FOUND); } // Validate token if (!isValid(request)) { throw new IllegalStateException(EXCEPTION_INVALID_TOKEN); } // Validate URL if (request.getUrl().length() == 0 || request.getUrl().charAt(0) != PATH_SEPARATOR) { throw new IllegalStateException(EXCEPTION_INVALID_REQUEST_URL, new IllegalStateException("URL need to start with a /")); } OwnCloudClientManager ownCloudClientManager = OwnCloudClientManagerFactory.getDefaultSingleton(); OwnCloudAccount ocAccount = new OwnCloudAccount(account, context); OwnCloudClient client = ownCloudClientManager.getClientFor(ocAccount, context); HttpMethodBase method = buildMethod(request, client.getBaseUri(), requestBodyInputStream); if (request.getParameterV2() != null && !request.getParameterV2().isEmpty()) { method.setQueryString(convertListToNVP(request.getParameterV2())); } else { method.setQueryString(convertMapToNVP(request.getParameter())); } method.addRequestHeader("OCS-APIREQUEST", "true"); for (Map.Entry> header : request.getHeader().entrySet()) { // https://stackoverflow.com/a/3097052 method.addRequestHeader(header.getKey(), TextUtils.join(",", header.getValue())); if ("OCS-APIREQUEST".equalsIgnoreCase(header.getKey())) { throw new IllegalStateException( "The 'OCS-APIREQUEST' header will be automatically added by the Nextcloud SSO Library. " + "Please remove the header before making a request"); } } client.setFollowRedirects(request.isFollowRedirects()); int status = client.executeMethod(method); // Check if status code is 2xx --> https://en.wikipedia.org/wiki/List_of_HTTP_status_codes#2xx_Success if (status >= HTTP_STATUS_CODE_OK && status < HTTP_STATUS_CODE_MULTIPLE_CHOICES) { return new Response(method); } else { InputStream inputStream = method.getResponseBodyAsStream(); String total = "No response body"; // If response body is available if (inputStream != null) { total = inputStreamToString(inputStream); Log_OC.e(TAG, total); } method.releaseConnection(); throw new IllegalStateException(EXCEPTION_HTTP_REQUEST_FAILED, new IllegalStateException(String.valueOf(status), new IllegalStateException(total))); } } private boolean isValid(NextcloudRequest request) { String callingPackageName = context.getPackageManager().getNameForUid(Binder.getCallingUid()); SharedPreferences sharedPreferences = context.getSharedPreferences(SSO_SHARED_PREFERENCE, Context.MODE_PRIVATE); String hash = sharedPreferences.getString(callingPackageName + DELIMITER + request.getAccountName(), ""); return validateToken(hash, request.getToken()); } private boolean validateToken(String hash, String token) { if (hash.isEmpty() || !hash.contains("$")) { throw new IllegalStateException(EXCEPTION_INVALID_TOKEN); } String salt = hash.split("\\$")[1]; // TODO extract "$" String newHash = EncryptionUtils.generateSHA512(token, salt); // As discussed with Lukas R. at the Nextcloud Conf 2018, always compare whole strings // and don't exit prematurely if the string does not match anymore to prevent timing-attacks return isEqual(hash.getBytes(), newHash.getBytes()); } // Taken from http://codahale.com/a-lesson-in-timing-attacks/ private static boolean isEqual(byte[] a, byte[] b) { if (a.length != b.length) { return false; } int result = 0; for (int i = 0; i < a.length; i++) { result |= a[i] ^ b[i]; } return result == 0; } private static String inputStreamToString(InputStream inputStream) { try { StringBuilder total = new StringBuilder(); BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream)); String line = reader.readLine(); while (line != null) { total.append(line).append('\n'); line = reader.readLine(); } return total.toString(); } catch (Exception e) { return e.getMessage(); } } @VisibleForTesting public static NameValuePair[] convertMapToNVP(Map map) { NameValuePair[] nvp = new NameValuePair[map.size()]; int i = 0; for (String key : map.keySet()) { nvp[i] = new NameValuePair(key, map.get(key)); i++; } return nvp; } @VisibleForTesting public static NameValuePair[] convertListToNVP(Collection list) { NameValuePair[] nvp = new NameValuePair[list.size()]; int i = 0; for (QueryParam pair : list) { nvp[i] = new NameValuePair(pair.key, pair.value); i++; } return nvp; } }