/* * Copyright (C) 2013-2017 microG Project Team * * Licensed 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.microg.gms.auth; import static org.microg.gms.auth.AuthConstants.DEFAULT_ACCOUNT_TYPE; import static org.microg.gms.auth.AuthConstants.PROVIDER_EXTRA_ACCOUNTS; import static org.microg.gms.auth.AuthConstants.PROVIDER_EXTRA_CLEAR_PASSWORD; import static org.microg.gms.auth.AuthConstants.PROVIDER_METHOD_CLEAR_PASSWORD; import static org.microg.gms.auth.AuthConstants.PROVIDER_METHOD_GET_ACCOUNTS; import android.Manifest; import android.accounts.Account; import android.accounts.AccountManager; import android.content.ContentProvider; import android.content.ContentValues; import android.content.pm.PackageManager; import android.database.Cursor; import android.net.Uri; import android.os.Binder; import android.os.Build; import android.os.Bundle; import android.util.Log; import androidx.annotation.Nullable; import org.microg.gms.common.PackageUtils; import java.util.Arrays; public class AccountContentProvider extends ContentProvider { private static final String TAG = "GmsAuthProvider"; @Override public boolean onCreate() { return true; } @Nullable @Override public Bundle call(String method, String arg, Bundle extras) { String suggestedPackageName = null; if (Build.VERSION.SDK_INT > Build.VERSION_CODES.KITKAT) { suggestedPackageName = getCallingPackage(); } String packageName = PackageUtils.getAndCheckCallingPackage(getContext(), suggestedPackageName); Log.d(TAG, "Call from " + packageName); if (!PackageUtils.callerHasExtendedAccess(getContext())) { String[] packagesForUid = getContext().getPackageManager().getPackagesForUid(Binder.getCallingUid()); if (packagesForUid != null && packagesForUid.length != 0) Log.w(TAG, "Not granting extended access to " + Arrays.toString(packagesForUid) + ", signature: " + PackageUtils.firstSignatureDigest(getContext(), packagesForUid[0])); if (getContext().checkCallingPermission(Manifest.permission.GET_ACCOUNTS) != PackageManager.PERMISSION_GRANTED) throw new SecurityException("Access denied, missing GET_ACCOUNTS or EXTENDED_ACCESS permission"); } if (PROVIDER_METHOD_GET_ACCOUNTS.equals(method)) { Bundle result = new Bundle(); Account[] accounts = null; if (arg != null && (arg.equals(DEFAULT_ACCOUNT_TYPE) || arg.startsWith(DEFAULT_ACCOUNT_TYPE + "."))) { AccountManager am = AccountManager.get(getContext()); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) { accounts = am.getAccountsByTypeForPackage(arg, packageName); } if (accounts == null || accounts.length == 0) { accounts = am.getAccountsByType(arg); } } if (accounts == null) { accounts = new Account[0]; } result.putParcelableArray(PROVIDER_EXTRA_ACCOUNTS, accounts); return result; } else if (PROVIDER_METHOD_CLEAR_PASSWORD.equals(method)) { Account a = extras.getParcelable(PROVIDER_EXTRA_CLEAR_PASSWORD); AccountManager.get(getContext()).clearPassword(a); return null; } throw new UnsupportedOperationException(String.format("Unsupported method call %s(%s).", method, arg)); } @Nullable @Override public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) { throw new UnsupportedOperationException(); } @Nullable @Override public String getType(Uri uri) { return "text/plain"; } @Nullable @Override public Uri insert(Uri uri, ContentValues values) { throw new UnsupportedOperationException(); } @Override public int delete(Uri uri, String selection, String[] selectionArgs) { throw new UnsupportedOperationException(); } @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { throw new UnsupportedOperationException(); } }