'nodejs'에 해당되는 글 2건

  1. 2019.07.14 nodejs를 이용한 paypal 결제 연동하기 (구독) 1
  2. 2019.04.09 Android 결제 서버 개발

이 페이지는 express에 대한 설명을 하지 않습니다. 각 REST API에 대한 설명을 표시합니다.

 

[주요 사이트]

페이팔 개발자 사이트 : developer.paypal.com .

 

[공식문서]

연동 페이지 : https://developer.paypal.com/docs/subscriptions/integrate .

 

[참고사이트]

cURL 설명 사이트 : https://www.lesstif.com/pages/viewpage.action?pageId=14745703 .
cURL 을 Nodejs 등으로 변환해 주는 사이트 : https://curl.trillworks.com/#node .

 

 

1. 개요.

. 모든 조작은 Access 토큰을 얻고, 이를 이용해 유저를 확인해, 데이터를 내어 줍니다.

. 실제 판매하는 상품인 Product 를 등록해야 합니다.

. Product를 어떤 식으로 구독할 것인지, 설정하는 Plan을 만들어야 합니다.

. 실제 구독 처리를 하는 Subscript를 만들어야 합니다.

. paypal은 라이브 서버와 센드박스 서버를 운영하고 있는데, 여기서는 실제 구매가 일어나지 않는 sendbox서버에서의 동작만을 테스트 합니다.

. 이 것을 실행하기 위해서는 프로젝트를 생성해, client_id와 이에 대한 secret을 획득해야 합니다.

. 페이팔은 입력값이 하나라도 틀리면, 500 번 오류등을 보낼 뿐 왜, 처리가 되었는지 하나의 값도 반환하지 않습니다. 즉 원인을 반환하지 않으므로, 잘 입력해야 합니다.

 

 

2. 에세스 토큰 얻기.

. access 토큰을 얻기 위한 cURL 예제.

curl -v https://api.sandbox.paypal.com/v1/oauth2/token \
   -H "Accept: application/json" \
   -H "Accept-Language: en_US" \
   -u "client_id:secret" \
   -d "grant_type=client_credentials"

각 옵션 설명.

-H : 헤더.
-d : 포스트 데이터.
-u : 유저 아이디.

유저 아이디에 들어 갈값은 각각 생성한 프로젝트의 client_id와 secret을 넣어 줍니다.

uURL을 한줄로 만든 소스. (시크릿 값을 바꿔 놓아 동작하지 않습니다. 예시일 뿐입니다.)

curl -v https://api.sandbox.paypal.com/v1/oauth2/token -H "Accept: application/json" -H "Accept-Language: en_US" -u "AVPXMdI_2MFO__tJGV71Ve-YCQko8v1iIg4u7XEm24f1N3XAKQBjLzdGrzFEn4-dkpiwzxtdLTCjKKZM:EEv2R_AKdfni2FfPgO2YCt4RhScM7E8tkqbX7PDJB2HaoMno3bK4xvtny3" -d "grant_type=client_credentials"

 

 

3. 상품관련 REST API.

1. 상품 생성하기.

. 상품을 생성하기 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/catalogs/products \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token" \
  -H "PayPal-Request-Id: merchant-generated-ID \ // Optional and if passed, helps identify idempotent requests
-d '{
  "name": "Video Streaming Service",
  "description": "Video streaming service",
  "type": "SERVICE",
  "category": "SOFTWARE",
  "image_url": "https://example.com/streaming.jpg",
  "home_url": "https://example.com/home"
}'

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/catalogs/products -H "Content-Type: application/json" -H "Authorization: Bearer A21AAGncecjv_2d8uNxHxtU3LUzi3I7uHUS1Trj7syi9Fzmwx8bkx0KuH1tEFXS7t_nX8AoXtBGxBi5-xwYdUwt63XSYgX9Ew" -d '{  "name": "Video Streaming Service", "description": "Video streaming service", "type": "SERVICE", "category": "SOFTWARE", "image_url": "https://example.com/streaming.jpg", "home_url": "https://example.com/home" }'

.  주의 할점.

결과 코드가 201번이 OK 입니다. 200번 성공으로 검사를 하다가 낭패를 볼 수 있습니다. (저는 한 시간 헤맷음).

 

2. 상품 정보 표시.

cURL에서 products 다음에는 상품의 아이디를 입력해 줍니다.

. 상품 정보를 가지고 오기 위한 cURL 예제.

curl -v -X GET https://api.sandbox.paypal.com/v1/catalogs/products/72255d4849af8ed6e0df1173 \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X GET https://api.sandbox.paypal.com/v1/catalogs/products/PROD-2UH90666Y0969991W -H "Content-Type: application/json" -H "Authorization: Bearer A21AAFI0ZSRjDDHfhFYGGgcVgkU9BP1aJ242GlqfWIEcf-aWLrddTJbNJF6lSwNpzH4JDzaLEbQ136ah9SHZ0WwsLGlLBAPdQ"

 

 

3. 상품 업데이트.

. 상품 정보를 업데이트 하기 위한 cURL 예제.

curl -v -X PATCH https://api.sandbox.paypal.com/v1/catalogs/products/72255d4849af8ed6e0df1173 \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token" \
-d '[
  {
    "op": "replace",
    "path": "/description",
    "value": "Premium video streaming service"
  }
]'

. cURL 소스를 한줄로 만든 문자열.

curl -v -X PATCH https://api.sandbox.paypal.com/v1/catalogs/products/PROD-9US56197H5602373L -H "Content-Type: application/json" -H "Authorization: Bearer A21AAHuy8FMsUdRiNLznO2FWYCrlykkjrcCDkXYyq9Q-YErZBF4Y4UPD37JLL2ZBMwc0DkibctYFxAMIBkh1KQOvhq5Ki52sA" -d '[{"op": "replace","path": "/description","value": "Premium video streaming service no"}]'

주의 할점.

결과 코드가 204번이 OK 입니다. 200번으로 성공 검사를 하다가 낭패를 볼 수 있습니다.

 

 

4. 상품 리스트.

. 상품 리스트를 가지고 오기 위한 cURL 예제.

curl -v -X GET https://api.sandbox.paypal.com/v1/catalogs/products?page_size=2&page=1&total_required=true \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X GET https://api.sandbox.paypal.com/v1/catalogs/products?page_size=2&page=1&total_required=true -H "Content-Type: application/json" -H "Authorization: Bearer A21AAEYDcoQtmrOta05Wj6Au2OmTZz90qqGIvHjUnA-iWAJ0y56XW-7gNgrghvxGcfu1PCqxAwPua6x4ajE47EGuznS0u2JTw"

 

 

4. 플랜 관련 REST API.

1. 플랜 생성.

. 플랜을 생성하기 위한 cURL 예제.

옵션인 플랜 아이디는 따로 주지 않아도 됩니다.

curl -v -k -X POST https://api.sandbox.paypal.com/v1/billing/plans \
  -H "Accept: application/json" \
  -H "Authorization: Bearer Access-Token" \
  -H "PayPal-Request-Id: PLAN-18062019-001" \  // merchant generated ID, optional and needed for idempotent samples
  -H "Prefer: return=representation" \
  -H "Content-Type: application/json" \
  -d '{
      "product_id": "PROD-6XB24663H4094933M",
      "name": "Basic Plan",
      "description": "Basic plan",
      "billing_cycles": [
        {
          "frequency": {
            "interval_unit": "MONTH",
            "interval_count": 1
          },
          "tenure_type": "TRIAL",
          "sequence": 1,
          "total_cycles": 1
        },
        {
          "frequency": {
            "interval_unit": "MONTH",
            "interval_count": 1
          },
          "tenure_type": "REGULAR",
          "sequence": 2,
          "total_cycles": 12,
          "pricing_scheme": {
            "fixed_price": {
              "value": "10",
              "currency_code": "USD"
            }
          }
        }
      ],
      "payment_preferences": {
        "auto_bill_outstanding": true,
        "setup_fee": {
          "value": "10",
          "currency_code": "USD"
        },
        "setup_fee_failure_action": "CONTINUE",
        "payment_failure_threshold": 3
      },
      "taxes": {
        "percentage": "10",
        "inclusive": false
      }
    }'

- 빌링 사이클 : 말 그대도 결제 사이클 입니다. ( 공식페이지 )
   + frequency : 한달에 한번, 혹은 일주일에 한 번, 혹은 1년에 한 번등을 설정하는 값 입니다.
   + tenure_type : TRIAL (테스트), REGULAR (정식), TRIAL은 트라이얼 결제를 통해 무료한달 등을 사용할 수 있게 하기 위함.
   + sequence : 결제 실행 순서 입니다. 위 예제에서는 트라이얼 결제가 먼저 일어 납니다.
   + total_cycles : 결제를 몇번 동안 계속 할 것인가 입니다. 위 예제에서는 1년동안 구독이 됩니다. (최대 99회)
   + pricing_scheme : 1회 결제시 가격 입니다. value 값. currency_code는 ( 코드 표 링크 )를 참조 합니다.

payment_preferences : 결제를 위한 설정입니다. (자동 구독을 위한 설정).
   + auto_bill_outstanding :  자동으로 다음 결제를 할 것인지 설정하는 값, 기본 (true)
   + set-fee : 첫 구독 시 결제 비용 입니다. (왜 따로 두었는지는 모르겠습니다.)
   + setup_fee_failure_action : 초기 결제가 실패한 경우, 동작 CANCEL (기본값) 은 결제 실패, CONTINUE는 재시도.
   + payment_failure_threshold : 실패시 재 시도 횟수.
   + taxes : 세금정보, percentage -> 세금비율, inclusive -> 금액에 세금이 들어 가 있는 지 여부 (default:true).

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans -H "Content-Type: application/json" -H "Authorization: Bearer A21AAG9M3f0j-aS8q4lgK4BPH-JRP0t-je8jL3e5yI8mybt7rgyBvh6F-6EwHBBgoJF5CQf3LEeC-lrCwJAuvy1O1pcm8A7GQ" -d '{"product_id": "PROD-2UH90666Y0969991W","name": "Video Streaming Service Plan","description": "Video Streaming Service basic plan","status":"ACTIVE","billing_cycles": [{"frequency":{"interval_unit":"MONTH","interval_count": 1},"tenure_type": "TRIAL","sequence":1,"total_cycles": 1,"pricing_scheme": {"fixed_price":{"value":"10","currency_code":"USD"}}},{"frequency":{"interval_unit":"MONTH","interval_count":1},"tenure_type": "REGULAR","sequence": 2,"total_cycles":12,"pricing_scheme":{"fixed_price":{"value": "100","currency_code":"USD"}}}],"payment_preferences": {"auto_bill_outstanding": true,"setup_fee":{"value": "10","currency_code": "USD"},"setup_fee_failure_action":"CONTINUE","payment_failure_threshold": 3},"taxes":{"percentage": "10","inclusive": false}}'

. 주의 할 점.

생성시 201 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

 

2. 유효한 플랜 리스트.

. 플랜리스트를 가지고 오기 위한 cURL 예제.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/plans?page_size=2&page=1&total_required=true \
-H "Content-Type: application/json" \
-H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/plans?page_size=2&page=1&total_required=true -H "Content-Type: application/json" -H "Authorization: Bearer A21AAHI_Flrh2uvWHQbILGIksjorab6ePahqPnKrTPNQYsWuS-DL-X5sJoIZe5c4k85vstm3eAdLUcV1BlXHoynwx9HYIt0jg"

 

 

3. 플랜 정보 표시.

. 플랜 정보 표시를 위한 cURL 예제.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/plans/P-2UF78835G6983425GLSM44MA \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/plans/P-6CT89946GV786361JLUVLPRQ -H "Content-Type: application/json" -H "Authorization: Bearer A21AAH5qauXi5CvpnecoUuKXMU9zrnJVzwPuxRO5kykkdqkkUZj_QbyuawB3ADMK_f16D3ExfbaBQfrmg51ZA3NwZzxEioYww"

 

 

4. 플랜 활성화.

. 플랜을 활성화 하기 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans/P-7GL4271244454362WXNWU5NQ/activate \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans/P-2GL72983FH4989143LUVNSWQ/activate -H "Content-Type: application/json" -H "Authorization: Bearer A21AAG3Er__iBWSK-E8eDuhkpiTzqeuZ9Cf3hkCUFKc4vldCBmVOZ4o4DWvONMjV8eHLTy558lzy_346TzVniHKdbPR1HXr5A"

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

 

5. 비활성화 플랜.

. 플랜을 비활성화 하기 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans/P-7GL4271244454362WXNWU5NQ/deactivate \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans/P-2GL72983FH4989143LUVNSWQ/deactivate -H "Content-Type: application/json" -H "Authorization: Bearer A21AAH5qauXi5CvpnecoUuKXMU9zrnJVzwPuxRO5kykkdqkkUZj_QbyuawB3ADMK_f16D3ExfbaBQfrmg51ZA3NwZzxEioYww"

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

 

6 플랜 업데이트.

* 플랜중 가격을 제외한 부분을 변경할 수 있으므며 아래와 같습니다.
   -  description, auto_bill_outstanding, texes.percentage, payment_preferences.payment_failure_threshold

. 플랜을 업데이트 하기 위한 cURL 예제.

curl -v -X PATCH https://api.sandbox.paypal.com/v1/billing/plans/P-7GL4271244454362WXNWU5NQ \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token" \
  -d '[
    {
      "op": "replace",
      "path": "/payment_preferences/payment_failure_threshold",
      "value": 7
    }
  ]'

. cURL 소스를 한줄로 만든 문자열.

curl -v -X PATCH https://api.sandbox.paypal.com/v1/billing/plans/P-6CT89946GV786361JLUVLPRQ -H "Content-Type: application/json" -H "Authorization: Bearer A21AAG0nNx3tfO_Lm4f3gQh6-hkfUYFLJgkMDJGAG3azztvPQFmkITxtdMiWuS8wypPf4Dclvv78Qpji9CqOrxHWjlapaGBkQ" -d '[{"op": "replace","path": "/payment_preferences/payment_failure_threshold","value": 7}]'

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

세금도 변경하실 분이 계실것 같은데, 기존에 10.0등 소수점 자리가 있으면, "" 로 묶어야 합니다. 아니면 오류 납니다.

 

 

7. 플랜 프라이싱.

* 플랜 중 빌링 사이클 (주로 가격 및 지불 방법)을 변경할 때 사용합니다.

. 플랜의 가격및 일정등을 조절 하기 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans/P-2UF78835G6983425GLSM44MA/update-pricing-schemes \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token" \
-d '{
  "pricing_schemes": [{
    "billing_cycle_sequence": 2,
    "pricing_scheme": {
      "fixed_price": {
        "value": "50",
        "currency_code": "USD"
        }
      }
    }
  ]
}'

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/plans/P-6CT89946GV786361JLUVLPRQ/update-pricing-schemes -H "Content-Type: application/json" -H "Authorization: Bearer A21AAHPEgYcmHBslHbSE8_8h93mFN892b8VdAmUN1QTkXSOZZKcTBTdJnNJDYBd5uR8OVB4EeNkO8kVtH4ETqF38R1LGE3GXA" -d '{"pricing_schemes": [{"billing_cycle_sequence": 2,"pricing_scheme": {"fixed_price": {"value": "50","currency_code": "USD"}}}]}'

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

원래가격의 20% 이상 변경한다면, 오류가 납니다.

가격 변경은 단 한 번만 할 수 있습니다.

 

 

 

5. 구독을 위한 REST API.

구독은 실제 사용자의 결제로 부터 시작되며, 이 것을 검증하고 활성화, 결제 내역 확인등의 작업을 처리하게 됩니다.

 

1. 구독 생성.

. 사용자 구독 신청을 하기 위한 웹페이지 소스

<body>
  <script
       src="https://www.paypal.com/sdk/js?client-id=SB_CLIENT_ID&vault=true">
  </script>

  <div id="paypal-button-container"></div>

  <script>
  
    paypal.Buttons({

      createSubscription: function(data, actions) {

        return actions.subscription.create({

          'plan_id': 'P-2UF78835G6983425GLSM44MA'

        });

      },


      onApprove: function(data, actions) {

        alert('You have successfully created subscription ' + data.subscriptionID);

      }


    }).render('#paypal-button-container');
    
  </script>
</body>

. 구독을 생성하기 위한 cURL 예제.

curl -v -k -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions \
   -H "Accept: application/json" \
   -H "Authorization: Bearer Access-Token" \
   -H "PayPal-Request-Id: SUBSCRIPTION-21092019-001" \
   -H "Prefer: return=representation" \
   -H "Content-Type: application/json" \
   -d '{
      "plan_id": "P-2UF78835G6983425GLSM44MA",
      "start_time": "2019-03-27T06:00:00Z",
      "subscriber": {
        "name": {
          "given_name": "John",
          "surname": "Doe"
        },
        "email_address": "customer@example.com"
      },
      "auto_renewal": true,
      "application_context": {
        "brand_name": "example",
        "locale": "en-US",
        "shipping_preference": "SET_PROVIDED_ADDRESS",
        "user_action": "SUBSCRIBE_NOW",
        "payment_method": {
          "payer_selected": "PAYPAL",
          "payee_preferred": "IMMEDIATE_PAYMENT_REQUIRED"
        },
        "return_url": "https://example.com/returnUrl",
        "cancel_url": "https://example.com/cancelUrl"
      }
    }'

plan_id : 구독 대상이되는 플랜의 아이디 입니다. (이 것만 입력하면됩니다.)
- start_time : 이 구독이 시작되는 시간. (입력하지않으면 현재시간; 페이팔 서버와 시차등을 계산해야 하므로, 입력하지 않을 것을 권장)
quantity : 구독을 몇개 했는지 여부 입니다. (대체로 1을 놓으면 된다라고 생각합니다.
shipping_amount : 배송비 입니다.
subscriber : 구독자의 정보 입니다.
auto_renewal : 자동으로 구매 처리를 할 것인가 여부.
application_context : 마켓 이라면, 어떤 마켓을 통해 구매 했는지 등을 남기면됩니다. (웹을 통한 경우, 웹이라 입력하면됨.).

. cURL 소스를 한줄로 만든 문자열.

curl -v -k -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions -H "Accept: application/json" -H "Authorization: Bearer A21AAF2sr1xlf9jswwJRqle3ncuX8GaEEPqcxtfW-a-HlkXCDt8ZTYlDrMvRlU0KbhFqFkjS8g5DdTAXa2lqZ-Gp3BvBqVlpA" -H "PayPal-Request-Id: I-5KL7FJ0N1RKW" -H "Prefer: return=representation" -H "Content-Type: application/json" -d '{"plan_id": "P-6CT89946GV786361JLUVLPRQ","start_time": "2019-03-27T06:00:00Z","subscriber": {"name": {"given_name": "John","surname": "Doe"},"email_address": "customer@example.com"},"auto_renewal": true,"application_context": {"brand_name": "example","locale": "en-US","shipping_preference": "SET_PROVIDED_ADDRESS","user_action":"SUBSCRIBE_NOW","payment_method": {"payer_selected": "PAYPAL","payee_preferred": "IMMEDIATE_PAYMENT_REQUIRED"},"return_url": "https://example.com/returnUrl","cancel_url": "https://example.com/cancelUrl"}}'

. 주의 할 점.

시간을 입력할 때는 javascript의 Date객체로 시간을 생성한뒤 toISOString()  라는 멤버 메소드를 이용해 값을 출력하면됩니다.

생성시 201 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

웹으로 구독을 신청하면, 구독이 이미 생성된 것 입니다. REST API로 하는 것은 소매점 오프라인을 위한 것으로 보입니다. 일단 저는 이부분만은 REST API로 해 보지 않았습니다. 웹에서 하면 이미 끝이니까요

 

2. 구독 자세히 보기.

. 구독의 상세 정보 확인을 위한 cURL 예제.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BY6XKDWF90W7 -H "Content-Type: application/json" -H "Authorization: Bearer A21AAFnNqv1rB6Zj0u6ayyNQtqqb3ZgkJBoIqng_cdUym_4uHHXyou9v9H7_JlhcnuzPDG03zzzs6n6c8r6w1ETBf1lKRr9AQ"

 

3. 구독 정보 업데이트.

. 구독 정보 업데이트를 위한 cURL 예제.

curl -v -X PATCH https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G \
-H "Content-Type: application/json" \
-H "Authorization: Bearer Access-Token" \
-d '[
  {
    "op": "replace",
    "path": "/billing_info/outstanding_balance",
    "value": {
      "currency_code": "USD",
      "value": "50.00"
    }
  }
]'

- 업데이트 하기 위해 할 수 있는 것은 아래 3가지 입니다.
   + subscripber.shipping_address : 배송 주소.
   + shipping_amount : 배송비
   + billing_info.outstanding_balance : 미 결제 잔액.

. cURL 소스를 한줄로 만든 문자열.

curl -v -X PATCH https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G -H "Content-Type: application/json" -H "Authorization: Bearer A21AAG1TgiPs8x7--4lmuFT-gnoUkhvnJs-yMRE3_4eeBY2L0IzI8X12V2cMHL9okRMThBt3p5Gd_9e37pFfMw4B1Wn8cYIcg" -d '[{"op": "replace","path": "/billing_info/outstanding_balance","value": {"currency_code": "USD","value": "50.00"}}]'

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

4. 구독 정지.

. 구독을 정지하기 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/suspend \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-LC86V1F6JKFW/suspend -H "Content-Type: application/json" -H "Authorization: Bearer A21AAH3b4Hfn5gOd3SKJK7aUpOWAy2hB_gPEhuq_SeURA4aqopM1m1NwV3-ECZwRLVmdT3ZrME2Qq6sicsO-Dfl3dGHrmGPhg"

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

5. 구독 취소.

. 구독 취소를 위한 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/cancel \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/cancel -H "Content-Type: application/json" -H "Authorization: Bearer Access-Token"

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

6. 구독 활성화

. 구독을 활성화 하기 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/activate \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/activate -H "Content-Type: application/json" -H "Authorization: Bearer Access-Token"

 

7. 미 결제 약정 잔액에 대한 지불.

* 이 것은 잔액이 없거나 하는 등의 이유로 지불해야 할 돈을 지불하지 않은 경우, 지불을 시도하는 옵션으로 보입니다.

. 미 결제 약정 잔액을 위한 cURL 예제.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/capture \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"
 -d '{
   "note": "Charging as the balance reached the limit",
   "capture_type": "OUTSTANDING_BALANCE",
   "amount": {
     "value": "100",
     "currency_code": "USD"
   }
 }'

. cURL 소스를 한줄로 만든 문자열.

curl -v -X POST https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/capture -H "Content-Type: application/json" -H "Authorization: Bearer Access-Token" -d '{"note": "Charging as the balance reached the limit","capture_type": "OUTSTANDING_BALANCE","amount": {"value": "100","currency_code": "USD"}}'

 

. 주의 할 점.

생성시 204 status 코드를 반환합니다. 200번으로 검사하시면 안됩니다.

 

8. 구독 결제 나열.

. 구독 결제 나열을하기 위한 cURL 예제.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/transactions?start_time=2018-01-21T07:50:20.940Z&end_time=2018-08-21T07:50:20.940Z \
  -H "Content-Type: application/json" \
  -H "Authorization: Bearer Access-Token"

- 시간을 입력할 때는 javascript의 Date객체로 시간을 생성한뒤 toISOString()  라는 멤버 메소드를 이용해 값을 출력하면됩니다.

. cURL 소스를 한줄로 만든 문자열.

curl -v -X GET https://api.sandbox.paypal.com/v1/billing/subscriptions/I-BW452GLLEP1G/transactions?start_time=2018-01-21T07:50:20.940Z&end_time=2018-08-21T07:50:20.940Z -H "Content-Type: application/json" -H "Authorization: Bearer Access-Token"

 

9. 구독 업그레이드 / 다운 그레이드.

. 구독 업그레이드 / 다운 그레이드는 시간 관계상 하지 않았습니다. 저도 사용하지 않기도 해서요.

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

Posted by 창업닉군
,

안드로이드 결제를 하면, 이를 검증하고 DB에 반영할 서버가 필요합니다. 이 node.js로 구현해 볼 생각 입니다. Android 개발자 사이트 공식 문서에는 결제 관련 데이터 접근을 Google Play Developer API로 하도록 하고 있습니다. 이 기능은 앱의 새 버전을 업로드하고 출시 하는 기능과 인앱 상품의 구매 이력을 관리할 수 있는 기능을 구현할 수 있습니다. 저는 이 중 인앱 상품 관련 API 를 이용해 볼 생각 입니다.

[관련 사이트]

구글 플레이 개발자 API : https://developers.google.com/android-publisher
구글 클라이언트 라이브러리 : https://developers.google.com/api-client-library/

 

[관련문서]

공식문서 : https://developer.android.com/google/play/developer-api

 

[선행작업]

출시용 APK 파일 만들기 : https://nicgoon.tistory.com/198
베타 버전으로 앱 등록하기 : https://nicgoon.tistory.com/199
안드로이드 클라이언트 인앱 결제 구현 : https://nicgoon.tistory.com/200

 

1. 참고 문서보기.

구글 플레이 개발자 API의 안내속 시작 하기 문서(링크) 속에는 구글 플레이 개발자 API기능에 대한 설명이 들어 있습니다. 여러분이 보는 시기에 이 문서가 어떻게 표시되는지 모르겠지만, 제가 보고 있는 문서는 한글로 번역이 되어 있습니다. 이 문서를 읽어 보면, 개발 진행사항에서 API를 통해, 개발 버전을 관리할 수 있고, 상품에 달려 있는 리플들을 가지고와 관리할 수 있는 기능, 구매 관련 기능이 있음을 설명하고 있습니다. 저는 이 중 구매 관련, (지정한 상품이 유효한지 여부)를 확인하는 기능을 구현할 생각 입니다. 이 기능을 이용해 유저가 앱에서 실제 구매 했는지 검증을 할 것 입니다.

 

2. 새 API 프로젝트 생성.

구글 플레이 콘솔에서, 가장 아래 톱니바퀴 모양의 설정 버튼을 눌러 줍니다.

설정 페이지가 나오면, 개발자 계정 > API 액세스 를 눌러 해당 페이지를 호출 합니다. 페이지가 나오면, 새 프로젝트를 눌러 API 프로젝트를 만들어 API 생성 페이지를 만들어 줍니다.

 

그렇게  API 프로젝트를 만들고 나면,  Google Play Android Develope, Game Service Publishing API 두개의 프로젝트가 설정되어 있습니다. 이 것은 Goolge Play Android Developer 만 연결된 상태로 되어 있는데 그대로 두면 됩니다. 

2. Autho 클라이언트 만들기.

결제등의 데이터를 확인할 때, 구글 플레이 서버는 서버라 칭하고, 구글 플레이 서버에 데이터를 요청하는 쪽을 클라이언트라고 합니다. 우리는 node js를 통해, 구글 플레이 서버에 데이터를 요청하지만, 요청자 이므로, 클라이언트라 불립니다. node js를 통해 접근할 때 사용할 클라이언트 계정을 만드는 단계 입니다.

 

그럼 자동으로 OAuth 클라이언트가 생성됩니다. 아이디는 바로 표시되지만, 비밀번호는 표시되지 않고, 구글 APIs로 이동한 뒤 확인하도록 되어 있습니다. Google Play Console에서 보기 버튼을 눌러 사용자 인증 페이지 창을 부릅니다.

 

그냥 만들어진 사용자는 지워 버립니다. 이 계정은 구글에서 사용하는 클라이언트 라이브러리를 사용할 때, 사용하게 되는 계정입니다. 우리는, nodejs REST FULL API 바식으로 Autho 2.0 인증을 사용할 것 입니다.

 

그럼 구글 APIs의 사용자 인증 정보 페이지가 표시되고, 클라이언트 아이디 리스트가 나타납니다. 여기서 비밀 번호를 바로 확인할 수는 없고, 연필버튼을 눌러 수정 창으로 들어가서 확인이 가능합니다. 참고로 말씀 드리면, 우리가 사용할 Google Play Android Developer는 구글 APIs 사이트의 하나의 API 입니다. (즉 바로 오셔도 됩니다.).

 

자 새로운 페이지에서 사용자 인증 정보 만들기 버튼을 눌러 풀다운 메뉴가 표시되면, OAuth 클라이언트 ID 항목을 선택해 계정 생성 창을 띄워 줍니다.

 

그럼 OAuth 클라이언트 ID 만들기 창이 생성되고, [애플리케이션 유형]은 [웹 애플리케이션]으로 선택 [이름]은 적당히, 승인된 리디렉션 URI 는 http://localhost/androidpublisher/exchange_token로 입력하고 (엔터 꼭,), 입력된 것을 확인 후 생성 버튼을 누릅니다.

 

3. 서비스 계정 만들기.

Autho2 로그인의 가장 큰 특징은 서버에 유저 패스워드나 비밀번호를 두지 않는 다는 점 입니다. 바로 SNS 등의 계정을 통해 로그인을 할 수 있는 데, 구글은 구글 계정을 통해 로그인을 하도록 되어 있습니다. 이 로그인을 할 때 사용하는 서비스 계정을 만들어 보도록 하겠습니다.

서비스 계정 만들기를 눌러 호출된 팝업에서, Google API 콘솔을 눌러 줍니디다.

여기서, [서비스 계정 만들기]를 눌러, 서비스 계만들기 창을 호출하도록 합니다.

적당한 값을 입력하고, [만들기] 버튼을 눌러 주도록 합니다.

이후 역할을 지정하도록 되어 있는데, 별다른 역할 없이 계속 계속을 눌러주면 되겠습니다.

 

4. node js로 영수증 검증 페이지를 만들기 위한 사전 정보 조사.

   1) 클라이언트 아이디, 시크릿 키, 리딕렉션 URI, 서비스 계정.

   이 값들은 위에서, 모두 보셨거나 직접 입력한 값들 입니다. 하지만, 언제든 다시 확인할 필요성이 있으므로, 다시 확인하는 방법을 알려 드립니다. 먼저 플레이 콘솔로 접속 합니다. ( 링크 ). 그리고, 나온 페이지 좌측에 설정을 눌러 줍니다.

   나온 페이지에서, [개발자 계정] > [API 액세스] 를 선택해 나온 페이지에서, 서비스 계정을 확인해 둡니다.

그리고, 구글 플레이 콘솔에서, 보기를 눌러, 사용자 인증 정보 창이 뜨면 클라이언트 아이디의 이름 부분을 클릭해 정보 페이지를 호출합니다.

클라이언트 ID, 클라이언트 보안 비밀, 승인된 리디렉션 URI 를 확인해 둡니다.

 

5. node js 사전 준비.

이 내용은 express framework를 이용해, router 다루는 방법을 알아야 합니다. nodejs로 웹서버 구축까지 싣기에는 양이 너무 많습니다.

express 서버를 포트 80번으로 설치해 localhost를 기본적으로 사용할 수 있도록 합니다.

 

1) request 모듈 설치.

$ npm i request

서버에서 다른 웹사이트의 내용을 읽어 올 때 사용됩니다.

 

2) querystring 모듈 설치.

$ npm i query

GET으로 리디렉션 할 때, URL 인코딩 (특수 문자들을 주로소 넣어도 문제가 없도록) 하도록 합니다.

 

3) androidpublisher.js 라우트 만들어 연결 시키기. 소스는 아래와 같습 니다. 일단 동작을 하는 모듈은 넣지 않고 가장 기본적인 상태로 두었습니다.

var express = require('express');
var router = express.Router();


// 필요한 모듈을 가지고 옵니다.
var request = require('request');
var qs = require('querystring');



router.get('/:name', function( req, res, next ){


    // 파라메터 값을 가지고 옵니다.
    let filename = req.params.name;


    // 처리 방법이 저정되지 않으면 다음으로 넘겨 버립니다.
    next();


});



module.exports = router;

아래와 같이 익스프레스 서버에 연결하도록 합니다.

var androidpublisher = require('./androidpublisher');


app.use('/androidpublisher', androidpublisher);

 

5. node js 코드

   1) 이 항목을 작성하는데, 읽어야할 기본 문서들.

이 페이지는 제가 필요한 부분들만 다루고 있습니다. 빠진 정보가 있을 수 있으므로, 공식 문서를 확인하도록 합니다. android 서버 부분은 android 개발 부분 뿐 이니라, 구글 APIs 부분도 봐야 합니다.

안드로이드 개발자 페이지에서, 서버 부분을 언급한 페이지 : https://developer.android.com/google/play/developer-api (소개수준)
구글 아이덴티티 플랫폼 OAuth 2.0 개요 : https://developers.google.com/identity/protocols/OAuth2 (많은 도움이 됩니다.)
웹서버 응용프로그램 (OAuth 2.0 구현) : https://developers.google.com/identity/protocols/OAuth2WebServer (실제 사용문서)
권한 범위 : https://developers.google.com/identity/protocols/googlescopes (OAuth 2.0 에서 권한을 요청할 때 사용되는 값들.)

위 문서들 중 별다은 언급이 없이 공식 문서를 참조 했다고 하면, 웹서버 응용프로그램 (OAuth 2.0 구현) 문서라 생각하시면되겠습니다.

 

   2) 로그인 페이지 호출 하게 하기.

OAuth 2.0의 핵심은, API를 가진 서버에, 자원을 활용하기 위해 웹브라우저를 이용하고, 우리 서버는 이 계정의 정보를 처리 하지 않음으로써, 안정성을 높이고, 우리는 계정 관리의 부담을 더는 것입니다. 즉, 우리서버를 아무리 해킹해도, 계정 정보는 남아 있지 않습니다.

이러한 작업을 할 수 있는 페이지를 호출 하고, 입력했던 리디렉션 주소로 결과를 받아 이 것이 올바르게 로그인 되었는지 한 번더 확인을 위한 토큰 교환을 시도하고, 성공하면 받은 토큰으로 API를 가진 서버의 자원에 접근할 수 있게 됩니다.

일단 저는 이 작업을 할 수 있는 페이지를 호출 하기위해 get_code 라는 페이지를 통해 페이지를 호출 할 수 있도록 하였습니다.

위 라우터에 http://localhost/androidpublisher/get_code 라고 브라우저 주소창에 입력하면, 구글 로그인 페이지가 호출 될 수 있도록 하기위해 다음과 같이 코드를 변경합니다. 앞으로 사용할 get_code와 exchange_token 메소드를 분기를 통해 처리할 수 있도록 하였습니다.
중요한 점은 recently_token 변수 인데, 잊지 않고 선언해 주도록 합니다.

router.get('/:name', function( req, res, next ){


    // 파라메터 값을 가지고 옵니다.
    let filename = req.params.name;


    if( filename == "get_code" )
    {
        get_code( req, res, next );
        return;             // 리턴 중요.
    }

    else if( filename == "exchange_token" )
    {
        exchange_token( req, res, next );
        return;             // 리턴 중요.
    }

    // 처리 방법이 저정되지 않으면 다음으로 넘겨 버립니다.
    next();


});



// 처음 코드를 가지고 오는 메소드 입니다.
function get_code( req, res, next )
{

}


// 토큰을 받고, 저장한 값.
var recently_token = "";


// 코드를 가지고 오는 메소드 입니다.
function exchange_token( req, res, next )
{

}

이렇게 입력하고, 코드를 가지고 오는 부분에서 공식 문서에서 지정한 로그인 페지를 불러 올 수 있도록, 주소를 입력합니다. 공식문서의 HTTP/REST 부분을 봐야 합니다.

코드로 다음과 같이 하면 될 것 같습니다. 여기서 사용되는 클라이언트 아이디, 클라이언트 시크릿 키, 리디게션 url은 위에서 모두 확인했던 값입니다. socpe (권한 범위는), 권한 범위 문서 (위에 링크 있음)을 참고 하시면됩니다.

// 결제 관리를 위한 상수값들 입니다. (앞서 확인했던 값들, 적당한 곳에서 상수로 선언).
var inn_client_id = "769707945781-kqe7j3bk8j3kbq3drliqv5josu74ukgn.apps.googleusercontent.com";
var inn_client_secret = "BfVV8gCKVlvGR_7G7E3IlWtN";
var inn_redirect_url = "http://localhost/androidpublisher/exchange_token";




// 처음 코드를 가지고 오는 메소드 입니다.
function get_code( req, res, next )
{


    // 코드를 요청하기 위한 주소를 만들어 줍니다.
    const url = "https://accounts.google.com/o/oauth2/v2/auth";
    const scope = "https://www.googleapis.com/auth/androidpublisher";


    // 코드를 획득하기 위한 요청을 만들어 줍니다.
    let code_url = url
    + `?scope=${ qs.escape(scope) }`
    + `&access_type=offline`
    // + `&include_granted_scopes=true`
    + `&redirect_uri=${ qs.escape(inn_redirect_url) }`
    + `&response_type=code`
    + `&client_id=${inn_client_id}`
    ;


    // 리디렉션 페이지를 호출 합니다.
    res.redirect( code_url );


}

이렇게 입력하면, 계정으로 로그인 창이 뜰 것 입니다.

 

   3) 토큰 교환.

그럼 토큰을 교환하는 페이지를 만들도록 합시다. 위 로그인을 시도하면, 성공시, 입력한 리디렉션 (http://localhost/androidpublisher/exchange_token) 페이지를 마지막으로 호출 하며, 코드를 get 타입으로 서버에 넘겨 줍니다. 받은 코드로 토큰 교환을 시도할 것이며, 보안을 위해, POST 메소드, 그리고, 서버 내에서 교환할 것 입니다. POST로 다른 서버의 REST API를 호출하기 위해서는 request 모듈이 필요합니다. (그냥 참고).

exchange_token 부분을 아래와 같이 코드를 작성하면, 토큰 교환이 완료됩니다.

// 토큰을 받고, 저장한 값.
var recently_token = "";


// 코드를 가지고 오는 메소드 입니다.
function exchange_token( req, res, next )
{


    let postForm = {
        grant_type: "authorization_code"
        , code: req.query.code
        , client_id: inn_client_id
        , client_secret: inn_client_secret
        , redirect_uri: inn_redirect_url
    };


    // form 데이터 형태로 토큰을 교환할 것 이므로, 폼데이터를 만들어 줍니다.
    request.post(
        "https://www.googleapis.com/oauth2/v4/token"
        , {form: postForm}, function( error, response, body ){

            // 오류가 난 경우 오류를 출력하도록 합니다.
            if( error )
            {
                res.end('Error!!');
                return;
            }


            let parsed_body = JSON.parse( body );
            recently_token = parsed_body.access_token;

            // 오류가 없을 경우 token을 그대로 받아 화면에 출력합니다.
            res.end( body );

        });

}

자 그럼, nodejs를 실행하고, 웹브라우저를 통해, http://localhost/androidpublisher/get_code 라고 입력합니다.

별문제가 없다면, 아래와 같은 창을 보게 될 것 입니다. 다른 계정 사용을 클릭 합니다.

그럼 로그인을 하라고 하는데, 여기서, 구글 플레이 개발자 계정으로 사용하는 계정(등록 비용을 지불하고 등록된) 이나, 서비스로 입력했던, 계정을 입력하면 됩니다. 여기서는 위에서 만들어 두었던 서비스 계정을 사용하도록 하겠습니다. 여기서는 편의를 위해 구글 개발자 계정(결제후 등록된) 계정으로 로그인하겠습니다. (이 부분은 그냥 넘어 가는 부분이므로, 화면 첨부를 화지 않겠습니다.).

성공했다면, 대게 이런 화면을 보게 될 것 입니다. access_token만 사용할 것이므로, 일단 이 것을 내부적으로,  recently_token에 저장되도록 하였습니다.

{
  "access_token": "ya29.GlvmBgcLk7RhR4EmBuKfPgxfiQVmQk9z57phE6kxEOu30yvM9kdYqW9Cn5zHMHmEPJISC7VdWaJMvScOFISq1A2I_-38YI861xRzwtmgJYacRnbxvVDhhomOHcEq",
  "expires_in": 3600,
  "refresh_token": "1/99m47DPXz7NVekvFGgULmpJv_jNSG7H6j0fFKeNR7j4JGiH2y8Xq53Nqp-67_XO3",
  "scope": "https://www.googleapis.com/auth/androidpublisher",
  "token_type": "Bearer"
}

 

   4) 리프레쉬 토큰을 이용한 토큰 다시 받기.

    이 부분은 다음에 시간 날 때 채워 넣도록 하겠습니다. (어렵지는 않고, 공식 문서만 참고 하셔도 충분합니다.).

 

6. 구매 검증.

구매 검증은 또, 아래 문서들을 참고해야 합니다.

Google Play Developer API : https://developers.google.com/android-publisher/getting_started ( 이 작업의 개요 부분 입니다.).
검증 작업 API : https://developers.google.com/android-publisher/api-ref/purchases/products/get ( 실제 작업 페이지 )

 

1) Android 앱에서 구매 후 구매 토큰 획득하기.

상품 구매는 Android 앱에서 일어나고, 결과로 토큰을 받아 옵니다. 상품구매후 토큰을 서버로 날려 주면 서버에서 그 토큰이 유효한지 확인하면 됩니다.
이 문서에서는 안드로이드 부분은 다루고 있지 않습니다. 아래 링크에서, 토큰을 획득하시면되겠습니다.

 

안드로이드 인앱 결제 (Android inapp billing)

Android는 내부적으로 결제를 할 수 있도록, API를 제공하고 있습니다. 이 문서는 인앱 결제에 대한 설명만을 포함하고 있습니다. 우선 앱이 프로덕션 혹은 베타테스트 출시 상태에서 보시면 좋으리라 생각합니다...

nicgoon.tistory.com

 

2) nodejs 에서 토큰 검증하기.

get을 통해 구매토큰만 입력하면, 처리가 될 수 있도록 하겠습니다. 일단 검증 함수를 라우터 속에 만들어 줍니다.

// 구입 상품 검증하는 메소드.
function validate_purchase( req, res, next )
{


    const _packageName = 'com.tistory.nicgoon.android.inapptest';
    const _productId = 'gold_1000';
    let _token = req.query.token;



    // 주소를 확인합니다.
    url = `https://www.googleapis.com/androidpublisher/v3/applications/${_packageName}/purchases/products/${_productId}/tokens/${_token}`;
    reqUrl = url + `?access_token=${recently_token}`;


    let logSTR = '받은토큰:' + req.query.token;
    console.log( logSTR );

    // res.end( reqUrl );
    res.redirect( reqUrl );

}

그리고, 라우트 메인에 검증을 처리 코드의 분기를 작성해 줍니다.

router.get('/:name', function( req, res, next ){


    // 파라메터 값을 가지고 옵니다.
    let filename = req.params.name;


    if( filename == "get_code" )
    {
        get_code( req, res, next );
        return;             // 리턴 중요.
    }

    else if( filename == "exchange_token" )
    {
        exchange_token( req, res, next );
        return;             // 리턴 중요.
    }

    // 구입 상품 검증.
    else if( filename == "validate_purchase" )
    {
        validate_purchase( req, res, next );
        return;             // 리턴 중요.
    }

    // 처리 방법이 저정되지 않으면 다음으로 넘겨 버립니다.
    next();


});

 

3) 검증 시도.

   코드를 수정했으므로, node js를 재 시작 합니다. 그리고, access_token이 없을 것이므로( 재 시작으로 인해 ), OAuth 2.0로그인 과정을 다시 해 줍니다. 

그리고, 브라우저 창에, 아래와 같이 입력하면, 상품 검증이 끝납니다. (당연히 여려분이 실 구현할 때는 내부에서 처리되도록 해야 합니다. request 모듈을 사용해, 지금은 간단히 해 보기 위해서, 브라우저를 사용합니다.

http://localhost/androidpublisher/validate_purchase?token=구매토큰

저는 앱으로 상품을 구매하고, 아래의 키를 받았습니다.

mjfiodoogjkoklolhkglhpbc.AO-J1Oy39MXZpJdwlbbMuUYB33ezvLnQRSckTNOkyZy4BRj12szzaz-z5C_WFdQGPidlPgPgEwAbJorPIvlL9hBmnJG8STaKt5BUCeBYxdduxqVE2mRo9FdRq87WbWeTZ_TBiAqVYfLUAdEt_LLnrwBCqpaI8lxGCg

그래서 요청 주소는 아래와 같이 되었습니다.

http://localhost/androidpublisher/validate_purchase?token=mjfiodoogjkoklolhkglhpbc.AO-J1Oy39MXZpJdwlbbMuUYB33ezvLnQRSckTNOkyZy4BRj12szzaz-z5C_WFdQGPidlPgPgEwAbJorPIvlL9hBmnJG8STaKt5BUCeBYxdduxqVE2mRo9FdRq87WbWeTZ_TBiAqVYfLUAdEt_LLnrwBCqpaI8lxGCg

 

아래와 같은 값을 확인했다면, 검증에 성공했고, purchaseState 값이 0 이라면, 취소되지 않은 구매 상품 입니다.

{
 "kind": "androidpublisher#productPurchase",
 "purchaseTimeMillis": "1554870248765",
 "purchaseState": 0,
 "consumptionState": 0,
 "developerPayload": "",
 "orderId": "GPA.3305-1217-4902-80622",
 "purchaseType": 0
}

 

수고하셨습니다.

 

 

Posted by 창업닉군
,