Игнорирование исключения из готовой функции после тайм-аута - PullRequest
0 голосов
/ 03 мая 2019

Я работаю над облачной функцией HTTP, которая анализирует много данных. Эта функция длится дольше 60000 мс, поэтому я получаю Function execution took 60002 ms, finished with status: 'timeout', что ожидается, но функция продолжает работать.

Проблема возникает, когда через некоторое время возникает исключение после того, как функция закончена (тайм-аут). Выдает Ignoring exception from a finished function, поэтому я не могу отладить проблему, потому что не знаю, что происходит за кулисами.

Какой мне должен быть подход к решению этой проблемы?

import { region, config, auth as authTypes } from 'firebase-functions';
import { request } from 'https';
import to from 'await-to-js';
import { db, auth } from './app';
import { Investment } from './types/Investment';
import { InvestorRecord, InvestorTier, InvestorStatus } from './types/InvestorRecord';
import { Asset } from './types/Asset';
import { PaymentStatus } from './types/Payment';

const CONFIG = config();

exports = module.exports = region('europe-west1').https.onRequest(
  async (mainRequest, mainResponse): Promise<void> => {
    const { dealStageID, fundName } = mainRequest.query;
    if (!dealStageID || !fundName) {
      throw new Error('Error with params');
    }

    // Function that handles a request to meerdervoort's sharpspring API
    // It returns a fully parsed JSON response
    const meerdervoortDatabaseRequest = (data): Promise<{ [key: string ]: any }> => new Promise((resolve, reject): void => {
      const stringifiedData = JSON.stringify(data);

      const options = {
        hostname: 'api.sharpspring.com',
        port: 443,
        path: `/pubapi/v1/?accountID=${CONFIG.sharp_spring.account}&secretKey=${CONFIG.sharp_spring.key}`,
        method: 'POST',
        headers: {
          'Content-Type': 'application/json',
          'Content-Length': stringifiedData.length,
        },
      };

      // Opportunity Leads request
      const mRequest = request(options, (meerderResponse): void => {
        const bufferedData = [];

        meerderResponse.on('data', (dataBuffer): void => {
          bufferedData.push(dataBuffer);
        });

        meerderResponse.on('end', (): void => {
          resolve(JSON.parse(Buffer.concat(bufferedData).toString()));
        });
      });

      mRequest.on('error', (error): void => {
        reject(error);
      });

      mRequest.write(stringifiedData);
      mRequest.end();
    });

    const fund = {
      dealStageID: 399799298,
      name: 'MGF II',
    };
    const [speOppoError, speOppoSuccess] = await to(meerdervoortDatabaseRequest({
      method: 'getOpportunities',
      params: {
        where: {
          dealStageID,
        },
      },
      id: 1,
    }));
    if (speOppoError) {
      mainResponse.status(500).send(speOppoError.message);

      return;
    }

    const opportunities = (speOppoSuccess.result.opportunity as any[]).filter((opp): boolean => opp.isWon === '1');

    // Using custom asyncForEach because a promise.all with transactions makes too many collisions and Firebase doesn't support more than 5
    const asyncForEach = async (array, callback): Promise<any> => {
      for (let index = 0; index < array.length; index++) {
        try {
          await callback(array[index], index, array);
        } catch (e) {
          console.log('unfinished');
          mainResponse.status(500).send(e);
          throw new Error(e);
        }
      }
    };

    const leads = { };
    await asyncForEach(opportunities, async (oppo, i): Promise<void> => {
      console.log(`Loading... ${Math.round(i / opportunities.length * 100)} %`);

      // Here we are copying the pointer, not the data
      let user = leads[oppo.originatingLeadID || oppo.primaryLeadID];

      /** USER / INVESTOR PART */

      let investorId = '';
      // If the user (lead) has not been tracked down yet, we do it now
      if (!user || Object.keys(user).length === 0) {
        const [getLeadsError, getLeadsSuccess] = await to(meerdervoortDatabaseRequest({
          method: 'getLead',
          params: {
            // originatingLeadID can be '0' so we convert to Number and won't give problems with the OR condition
            id: Number(oppo.originatingLeadID) || Number(oppo.primaryLeadID),
          },
          id: i,
        }));
        if (getLeadsError) {
          throw new Error(getLeadsError.message);
        }

        const lead = getLeadsSuccess.result.lead[0];

        if (lead.emailAddress) {
          // console.log(lead.emailAddress);
          // Check if entry is in Firebase Auth
          const [getUserError, getUserSuccess] = await to(auth.getUserByEmail(lead.emailAddress));
          if (getUserError && getUserError.code !== 'auth/user-not-found') {
            throw new Error(getUserError.message);
          }

          // If not, register in Firebase Auth
          if (!getUserSuccess) {
            // Code to register
            const [registerUserError, registerUserSuccess] = await to(auth.createUser({
              disabled: false,
              email: lead.emailAddress,
              emailVerified: false,
            }));
            if (registerUserError && registerUserError.code !== 'auth/invalid-email') {
              throw new Error(registerUserError.message);
            }

            if (!registerUserError) {
              user = { ...registerUserSuccess };
            }
          } else {
            // Updating content inside the pointer
            user = { ...getUserSuccess };
          }
        }

        const extraData = (investor): { [key: string]: any } => {
          const tempInvestor: { [key: string]: any } = { ...investor };

          if (lead.firstName) {
            tempInvestor.name = lead.firstName;
          }
          if (lead.lastName) {
            tempInvestor.surName = lead.lastName;
          }
          if (lead.street) {
            tempInvestor.streetAddress = lead.street;
          }
          if (lead.zipcode) {
            tempInvestor.postalCode = (lead.zipcode as string).replace(/\s/g, '');
          }
          if (lead.mobilePhoneNumber || lead.phoneNumber) {
            tempInvestor.phone = lead.mobilePhoneNumber || lead.phoneNumber;
          }
          ['city', 'country'].forEach((key): void => {
            if (lead[key]) {
              tempInvestor[key] = lead[key];
            }
          });

          return tempInvestor;
        };

        // Check if entry is in Firestore as Investor
        if (user && (user as authTypes.UserRecord).uid) {
          const [getInvestorError, getInvestorSuccess] = await to(db.collection('investors').doc((user as authTypes.UserRecord).uid).get());
          if (getInvestorError) {
            throw new Error(getInvestorError.message);
          }

          const investorData = getInvestorSuccess.data();
          if ((getInvestorSuccess.exists && investorData && !investorData.identified) || !getInvestorSuccess.exists) {
            let investor: { [key: string]: any } = {
              updatedDatetime: Date.now(),
              identified: true,
            };

            investor = extraData(investor);

            const [updateIdentifiedError] = await to(db.collection('investors').doc((user as authTypes.UserRecord).uid).set(investor, { merge: true }));
            if (updateIdentifiedError) {
              throw new Error(updateIdentifiedError.message);
            }

            // Updating content inside the pointer
            user = { ...user, ...investorData };
          }

          investorId = getInvestorSuccess.id;
        } else {
          const [getInvestorError, getInvestorSuccess] = await to(
            db.collection('investors').where('name', '==', lead.firstName).where('streetAddress', '==', lead.street).get(),
          );
          if (getInvestorError) {
            throw new Error(getInvestorError.message);
          }

          if (getInvestorSuccess.empty) {
            const [getLastInvestorError, investorsQueryResult] = await to(db.collection('investors').orderBy('customId', 'desc').limit(1).get());
            if (getLastInvestorError) {
              throw new Error('There was an error reading investors data.');
            }

            const lastId = investorsQueryResult.size === 1 ? investorsQueryResult.docs[0].data().customId : 0;

            let investor: { [key: string]: any } = {
              name: lead.firstName,
              surName: lead.lastName,
              tier: InvestorTier.Starter,
              createdDatetime: Date.now(),
              updatedDatetime: Date.now(),
              status: InvestorStatus.Enabled,
              identified: true,
              customId: lastId + 1,
            };

            investor = extraData(investor);

            const investorRef = db.collection('investors').doc();
            const [registerInvestorError] = await to(investorRef.create(investor));
            if (registerInvestorError) {
              throw new Error(registerInvestorError.message);
            }

            user = { ...investor, uid: 'NOT REGISTERED' };
            investorId = investorRef.id;
          } else {
            user = { ...getInvestorSuccess.docs[0].data() };
            investorId = getInvestorSuccess.docs[0].id;
          }
        }
      }

      /** INVESTMENT / PAYMENT PART */
      const [getAssetError, getAssetSuccess] = await to(
        db.collection('assets').where('name', '==', fundName).get(),
      );
      if (getAssetError) {
        throw new Error(getAssetError.message);
      }

      if (getAssetSuccess.empty) {
        throw new Error('Asset not found.');
      }

      const asset = getAssetSuccess.docs[0].data() as Asset;
      const assetId = getAssetSuccess.docs[0].id;
      const [getInvestmentError, getInvestmentSuccess] = await to(
        db.collection('investments').where('assetId', '==', assetId).where('investorId', '==', investorId).get(),
      );
      if (getInvestmentError) {
        throw new Error(getInvestmentError.message);
      }

      const date = Date.now();
      const investment: Investment = {
        assetId,
        createdDatetime: date,
        updatedDatetime: date,
        userId: user.uid || 'NOT REGISTERED',
        payments: [],
        investorId,
      };

      const paidEuro = Number(oppo.amount);
      const sharesBought = paidEuro / asset.sharePrice;
      if (getInvestmentSuccess.empty) {
        investment.paidEuroTotal = Number(oppo.amount);
        investment.boughtSharesTotal = sharesBought;
      } else {
        const foundInvestment = getInvestmentSuccess.docs[0].data() as Investment;
        investment.createdDatetime = foundInvestment.createdDatetime;
        investment.payments = foundInvestment.payments;
        investment.paidEuroTotal = foundInvestment.paidEuroTotal + paidEuro;
        investment.boughtSharesTotal = foundInvestment.boughtSharesTotal + sharesBought;
      }

      // Get from example: looptijd_mgf_ii__1__5c34a9b697a02 that full keyname (cannot be null)
      const findKey = Object.keys(oppo).find((key): boolean => key.startsWith('looptijd_') && oppo[key]);

      // Get from example: '7 jaar' the first number/combination of strings
      let dividendsFormatYears = '';
      try {
        dividendsFormatYears = (oppo[findKey] as string).substr(0, (oppo[findKey] as string).indexOf(' '));
      } catch (e) {
        console.log(findKey);
        console.log(oppo);
        throw new Error('Error at parsing string');
      }

      try {
        investment.payments = [
          ...investment.payments,
          {
            createdDatetime: date,
            id: oppo.id,
            provider: 'Transfer',
            updatedDatetime: date,
            userId: user.uid || 'NOT REGISTERED',
            investmentId: getInvestmentSuccess.empty ? 'unknown' : getInvestmentSuccess.docs[0].id, // ToDo: fix unknown
            dividendsFormat: asset.dividendsFormat.find((contentsObject): boolean => contentsObject.contents[0] === dividendsFormatYears).contents,
            providerData: {
              id: oppo.id,
              amount: {
                currency: 'EUR',
                value: oppo.amount,
              },
              status: PaymentStatus.Paid,
            },
          },
        ];
      } catch (e) {
        throw new Error(e);
      }

      // Create/update investment
      const ref = getInvestmentSuccess.empty
        ? db.collection('investments').doc()
        : db.collection('investments').doc(getInvestmentSuccess.docs[0].id);

      const [setInvesmentError] = await to(ref.set(investment, { merge: true }));
      if (setInvesmentError) {
        throw new Error(setInvesmentError.message);
      }

      // Update Asset
      const [updateAssetError] = await to(db.collection('assets').doc(assetId).update({
        sharesAvailable: asset.sharesAvailable - sharesBought,
      }));
      if (updateAssetError) {
        throw new Error(updateAssetError.message);
      }
    });

    console.log('finished');
    // @ts-ignore
    mainResponse.status(200).send('OK');
  },
);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...