Checking Java Receipts and AppStore

I am trying to check the receipt of payment on the server side. I get {"status":21002, "exception":"java.lang.IllegalArgumentException"} in response

Here is the code:

 private final static String _sandboxUriStr = "https://sandbox.itunes.apple.com/verifyReceipt"; public static void processPayment(final String receipt) throws SystemException { final BASE64Encoder encoder = new BASE64Encoder(); final String receiptData = encoder.encode(receipt.getBytes()); final String jsonData = "{\"receipt-data\" : \"" + receiptData + "\"}"; System.out.println(receipt); System.out.println(jsonData); try { final URL url = new URL(_sandboxUriStr); final HttpURLConnection conn = (HttpsURLConnection) url.openConnection(); conn.setRequestMethod("POST"); conn.setDoOutput(true); final OutputStreamWriter wr = new OutputStreamWriter(conn.getOutputStream()); wr.write(jsonData); wr.flush(); // Get the response final BufferedReader rd = new BufferedReader(new InputStreamReader(conn.getInputStream())); String line; while ((line = rd.readLine()) != null) { System.out.println(line); } wr.close(); rd.close(); } catch (IOException e) { throw new SystemException("Error when trying to send request to '%s', %s", _sandboxUriStr, e.getMessage()); } } 

My receipt is as follows:

 {\n\t"signature" = "[exactly_1320_characters]";\n\t"purchase-info" = "[exactly_868_characters]";\n\t"environment" = "Sandbox";\n\t"pod" = "100";\n\t"signing-status" = "0";\n} 

Check data with a BASE64 encoded receipt is as follows:

 {"receipt-data" : "[Block_of_chars_76x40+44=3084_chars_total]"} 

Does anyone have an idea or sample code, how can I get from the line to receive the JSON response mentioned here ?

+6
source share
3 answers

21002: The problem was in Base64 encoding inside Java. When I do the encoding inside IOS and use it as a request from the server without any encoding in Java, then it worked.

 switch (status) { case 21000: msg = "The App Store could not read the JSON object you provided"; logger.info("\n 21000 : The App Store could not read the JSON object you provided. "); break; case 21002: msg = "The data in the receipt-data property was malformed."; logger.info("\n 21002 : The data in the receipt-data property was malformed.. "); break; case 21003: msg = "The data in the receipt-data property was malformed."; logger.info("\n 21003 : The receipt could not be authenticated. "); break; case 21004: msg = "TThe shared secret you provided does not match the shared secret on file for your account."; logger.info("\n 21004 : The shared secret you provided does not match the shared secret on file for your account. "); break; case 21005: msg = "The receipt server is not currently available."; logger.info("\n 21005 : The receipt server is not currently available. "); break; case 21006: msg = "This receipt is valid but the subscription has expired. When this status code is returned to your server, the receipt data is also decoded and returned as part of the response."; logger.info("\n 21006 : This receipt is valid but the subscription has expired. When this status code is returned to your server, the receipt data is also decoded and returned as part of the response. "); break; case 21007: msg = "This receipt is a sandbox receipt, but it was sent to the production service for verification."; logger.info("\n 21007 : This receipt is a sandbox receipt, but it was sent to the production service for verification. "); break; case 21008: msg = "This receipt is a production receipt, but it was sent to the sandbox service for verification."; logger.info("\n 21008 : This receipt is a production receipt, but it was sent to the sandbox service for verification. "); break; default: msg = "Active subscription."; logger.info("\n 0 : valid ....Active subscription. "); break; } 
+1
source
 CloseableHttpClient client = HttpClients.createDefault(); JSONObject requestData = new JSONObject(); requestData.put("receipt-data", recept); requestData.put("password", password); HttpPost httpPost = new HttpPost("https://sandbox.itunes.apple.com/verifyReceipt"); StringEntity entity = new StringEntity(requestData.toString()); httpPost.setEntity(entity); httpPost.setHeader("Content-type", "application/x-www-form-urlencoded"); CloseableHttpResponse response = client.execute(httpPost); BufferedReader rd = new BufferedReader(new InputStreamReader(response.getEntity().getContent())); StringBuffer result = new StringBuffer(); String line = ""; while ((line = rd.readLine()) != null) { result.append(line); } System.out.println(result.toString()); response.close(); 
+1
source

I am not familiar with this service, but I have seen similar errors with other services when the content type or accepts headers where they are not set accordingly.

Try something like

  con.setRequestProperty("Content-Type", "application/json"); con.setRequestProperty("Accept", "application/json"); 

(Or what they expect. I guess json)

0
source

Source: https://habr.com/ru/post/925445/


All Articles