我終於畢業啦(撒花),不過玩咖的路上是沒有停歇的,偶然在公司工程師群組看到一則的新聞:
Google上周五(6/9)釋出了支援Android平台的reCAPTCHA API,此一API採用的是Google甫於今年3月發表的隱形reCAPTCHA技術,可於背景辨識用戶是真人或機器人,去除用戶使用行動程式的障礙。
新聞來源:http://www.ithome.com.tw/news/114878
面白い(有趣)!我們就來玩玩看吧,於是找到了 reCAPTCHA 的官方文件:
SafetyNet reCAPTCHA API Android Developers
【!注意!】在開始以前,這邊有個小地方需要注意,為了能夠正常使用這個 API,你的應用程式必須在 AndroidManifest.xml
文件中將 minSdkVersion
設置為 14
或 更高
。
單純看注意,一定沒有人會聽的,所以我在這邊必須再次說明,你必須在 AndroidManifest.xml
當中加入 <uses-sdk android:minSdkVersion="14" />
,否則 reCAPTCHA 會完全失效!完全沒有用!完全沒有畫面,下面是我的 AndroidManifest.xml
配置:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.kumagaikanta.safetynetrecaptcha">
<uses-sdk android:minSdkVersion="14" /> <uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme">
<meta-data android:name="com.google.android.gms.version" android:value="@integer/google_play_services_version" />
<activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application>
</manifest>
|
至於附加服務條款什麼的 ... 畢竟是使用 Google 的服務,當你使用這的同時,就代表你同意 Google 的相關服務條款囉!好的不管他,我們得重點在如何應用。
- 您必須先到 reCAPTCHA 註冊一個 reCAPTCHA 的金鑰。
- Label: 指鑰匙的唯一標籤,通常是您使用公司或組織的名稱。
- Package Names: 應用程式的 Package 名稱,為了讓應用程式使用 API,您輸入的 Package 名稱必須與該應用的 Package 名稱完全匹配。
- 註冊完畢後,您會獲得
Site key
與 Secret key
,請把他們記下來,待會會使用到。
- 接著我們回到 Android Studio 專案當中,畢竟這是 Android 以外的服務,所以我們必須在
build.gradle
當中依賴我們需要的套件包到專案當中:
1 2 3 4 5 6 7 8
| apply plugin: 'com.android.application' ... dependencies { compile 'com.google.android.gms:play-services-base:11.0.0' compile 'com.google.android.gms:play-services-basement:11.0.0' compile 'com.google.android.gms:play-services-safetynet:11.0.0' compile 'com.google.android.gms:play-services-tasks:11.0.0' }
|
PS: 如果出現錯誤訊息,那很正常的!因為你的 IDE 應該還沒有 Install 這些套件包,只需要 Install 就可以囉,這些動作 Android Studio 都會很智慧的幫你做完。
- SafetyNet API 是 Google Play 所提供的服務之一,所以如果要連接到 API,您需要創建一個 GoogleApiClient 類的實例,創建與Google Play 服務的連接後,您可以使用 Google API Client 連接到 SafetyNet API。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| import com.google.android.gms.common.api.GoogleApiClient; import com.google.android.gms.safetynet.SafetyNet;
public class MainActivity extends AppCompatActivity {
private GoogleApiClient mGoogleApiClient;
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main);
mGoogleApiClient = new GoogleApiClient.Builder(this) .addApi(SafetyNet.API) .addConnectionCallbacks(this) .addOnConnectionFailedListener(this) .build();
mGoogleApiClient.connect(); } }
|
- 因為 SafetyNet 會連線至 Google API,所以我們必須實作
GoogleApiClient.ConnectionCallbacks
與 GoogleApiClient.OnConnectionFailedListener
介面,這邊參考了 這篇文章 的做法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
| import com.google.android.gms.common.ConnectionResult; import com.google.android.gms.common.api.GoogleApiClient; import com.google.android.gms.safetynet.SafetyNet;
public class MainActivity extends AppCompatActivity implements GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener {
private GoogleApiClient mGoogleApiClient;
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main);
mGoogleApiClient = new GoogleApiClient.Builder(this) .addApi(SafetyNet.API) .addConnectionCallbacks(this) .addOnConnectionFailedListener(this) .build();
mGoogleApiClient.connect(); }
@Override public void onConnected(Bundle bundle) { }
@Override public void onConnectionSuspended(int i) { }
@Override public void onConnectionFailed(ConnectionResult connectionResult) { } }
|
- 為了更好 Debug,我們在實作的實例當中加入一些 Toast 與 Log 來方便我們除錯:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| @Override public void onConnected(Bundle bundle) { Toast.makeText(this, "已經連線到 Google Services", Toast.LENGTH_LONG).show(); Log.d("onConnected", "已經連線到 Google Services"); }
@Override public void onConnectionSuspended(int i) { Toast.makeText(this, "Google Services 連線中斷,連線中斷代號 " + i, Toast.LENGTH_LONG).show(); Log.d("onConnectionSuspended", "Google Services 連線中斷,連線中斷代號 " + i); }
@Override public void onConnectionFailed(ConnectionResult connectionResult) { Toast.makeText(this, "Google Services 連線失敗," + connectionResult.getErrorMessage(), Toast.LENGTH_LONG).show(); Log.w("onConnectionFailed", "Google Services 連線失敗," + connectionResult.getErrorMessage()); }
|
- 再來我們希望說,點了某顆按鈕之後,reCAPTCHA 來幫我們驗證是不是機器人,所以請在 View 上拉一個 Button,並且賦予 onClick 事件:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
| public void ButtonOnClikc(View view) { final TextView tvResult = (TextView) findViewById(R.id.textView); Toast.makeText(this, "Button on Clikc!", Toast.LENGTH_LONG).show(); SafetyNet.SafetyNetApi.verifyWithRecaptcha(mGoogleApiClient, "YOUR_SITE_KEY") .setResultCallback(new ResultCallback<SafetyNetApi.RecaptchaTokenResult>() { @Override public void onResult(final SafetyNetApi.RecaptchaTokenResult result) { final Status status = result.getStatus(); runOnUiThread(new Runnable() { @Override public void run() { if ((status != null) && status.isSuccess()) { tvResult.setText("isSuccess()\n");
if (!result.getTokenResult().isEmpty()) { tvResult.append("!result.getTokenResult().isEmpty()"); } else { tvResult.append("result.getTokenResult().isEmpty()"); } } else { Log.e("MY_APP_TAG", "Error occurred " + "when communicating with the reCAPTCHA service."); tvResult.setText("Error occurred " + "when communicating with the reCAPTCHA service."); } } }); } }); }
|
成功連線到 Google Services
啟用 reCAPTCHA 的樣子
驗證成功狀態
防堵機器人驗證
驗證失敗狀態