I've integrated Google Sign-In
in my application. I can get user's Email
and DisplayName
. Now, I want to get user's Birthdate
and Gender
.
I've added all required requests
& Scopes
into GoogleApiClient
which all are granted by API. here's code.
// [START configure_signin]
// Configure sign-in to request the user's ID, email address, and basic
// profile. ID and basic profile are included in DEFAULT_SIGN_IN.
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
.requestEmail()
.requestProfile() <- This
.requestScopes(
new Scope(Scopes.PLUS_ME), new Scope(Scopes.PROFILE) <- This
)
.build();
// [END configure_signin]
// [START build_client]
// Build a GoogleApiClient with access to the Google Sign-In API and the
// options specified by gso.
mGoogleApiClient = new GoogleApiClient.Builder(this)
.enableAutoManage(this /* FragmentActivity */, new GoogleApiClient.OnConnectionFailedListener() {
@Override
public void onConnectionFailed(@NonNull ConnectionResult connectionResult) {
// An unresolvable error has occurred and Google APIs (including Sign-In) will not
// be available.
Log.d(TAG, "onConnectionFailed:" + connectionResult);
}
} /* OnConnectionFailedListener */)
.addApi(Auth.GOOGLE_SIGN_IN_API, gso)
.addScope(new Scope(Scopes.PLUS_ME)) <- This
.addScope(new Scope(Scopes.PROFILE)) <- This
.build();
// [END build_client]
Here's the granted Scopes in GoogleSignInAccount
private void setupUserData(GoogleSignInAccount acct) {
if (acct != null) {
mPrefs.setName(acct.getDisplayName());
mPrefs.setEmail(acct.getEmail());
if (acct.getPhotoUrl() != null) {
mPrefs.setPicURL(acct.getPhotoUrl().toString());
}
Set<Scope> scopes = acct.getGrantedScopes(); <- This
for (Scope scope : scopes) {
Log.d(TAG, "setupUserData: " + scope.toString()); <- This
}
}
}
Here's the log of granted scopes
D/SplashActivity: setupUserData: GrantedScopes size 6
D/SplashActivity: setupUserData: https://www.googleapis.com/auth/plus.me
D/SplashActivity: setupUserData: https://www.googleapis.com/auth/userinfo.email
D/SplashActivity: setupUserData: https://www.googleapis.com/auth/userinfo.profile
D/SplashActivity: setupUserData: email
D/SplashActivity: setupUserData: profile
D/SplashActivity: setupUserData: openid
Here's my Google Mobile Service's dependency
compile 'com.google.android.gms:play-services-auth:10.2.0'
compile 'com.google.android.gms:play-services-plus:10.2.0'
Now, I don't know how to get access to user's profile information
.
As mentioned in Getting people and profile information, to get additional profile information and a user's contacts, use the People API. You must get consent from the user to access this information by requesting additional scopes when the user signs in.
You can call people.get
, passing in a resource name, to get private contact and public profile data for each person. If your request is successful, the response contains an instance of a Person including birthday and gender.
You may want to visit the links I've provided for more information.
Here is the complete working example, hope it help for future reader. What the app do is sign in(Sign In API included name and email) first, then request birthday & gender(People API) authentication, and save it to SharedPreferences
for reuse on next launch. Finally it will print basic info and advanced (gender & birthday) info.
public class MainActivity extends AppCompatActivity { static final private int RC_SIGN_IN = 1; static final private String TAG = "hole"; private WeakReference<MainActivity> weakAct = new WeakReference<>(this); private GoogleSignInClient mGoogleSignInClient; private GoogleSignInAccount account; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Scope myScope = new Scope("https://www.googleapis.com/auth/user.birthday.read"); Scope myScope2 = new Scope(Scopes.PLUS_ME); Scope myScope3 = new Scope(Scopes.PROFILE); //get name and id GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN) .requestScopes(myScope, myScope2) .requestEmail() .requestProfile() .build(); mGoogleSignInClient = GoogleSignIn.getClient(this, gso); account = GoogleSignIn.getLastSignedInAccount(this); if (account == null) { reqPerm(); } else { SharedPreferences sharedPref = getSharedPreferences(account.getId(), MODE_PRIVATE); if (sharedPref.contains("gender")) { printBasic(); printAdvanced(); } else { new GetProfileDetails(account, weakAct, TAG).execute(); } } } private void reqPerm() { Intent signInIntent = mGoogleSignInClient.getSignInIntent(); startActivityForResult(signInIntent, RC_SIGN_IN); } private void printBasic() { account = GoogleSignIn.getLastSignedInAccount(this); if (account != null) { Log.d(TAG, "latest sign in: " + "\n\tPhoto url:" + account.getPhotoUrl() + "\n\tEmail:" + account.getEmail() + "\n\tDisplay name:" + account.getDisplayName() + "\n\tFamily(last) name:" + account.getFamilyName() + "\n\tGiven(first) name:" + account.getGivenName() + "\n\tId:" + account.getId() + "\n\tIdToken:" + account.getIdToken() ); } else { Log.w(TAG, "basic info is null"); } } private void saveAdvanced(Person meProfile) { account = GoogleSignIn.getLastSignedInAccount(this); if (account != null) { SharedPreferences sharedPref = getSharedPreferences(account.getId(), MODE_PRIVATE); SharedPreferences.Editor editor = sharedPref.edit(); if (n.size() > 0) { try { Log.d("hole", "g name: " + n); editor.putString("givenName", n.get(0).getGivenName()); editor.putString("familyName", n.get(0).getFamilyName()); editor.putString("id", n.get(0).getMetadata().getSource().getId()); } catch (Exception e) { e.printStackTrace(); //this one should act as fallback priority since it got problem to get name without wait for ~1 minute // ... when create new account will get empty name editor.putString("id", account.getId()); editor.putString("givenName", account.getGivenName()); editor.putString("familyName", account.getFamilyName()); } } List<Gender> genders = meProfile.getGenders(); if (genders != null && genders.size() > 0) { String gender = genders.get(0).getValue(); Log.d(TAG, "onPostExecute gender: " + gender); editor.putString("gender", gender); } else { Log.d(TAG, "onPostExecute no gender if set to private "); editor.putString("gender", ""); //save as main key to know pref saved } List<Birthday> birthdays = meProfile.getBirthdays(); if (birthdays != null && birthdays.size() > 0) { for (Birthday b : birthdays) { //birthday still able to get even private, unlike gender Date bdate = b.getDate(); if (bdate != null) { String bday, bmonth, byear; if (bdate.getDay() != null) bday = bdate.getDay().toString(); else bday = ""; if (bdate.getMonth() != null) bmonth = bdate.getMonth().toString(); else bmonth = ""; if (bdate.getYear() != null) byear = bdate.getYear().toString(); else byear = ""; editor.putString("bday", bday); editor.putString("bmonth", bmonth); editor.putString("byear", byear); } } } else { Log.w(TAG, "saveAdvanced no birthday"); } editor.commit(); //next instruction is print from pref, so don't use apply() } else { Log.w(TAG, "saveAdvanced no acc"); } } private void printAdvanced() { account = GoogleSignIn.getLastSignedInAccount(this); if (account != null) { SharedPreferences sharedPref = getSharedPreferences(account.getId(), MODE_PRIVATE); if (sharedPref.contains("gender")) { //this checking works since null still saved String gender = sharedPref.getString("gender", ""); Log.d(TAG, "gender: " + gender); if (sharedPref.contains("bday")) { //this checking works since null still saved String bday = sharedPref.getString("bday", ""); String bmonth = sharedPref.getString("bmonth", ""); String byear = sharedPref.getString("byear", ""); Log.d(TAG, bday + "/" + bmonth + "/" + byear); } else { Log.w(TAG, "failed ot get birthday from pref"); } String givenName = sharedPref.getString("givenName", ""); String familyName = sharedPref.getString("familyName", ""); String id = sharedPref.getString("id", ""); } else { Log.w(TAG, "failed ot get data from pref -2"); } } else { Log.w(TAG, "failed ot get data from pref -1"); } } @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == RC_SIGN_IN) { if (resultCode == Activity.RESULT_OK) { Task<GoogleSignInAccount> task = GoogleSignIn.getSignedInAccountFromIntent(data); handleSignInResult(task); } else { Log.w(TAG, "failed, user denied OR no network OR jks SHA1 not configure yet at play console android project"); } } } private void handleSignInResult(Task<GoogleSignInAccount> completedTask) { try { GoogleSignInAccount account = completedTask.getResult(ApiException.class); // Signed in successfully, show authenticated UI. new GetProfileDetails(account, weakAct, TAG).execute(); } catch (ApiException e) { //cancel choose acc will come here with status code 12501 if not check RESULT_OK // , more status code at: //https://developers.google.com/android/reference/com/google/android/gms/auth/api/signin/GoogleSignInStatusCodes Log.w(TAG, "signInResult:failed code=" + e.getStatusCode()); } } static class GetProfileDetails extends AsyncTask<Void, Void, Person> { private PeopleService ps; private int authError = -1; private WeakReference<MainActivity> weakAct; private String TAG; GetProfileDetails(GoogleSignInAccount account, WeakReference<MainActivity> weakAct, String TAG) { this.TAG = TAG; this.weakAct = weakAct; GoogleAccountCredential credential = GoogleAccountCredential.usingOAuth2( this.weakAct.get(), Collections.singleton(Scopes.PROFILE)); credential.setSelectedAccount( new Account(account.getEmail(), "com.google")); HttpTransport HTTP_TRANSPORT = AndroidHttp.newCompatibleTransport(); JsonFactory JSON_FACTORY = JacksonFactory.getDefaultInstance(); ps = new PeopleService.Builder(HTTP_TRANSPORT, JSON_FACTORY, credential) .setApplicationName("Google Sign In Quickstart") .build(); } @Override protected Person doInBackground(Void... params) { Person meProfile = null; try { meProfile = ps .people() .get("people/me") .setPersonFields("names,genders,birthdays") .execute(); } catch (UserRecoverableAuthIOException e) { e.printStackTrace(); authError = 0; } catch (GoogleJsonResponseException e) { e.printStackTrace(); authError = 1; } catch (IOException e) { e.printStackTrace(); authError = 2; } return meProfile; } @Override protected void onPostExecute(Person meProfile) { MainActivity mainAct = weakAct.get(); if (mainAct != null) { mainAct.printBasic(); if (authError == 0) { //app has been revoke, re-authenticated required. mainAct.reqPerm(); } else if (authError == 1) { Log.w(TAG, "People API might not enable at" + " https://console.developers.google.com/apis/library/people.googleapis.com/?project=<project name>"); } else if (authError == 2) { Log.w(TAG, "API io error"); } else { if (meProfile != null) { mainAct.saveAdvanced(meProfile); mainAct.printAdvanced(); } } } } } }
Reminder:
<uses-permission android:name="android.permission.INTERNET" />
in AndroidManifest.xml.implementation 'com.google.android.gms:play-services-auth:12.0.1'
, implementation 'com.google.apis:google-api-services-people:v1-rev255-1.23.0'
, and implementation 'com.google.api-client:google-api-client-android:1.23.0'
in dependencies {}
of build.gradle
.compileSdkVersion
, targetSdkVersion
, and appcompat-v7
from 27 to 26 since I got warning after #2 dependencies added.signingConfigs { debug { storeFile file('<path to jks file>') keyAlias '<your key alias>' keyPassword '<your key password>' storePassword '<your store password>' } }
in build.gradle
, which jks file generated from Build
-> Generated Signed APK...
-> Create new...
keytool -exportcert -keystore <path to jks file> -list -v
to get SHA1 hex key, then visits play console, fill in project name, app package name, SHA1 hex key.Scopes.BIRTHDAY
in library, so I have to hard-coded the birthday endpoint URL"https://www.googleapis.com/auth/user.birthday.read"
, which the link can get from https://developers.google.com/people/v1/how-tos/authorizing#profile-scopes OR "Show Scopes" in "Try it API" panel at https://developers.google.com/people/api/rest/v1/people/get account.getGivenName()
and account.getFamilyName()
.Gradle
implementation "com.google.android.gms:play-services-auth:${google_play_service_version}"
implementation 'com.google.apis:google-api-services-people:v1-rev354-1.25.0'
implementation ('com.google.api-client:google-api-client-android:1.23.0') {
exclude group: 'org.apache.httpcomponents'
}
Authentication
private void setupGoogleLogin() {
GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
.requestIdToken(getString(R.string.default_web_client_id))
.requestScopes(new Scope(PeopleApi.CONTACT_SCOPE), new Scope(PeopleApi.BIRTHDAY_SCOPE))
.requestEmail()
.build();
mGoogleApiClient = new GoogleApiClient.Builder(this)
.enableAutoManage(this, mOnConnectionFailedListener)
.addApi(Auth.GOOGLE_SIGN_IN_API, gso)
.build();
}
PeopleApi
public class PeopleApi {
public static final String CONTACT_SCOPE = "https://www.googleapis.com/auth/contacts.readonly";
public static final String BIRTHDAY_SCOPE = "https://www.googleapis.com/auth/user.birthday.read";
private static PeopleService mInstance;
private static PeopleService getService() {
if (mInstance == null) mInstance = initializeService();
return mInstance;
}
private static PeopleService initializeService() {
Context context = BHApp.getContext();
GoogleAccountCredential credential =
GoogleAccountCredential.usingOAuth2(context, Arrays.asList(CONTACT_SCOPE, BIRTHDAY_SCOPE));
credential.setSelectedAccount(GoogleSignIn.getLastSignedInAccount(context).getAccount());
return new PeopleService.Builder(AndroidHttp.newCompatibleTransport(), JacksonFactory.getDefaultInstance(), credential)
.setApplicationName(context.getString(R.string.app_name)).build();
}
public static Person getProfile() {
try {
return getService().people().get("people/me")
.setPersonFields("genders,birthdays,addresses")
.execute();
} catch (Exception e) {
Utils.handleException(e);
return null;
}
}
public static String getBirthday(Person person) {
try {
List<Birthday> birthdayList = person.getBirthdays();
if (birthdayList == null) return Utils.EMPTY_STRING;
Date date = null;
for (Birthday birthday : birthdayList) {
date = birthday.getDate();
if (date != null && date.size() >= 3) break;
else date = null;
}
if (date == null) return Utils.EMPTY_STRING;
Calendar calendar = Calendar.getInstance();
calendar.set(date.getYear(), date.getMonth() - 1, date.getDay());
return Utils.convertDateToString(calendar);
} catch (Exception e) {
Utils.handleException(e);
return Utils.EMPTY_STRING;
}
}
private static final String CITY_SUFFIX = " city";
public static android.location.Address getLocation(Person person) {
try {
List<Address> addressList = person.getAddresses();
if (addressList == null) return null;
String city = null;
for (Address add : addressList) {
city = add.getCity();
if (!TextUtils.isEmpty(city)) break;
}
if (TextUtils.isEmpty(city)) return null;
Geocoder geocoder = new Geocoder(BHApp.getContext());
List<android.location.Address> addresses = geocoder.getFromLocationName(city + CITY_SUFFIX, 1);
if (addresses == null || addresses.isEmpty()) return null;
return addresses.get(0);
} catch (Exception e) {
Utils.handleException(e);
return null;
}
}
public static String getGender(Person person) {
List<Gender> genders = person.getGenders();
if (genders == null || genders.isEmpty()) return null;
Gender gender = genders.get(0);
return String.valueOf(Enum.Gender.getEnumByValue(gender.getValue()).getId());
}
}
I hope it helps :)
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With