Как передать данные через firebase-функции в Paypal? - PullRequest
0 голосов
/ 01 января 2019

У меня есть приложение на Android и iOS на одном проекте Firebase.Последняя часть приложения - отправка данных из моего приложения в firebase-function, а затем в PayPal.Это работает в Android, но не в iOS.Пожалуйста, смотрите код ниже.

Я даже говорил с PayPal, но они думают, что это код iOS.У меня есть сообщения в GitHub, PayPal, Stack Overflow и т. Д., Без разрешения.

Обычно, когда я нажимаю кнопку - GET PAYOUT, выполняется метод payoutRequest ().Он должен отправить 'uid' и 'email' в функцию, а затем в PayPal.

Пакет PayPal SDK включен в функцию.

Он не отправляет данные в PayPal, нораспознается в функции (функция https не является вызываемой функцией)

рабочий код Android

public static final MediaType MEDIA_TYPE = MediaType.parse("application/json"); ProgressDialog progress;

private void payoutRequest() {

progress = new ProgressDialog(this);
progress.setTitle("Processing your payout ...");
progress.setMessage("Please Wait .....");
progress.setCancelable(false);
progress.show();

// HTTP Request ....
final OkHttpClient client = new OkHttpClient();

// in json - we need variables for the hardcoded uid and Email
JSONObject postData = new JSONObject();

try {
    postData.put("uid", FirebaseAuth.getInstance().getCurrentUser().getUid());
    postData.put("email", mPayoutEmail.getText().toString());

} catch (JSONException e) {
    e.printStackTrace();
}

// Request body ...
RequestBody body = RequestBody.create(MEDIA_TYPE, postData.toString());

// Build Request ...
final Request request = new Request.Builder()
        .url("https://us-central1-ryyde-sj.cloudfunctions.net/payout")
        .post(body)
        .addHeader("Content-Type", "application/json")
        .addHeader("cache-control", "no-cache")
        .addHeader("Authorization", "Your Token")
        .build();

client.newCall(request).enqueue(new Callback() {
    @Override
    public void onFailure(Call call, IOException e) {
        // something went wrong right off the bat
        progress.dismiss();
    }

    @Override
    public void onResponse(Call call, Response response) throws IOException {
        // response successful ....
        // refers to response.status('200') or ('500')
        int responseCode = response.code();
        if (response.isSuccessful()) {
            switch(responseCode) {
                case 200:
                    Snackbar.make(findViewById(R.id.layout),
                            "Payout Successful!", Snackbar.LENGTH_LONG)
                            .show();
                    break;

                case 500:
                    Snackbar.make(findViewById(R.id.layout),
                            "Error: no payout available", Snackbar
                                    .LENGTH_LONG).show();
                    break;

                default:
                    Snackbar.make(findViewById(R.id.layout),
                            "Error: couldn't complete the transaction",
                            Snackbar.LENGTH_LONG).show();
                    break;
            }

        } else {
            Snackbar.make(findViewById(R.id.layout),
                    "Error: couldn't complete the transaction",
                    Snackbar.LENGTH_LONG).show();
        }

        progress.dismiss();
    }
});
}

Firebase-функция: index.js

'use strict';
const functions = require('firebase-functions');
const paypal = require('paypal-rest-sdk');
const admin = require('firebase-admin');
admin.initializeApp(functions.config().firebase);

paypal.configure({
    mode: 'sandbox',
    client_id: functions.config().paypal.client_id,
    client_secret: functions.config().paypal.client_secret
})

exports.newRequest = functions.database.ref('/history/{pushId}').onCreate((snapshot, context) => {
    var requestSnapshot = snapshot.val();
    var price  = snapshot.child('price').val();
    var pushId = context.params.pushId;

    return snapshot.ref.parent.child(pushId).child('price').set(price);
 });


function getPayoutsPending(uid) {
    return admin.database().ref('Users/Drivers/' + uid + '/history').once('value').then((snap) => {
        if(snap === null){
            throw new Error("profile doesn't exist");
        }
        var array = [];
        if(snap.hasChildren()){
            snap.forEach(element => {
                if (element.val() === true) {
                    array.push(element.key);
                }
            });
        }
        return array;
    }).catch((error) => {
        return console.error(error);
    });
}

function getPayoutsAmount(array) {
    return admin.database().ref('history').once('value').then((snap) => {
        var value = 0.0;
        if(snap.hasChildren()){
            snap.forEach(element => {
                if(array.indexOf(element.key) > -1) {
                        if(element.child('price').val() !== null){
                            value += element.child('price').val();
                        }
                }
            });
            return value;
        }
        return value;
    }).catch((error) => {
        return console.error(error);
    });
}

function updatePaymentsPending(uid, paymentId) {
    return admin.database().ref('Users/Drivers/' + uid + '/history').once('value').then((snap) => {
        if(snap === null){
            throw new Error("profile doesn't exist");
        }

        if(snap.hasChildren()){
            snap.forEach(element => {
                if(element.val() === true) {
                    admin.database().ref('Users/Drivers/' + uid + '/history/' + element.key).set( {
                        timestamp: admin.database.ServerValue.TIMESTAMP,
                        paymentId: paymentId
                    });
                    admin.database().ref('history/' + element.key + '/driverPaidOut').set(true);
                }
            });
        }
        return null;
    }).catch((error) => {
        return console.error(error);
    });
}

exports.payout = functions.https.onRequest((request, response) => {
    return getPayoutsPending(request.body.uid)
        .then(array => getPayoutsAmount(array))
        .then(value => {
            var valueTrunc = parseFloat(Math.round((value * 0.75) * 100) / 100).toFixed(2);
            const sender_batch_id = Math.random().toString(36).substring(9);
            const sync_mode = 'false';
            const payReq = JSON.stringify({
                sender_batch_header: {
                    sender_batch_id: sender_batch_id,
                    email_subject: "You have a payment"
                },
                items: [
                    {
                        recipient_type: "EMAIL",
                        amount: {
                            value: valueTrunc,
                            currency: "CAD"
                        },
                        receiver: request.body.email,
                        note: "Thank you.",
                        sender_item_id: "Payment"
                    }
                ]
            });

            return paypal.payout.create(payReq, sync_mode, (error, payout) => {
                if (error) {
                    console.warn(error.response);
                    response.status('500').end();
                    throw error;
                }
                console.info("uid: " + request.body.uid + " email: " + request.body.email) // testing
                console.info("payout created");
                console.info(payout);
                return updatePaymentsPending(request.body.uid, sender_batch_id)
            });
        }).then(() => {
            response.status('200').end();
            return null;
        }).catch(error => {
            console.error(error);
        });
});

Код IOS: payoutRequest ()

var params: Parameters = ["uid": FIRAuth.auth()?.currentUser!.uid as Any, "email": txtPayoutEmail.text!]

    let url = URL(string: "https://us-central1-ryyde-sj.cloudfunctions.net/payout")

    let headers: HTTPHeaders = [
        "Content-Type": "application/json",
        "Authorization": "Your Token",
        "cache-control": "no-cache"
    ]

    let request = NSMutableURLRequest(url: url!)
    let session = URLSession.shared
    request.httpMethod = "POST"
    params = params as! Dictionary<String, String>
    request.httpBody = try! JSONSerialization.data(withJSONObject: params, options: [])
    request.allHTTPHeaderFields = headers

    let task = session.dataTask(with: request as URLRequest) { data, response, error in
        guard data != nil, error == nil else {
            print("no data found: \(String(describing: error))")
            return
        }

        if let httpStatus = response as? HTTPURLResponse, httpStatus.statusCode != 200 {
            print("status code: \(httpStatus.statusCode)")
            print("response: \(String(describing: response))")
        }

        let jsonData = try? JSONSerialization.data(withJSONObject: params, options: .prettyPrinted)
        let parsedObject = try! JSONSerialization.jsonObject(with: jsonData!, options: .allowFragments)
        print(parsedObject)
    }

    task.resume()

Что происходит

Когда метод (payoutRequest ()), журналы в Firebase Function показывают, что параметры 'uid' и 'email' сохраняются:

firebase logs

, как вы можете видетьв 3-й строке консоли параметры uid и email сохраняются.

Следующее, что должно произойти, это то, что данные (параметры) должны быть отправлены в Paypal, а уведомление отправлено на адрес электронной почты с указаниемплатеж получен, как это:

paypal notification

, но такого уведомления нет.

Если я проверяю вызовы API для Песочницы, я вижу это:

PayPal API Call for Sandbox

Затем через 4 часа до дня я получаюэта ошибка в журналах, в основном, была восстановлена, потому что PayPal не сохранил отправленные данные:

firebase logs 2

1 Ответ

0 голосов
/ 07 января 2019

Одна из проблем, с которыми вы сталкиваетесь в коде iOS, заключается в том, что значение params должно иметь тип [String: String] (это обязательный тип для данных POST).

Я имею в видуконкретно эта строка:

var params: Parameters = ["uid": FIRAuth.auth () ?. currentUser! .uid as Any, "email": txtPayoutEmail.text!]

Попробуйте преобразовать uid вместо String:

var params: Parameters = ["uid": FIRAuth.auth () ?. currentUser! .Uid as String, "email": txtPayoutEmail.text!]

...