1. 程式人生 > >接入阿里雲簡訊,OSS,實人認證

接入阿里雲簡訊,OSS,實人認證

公司開發用到了很多阿里雲的產品,幾乎第三方SDK,雲伺服器什麼的老闆都用阿里雲的。

一:發簡訊功能

引入pom依賴

</dependency> 
        <dependency>
		 <groupId>com.aliyun</groupId>
		 <artifactId>aliyun-java-sdk-core</artifactId>
		 <version>3.2.8</version>
</dependency>
<dependency>
		<groupId>com.aliyun</groupId>
		<artifactId>aliyun-java-sdk-dysmsapi</artifactId>
		<version>1.1.0</version>
</dependency>

這個直接寫了一個工具類

import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsRequest;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.exceptions.ServerException;
import com.aliyuncs.http.MethodType;
import com.aliyuncs.profile.DefaultProfile;
import com.aliyuncs.profile.IClientProfile;

public class SendPhoneCode {
	public void sendCode(String telephone,String randomValidateCode,String smsParameters) throws ServerException, ClientException {
		final String product = "Dysmsapi";
		final String domain = "dysmsapi.aliyuncs.com";
		final String accessKeyId = "阿里雲accessKeyId";
		final String accessKeySecret = "阿里雲accessKeySecret";
		IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", accessKeyId, accessKeySecret);
		DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", product, domain);
		IAcsClient acsClient = new DefaultAcsClient(profile);
		SendSmsRequest request = new SendSmsRequest();
		request.setMethod(MethodType.POST);
		request.setPhoneNumbers(telephone);//手機號
		request.setSignName("XX網路");//簡訊名
		request.setTemplateCode(smsParameters);
		request.setTemplateParam("{\"code\":" + randomValidateCode + "}");
		SendSmsResponse sendSmsResponse = acsClient.getAcsResponse(request);
		if (sendSmsResponse.getCode() != null && sendSmsResponse.getCode().equals("OK")) {
			System.out.println("傳送成功~");
		} else {
			System.out.println("傳送失敗~");
		}
	}
}

然後根據業務需要在需要傳送簡訊驗證碼的地方直接呼叫這個工具類即可。

二:上傳圖片到阿里雲OSS

引入pom依賴

<dependency>
	    <groupId>com.aliyun.oss</groupId>
	    <artifactId>aliyun-sdk-oss</artifactId>
	    <version>2.8.2</version>
	</dependency>

封裝上傳圖片工具類

import java.io.File;
import java.util.Date;

import com.aliyun.oss.OSSClient;


public class UploadSample {

	public static String uploadImage(String packageName,File file,String name){        
    	// Endpoint以杭州為例,其它Region請按實際情況填寫。
    	String endpoint = "oss-cn-beijing.aliyuncs.com";
    	// 阿里雲主賬號AccessKey擁有所有API的訪問許可權,風險很高。強烈建議您建立並使用RAM賬號進行API訪問或日常運維,請登入 https://ram.console.aliyun.com 建立RAM賬號。
    	String accessKeyId = "阿里雲accessKeyId";
    	String accessKeySecret = "阿里雲accessKeySecret";
    	// 建立OSSClient例項。
    	OSSClient ossClient = new OSSClient(endpoint, accessKeyId, accessKeySecret);
    	// 上傳檔案。
    	ossClient.putObject("aliyunosslaowu", packageName+"/"+ name+"", file);
    	// 關閉Client。
    	ossClient.shutdown();
    	
    	Date expiration = new Date(System.currentTimeMillis() + 3600L * 1000 * 24 * 365 * 10);
        String url = ossClient.generatePresignedUrl("aliyunosslaowu", packageName+"/"+ name+"", expiration).toString();
        return url;
	}
}

然後在業務需要的地方直接使用工具類傳入需要的引數即可。

三:實人認證

引入依賴

<dependency>
	    <groupId>com.aliyun</groupId>
	    <artifactId>aliyun-java-sdk-cloudauth</artifactId>
	    <version>1.1.2</version>
</dependency>

實人認證分兩步:

第一步 獲取實人認證token

使用者在APP端輸入身份證和姓名然後呼叫後臺的獲取實人認證token介面,後臺在校驗完身份證沒重複註冊和格式正確後呼叫阿里雲介面獲取實人認證token返回給APP端。

先是校驗身份證是否合法的工具類

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;

public class VerifyIdNumber {

	private static String cityCode[] = { "11", "12", "13", "14", "15", "21",
			"22", "23", "31", "32", "33", "34", "35", "36", "37", "41", "42",
			"43", "44", "45", "46", "50", "51", "52", "53", "54", "61", "62",
			"63", "64", "65", "71", "81", "82", "91" };
 
	/**
	 * 每位加權因子
	 */
	private static int power[] = { 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5,
			8, 4, 2 }; 
	/**
	 * 驗證所有的身份證的合法性
	 * 
	 * @param idcard
	 *            身份證
	 * @return 合法返回true,否則返回false
	 */
	public static boolean isValidatedAllIdcard(String idcard) {
		if (idcard == null || "".equals(idcard)) {
			return false;
		}
		if (idcard.length() == 15) {
			return validate15IDCard(idcard);
		}
		return validate18Idcard(idcard);
	} 
	/**
	 * <p>
	 * 判斷18位身份證的合法性
	 * </p>
	 * 根據〖中華人民共和國國家標準GB11643-1999〗中有關公民身份號碼的規定,公民身份號碼是特徵組合碼,由十七位數字本體碼和一位數字校驗碼組成。
	 * 排列順序從左至右依次為:六位數字地址碼,八位數字出生日期碼,三位數字順序碼和一位數字校驗碼。
	 * <p>
	 * 順序碼: 表示在同一地址碼所標識的區域範圍內,對同年、同月、同 日出生的人編定的順序號,順序碼的奇數分配給男性,偶數分配 給女性。
	 * </p>
	 * <p>
	 * 1.前1、2位數字表示:所在省份的程式碼; 2.第3、4位數字表示:所在城市的程式碼; 3.第5、6位數字表示:所在區縣的程式碼;
	 * 4.第7~14位數字表示:出生年、月、日; 5.第15、16位數字表示:所在地的派出所的程式碼;
	 * 6.第17位數字表示性別:奇數表示男性,偶數表示女性;
	 * 7.第18位數字是校檢碼:也有的說是個人資訊碼,一般是隨計算機的隨機產生,用來檢驗身份證的正確性。校檢碼可以是0~9的數字,有時也用x表示。
	 * </p>
	 * <p>
	 * 第十八位數字(校驗碼)的計算方法為: 1.將前面的身份證號碼17位數分別乘以不同的係數。從第一位到第十七位的係數分別為:7 9 10 5 8 4
	 * 2 1 6 3 7 9 10 5 8 4 2
	 * </p>
	 * <p>
	 * 2.將這17位數字和係數相乘的結果相加。
	 * </p>
	 * <p>
	 * 3.用加出來和除以11,看餘數是多少
	 * </p>
	 * 4.餘數只可能有0 1 2 3 4 5 6 7 8 9 10這11個數字。其分別對應的最後一位身份證的號碼為1 0 X 9 8 7 6 5 4 3
	 * 2。
	 * <p>
	 * 5.通過上面得知如果餘數是2,就會在身份證的第18位數字上出現羅馬數字的Ⅹ。如果餘數是10,身份證的最後一位號碼就是2。
	 * </p>
	 * 
	 * @param idcard
	 * @return
	 */
	public static boolean validate18Idcard(String idcard) {
		if (idcard == null) {
			return false;
		}
 
		// 非18位為假
		if (idcard.length() != 18) {
			return false;
		}
		// 獲取前17位
		String idcard17 = idcard.substring(0, 17);
 
		// 前17位全部為數字
		if (!isDigital(idcard17)) {
			return false;
		}
 
		String provinceid = idcard.substring(0, 2);
		// 校驗省份
		if (!checkProvinceid(provinceid)) {
			return false;
		} 
		// 校驗出生日期
		String birthday = idcard.substring(6, 14); 
		SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd");
 
		try {
			Date birthDate = sdf.parse(birthday);
			String tmpDate = sdf.format(birthDate);
			if (!tmpDate.equals(birthday)) {// 出生年月日不正確
				return false;
			}
 
		} catch (ParseException e1) {
 
			return false;
		} 
		// 獲取第18位
		String idcard18Code = idcard.substring(17, 18); 
		char c[] = idcard17.toCharArray(); 
		int bit[] = converCharToInt(c); 
		int sum17 = 0; 
		sum17 = getPowerSum(bit);
 
		// 將和值與11取模得到餘數進行校驗碼判斷
		String checkCode = getCheckCodeBySum(sum17);
		if (null == checkCode) {
			return false;
		}
		// 將身份證的第18位與算出來的校碼進行匹配,不相等就為假
		if (!idcard18Code.equalsIgnoreCase(checkCode)) {
			return false;
		} 
		return true;
	} 
	/**
	 * 校驗15位身份證
	 * 
	 * <pre>
	 * 只校驗省份和出生年月日
	 * </pre>
	 * 
	 * @param idcard
	 * @return
	 */
	public static boolean validate15IDCard(String idcard) {
		if (idcard == null) {
			return false;
		}
		// 非15位為假
		if (idcard.length() != 15) {
			return false;
		} 
		// 15全部為數字
		if (!isDigital(idcard)) {
			return false;
		} 
		String provinceid = idcard.substring(0, 2);
		// 校驗省份
		if (!checkProvinceid(provinceid)) {
			return false;
		} 
		String birthday = idcard.substring(6, 12); 
		SimpleDateFormat sdf = new SimpleDateFormat("yyMMdd"); 
		try {
			Date birthDate = sdf.parse(birthday);
			String tmpDate = sdf.format(birthDate);
			if (!tmpDate.equals(birthday)) {// 身份證日期錯誤
				return false;
			}
 
		} catch (ParseException e1) { 
			return false;
		} 
		return true;
	} 
	/**
	 * 將15位的身份證轉成18位身份證
	 * 
	 * @param idcard
	 * @return
	 */
	public static String convertIdcarBy15bit(String idcard) {
		if (idcard == null) {
			return null;
		} 
		// 非15位身份證
		if (idcard.length() != 15) {
			return null;
		} 
		// 15全部為數字
		if (!isDigital(idcard)) {
			return null;
		} 
		String provinceid = idcard.substring(0, 2);
		// 校驗省份
		if (!checkProvinceid(provinceid)) {
			return null;
		} 
		String birthday = idcard.substring(6, 12); 
		SimpleDateFormat sdf = new SimpleDateFormat("yyMMdd"); 
		Date birthdate = null;
		try {
			birthdate = sdf.parse(birthday);
			String tmpDate = sdf.format(birthdate);
			if (!tmpDate.equals(birthday)) {// 身份證日期錯誤
				return null;
			} 
		} catch (ParseException e1) {
			return null;
		} 
		Calendar cday = Calendar.getInstance();
		cday.setTime(birthdate);
		String year = String.valueOf(cday.get(Calendar.YEAR)); 
		String idcard17 = idcard.substring(0, 6) + year + idcard.substring(8);
 
		char c[] = idcard17.toCharArray();
		String checkCode = ""; 
		// 將字元陣列轉為整型陣列
		int bit[] = converCharToInt(c); 
		int sum17 = 0;
		sum17 = getPowerSum(bit); 
		// 獲取和值與11取模得到餘數進行校驗碼
		checkCode = getCheckCodeBySum(sum17); 
		// 獲取不到校驗位
		if (null == checkCode) {
			return null;
		}
		// 將前17位與第18位校驗碼拼接
		idcard17 += checkCode;
		return idcard17;
	} 
	/**
	 * 校驗省份
	 * 
	 * @param provinceid
	 * @return 合法返回TRUE,否則返回FALSE
	 */
	private static boolean checkProvinceid(String provinceid) {
		for (String id : cityCode) {
			if (id.equals(provinceid)) {
				return true;
			}
		}
		return false;
	} 
	/**
	 * 數字驗證
	 * 
	 * @param str
	 * @return
	 */
	private static boolean isDigital(String str) {
		return str.matches("^[0-9]*$");
	} 
	/**
	 * 將身份證的每位和對應位的加權因子相乘之後,再得到和值
	 * 
	 * @param bit
	 * @return
	 */
	private static int getPowerSum(int[] bit) {
 
		int sum = 0;
 
		if (power.length != bit.length) {
			return sum;
		}
 
		for (int i = 0; i < bit.length; i++) {
			for (int j = 0; j < power.length; j++) {
				if (i == j) {
					sum = sum + bit[i] * power[j];
				}
			}
		}
		return sum;
	} 
	/**
	 * 將和值與11取模得到餘數進行校驗碼判斷
	 * 
	 * @param checkCode
	 * @param sum17
	 * @return 校驗位
	 */
	private static String getCheckCodeBySum(int sum17) {
		String checkCode = null;
		switch (sum17 % 11) {
		case 10:
			checkCode = "2";
			break;
		case 9:
			checkCode = "3";
			break;
		case 8:
			checkCode = "4";
			break;
		case 7:
			checkCode = "5";
			break;
		case 6:
			checkCode = "6";
			break;
		case 5:
			checkCode = "7";
			break;
		case 4:
			checkCode = "8";
			break;
		case 3:
			checkCode = "9";
			break;
		case 2:
			checkCode = "x";
			break;
		case 1:
			checkCode = "0";
			break;
		case 0:
			checkCode = "1";
			break;
		default:
			break;
		}
		return checkCode;
	}
 
	/**
	 * 將字元陣列轉為整型陣列
	 * 
	 * @param c
	 * @return
	 * @throws NumberFormatException
	 */
	private static int[] converCharToInt(char[] c) throws NumberFormatException {
		int[] a = new int[c.length];
		int k = 0;
		for (char temp : c) {
			a[k++] = Integer.parseInt(String.valueOf(temp));
		}
		return a;
	}
}

然後是將身份證和姓名傳入阿里雲獲取實人認證token的工具類

import java.util.UUID;

import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.cloudauth.model.v20180504.GetVerifyTokenRequest;
import com.aliyuncs.cloudauth.model.v20180504.GetVerifyTokenResponse;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.exceptions.ServerException;
import com.aliyuncs.profile.DefaultProfile;

import net.sf.json.JSONObject;

public class RPBasic {

	private static String biz = "vortex";//您在控制檯上建立的、採用RPBioID認證方案的認證場景標識, 建立方法:https://help.aliyun.com/document_detail/59975.html
	public static JSONObject getToken(String realName,String idCardNo){ 
		DefaultProfile profile = DefaultProfile.getProfile(
		        "cn-hangzhou",             //預設
		        "AccessKeyID",         //您的Access Key ID
		        "AccessKeySecret");    //您的Access Key Secret
		IAcsClient client = new DefaultAcsClient(profile);
		String ticketId = UUID.randomUUID().toString(); //認證ID, 由使用方指定, 發起不同的認證任務需要更換不同的認證ID
		String token = null; //認證token, 表達一次認證會話
		//int statusCode = -1; //-1 未認證, 0 認證中, 1 認證通過, 2 認證不通過
		//1. 服務端發起認證請求, 獲取到token
		//GetVerifyToken介面文件:https://help.aliyun.com/document_detail/57050.html
		GetVerifyTokenRequest getVerifyTokenRequest = new GetVerifyTokenRequest();
		getVerifyTokenRequest.setBiz(biz);
		getVerifyTokenRequest.setTicketId(ticketId);
		//若需要binding圖片(如身份證正反面等), 且使用base64上傳, 需要設定請求方法為POST
		//getVerifyTokenRequest.setMethod(MethodType.POST);
		//通過binding引數傳入業務已經採集的認證資料,其中姓名、身份證號為必要欄位
		//若需要binding圖片資料,請控制單張圖片大小在 2M 內,避免拉取超時
		getVerifyTokenRequest.setBinding("{\"Name\": \"" + realName +"\",\"IdentificationNumber\": \""+ idCardNo+"\"}");
		try {
		    GetVerifyTokenResponse response = client.getAcsResponse(getVerifyTokenRequest);
		    System.out.println(response.getSuccess());
		    token = response.getData().getVerifyToken().getToken(); //token預設30分鐘時效,每次發起認證時都必須實時獲取
		    System.out.println(token);
		} catch (ServerException e) {
		    e.printStackTrace();
		} catch (ClientException e) {
		    e.printStackTrace();
		}
		JSONObject jSONObject = new JSONObject();
		jSONObject.put("realName", realName);
		jSONObject.put("idCardNo", idCardNo);
		jSONObject.put("ticketId", ticketId);
		jSONObject.put("token", token);		
		return jSONObject;
	}
}

第二步 使用者認證完APP再調一下同步資料的介面

// 先判斷是否完善,再進行實名認證介面驗證
				int statusCode = -1;
				String biz = "vortex";
				GetStatusRequest getStatusRequest = new GetStatusRequest();
				getStatusRequest.setBiz(biz);
				getStatusRequest.setTicketId(ticketId);
				DefaultProfile profile = DefaultProfile.getProfile("cn-hangzhou", // 預設
						"AccessKeyID", // 您的Access Key ID
						"AccessKeySecret"); // 您的Access Key Secret
				IAcsClient client = new DefaultAcsClient(profile);
				try {
					GetStatusResponse response = client.getAcsResponse(getStatusRequest);
					statusCode = response.getData().getStatusCode();
				} catch (ServerException e) {
					e.printStackTrace();
				} catch (ClientException e) {
					e.printStackTrace();
				}
				// 7. 服務端獲取認證資料
				// GetMaterials介面文件:https://help.aliyun.com/document_detail/57641.html
				GetMaterialsRequest getMaterialsRequest = new GetMaterialsRequest();
				getMaterialsRequest.setBiz(biz);
				getMaterialsRequest.setTicketId(ticketId);
				if (1 == statusCode) {// 認證通過
					try {
						GetMaterialsResponse response = client.getAcsResponse(getMaterialsRequest);
						UserInfo dbuser = userInfoService.selectUserById(userId);

						// 更新使用者資料
						dbuser.setRealName(response.getData().getName());
						dbuser.setIdCardNo(response.getData().getIdentificationNumber());
						if (response.getData().getSex().equals("m")) {
							dbuser.setSex(1);
						} else {
							dbuser.setSex(2);
						}
						//更新使用者認證資料到資料庫,這段程式碼省略
						
					} catch (Exception e) {
						e.printStackTrace();
					}
				}

到此實人認證就寫好了。