Skip to content
Snippets Groups Projects
viamapi-iframe.js 52.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • Damyan Mitev's avatar
    Damyan Mitev committed
              return encodeResponse("400", "", convResponse.status);
            }
          }
    
          const downloadResponse = await executeRestfulFunction(
    
            "private", window.viamApi, window.viamApi.documentGetDocumentByUUID, null, documentUUID, pdfContentType);
    
    Damyan Mitev's avatar
    Damyan Mitev committed
    
          if (downloadResponse.code !== "200") {
            return encodeResponse("400", "", downloadResponse.status);
          }
    
    
          const pdfRaw = base64ToByteArray(downloadResponse.data);
    
    Damyan Mitev's avatar
    Damyan Mitev committed
    
    
          let signedPdf;
          try {
            signedPdf = await signPdf(pdfRaw, certificateOneTime, passportChain, privateKeyOneTime);
          } catch (err) {
            console.error(err);
            return encodeResponse("500", "", err.message);
          }
    
    Damyan Mitev's avatar
    Damyan Mitev committed
    
    
          const signedPdfB64 = byteArrayToBase64(signedPdf);
    
    
          const uploadResponse = await executeRestfulFunction(
            "private", window.viamApi, window.viamApi.documentPutDocumentByUUID, null, documentUUID, pdfContentType, signedPdfB64);
          if (uploadResponse.code !== "200") {
            return encodeResponse("400", "", uploadResponse.status);
          }
    
          const signResponse = await executeRestfulFunction(
            "private", window.viamApi, window.viamApi.documentSignDocumentByUUID, null, passportUUID, documentUUID, pdfContentType);
          if (signResponse.code !== "200") {
            return encodeResponse("400", "", signResponse.status);
          }
    
          return encodeResponse("200", "", "Document signed");
    
    Damyan Mitev's avatar
    Damyan Mitev committed
        },
    
        documentCreateDocument: async (passportUUID, path, contentType, title) => {
    
          const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
          if (
            !authenticationPublicKey ||
            !window.loadedIdentities[authenticationPublicKey] ||
            !extendPinCodeTtl(authenticationPublicKey)
          ) {
            return encodeResponse("400", "", "Identity not authenticated");
          }
    
    
          path = encodeURI(path);
          contentType = encodeURI(contentType);
          title = encodeURI(title);
    
    
          const config = {
            headers: {
              path,
              passportuuid: passportUUID,
    
            }
          };
          const response = await executeRestfulFunction("private", window.viamApi, window.viamApi.documentCreateDocument,
            config);
    
          if (response.code !== "200") {
            return encodeResponse("400", "", response.status);
          }
    
    
          return encodeResponse("200", response.data, "Document created");
        },
    
        documentPutDocument: async (passportUUID, resourceid, contentType, file) => {
    
          const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
          if (
            !authenticationPublicKey ||
            !window.loadedIdentities[authenticationPublicKey] ||
            !extendPinCodeTtl(authenticationPublicKey)
          ) {
            return encodeResponse("400", "", "Identity not authenticated");
          }
    
    
          resourceid = encodeURI(resourceid);
          contentType = encodeURI(contentType);
    
    
          const config = {
            headers: {
              'Content-Type': 'multipart/form-data',
              passportuuid: passportUUID,
    
    
          const response = await executeRestfulFunction(
    
            "private", window.viamApi, window.viamApi.documentPutDocument, config, file);
    
          if (response.code !== "200") {
            return encodeResponse("400", "", response.status);
          }
    
    
          return encodeResponse("200", response.data, "Document stored");
        },
    
        hasSession() {
          return new Penpal.Promise(result => {
    
            const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
            if (authenticationPublicKey === null) {
    
              result({"data" : "",
                "code" : "400",
                "status" : "Identity not authenticated"
              });
            }
    
            if (window.loadedIdentities[authenticationPublicKey] === null) {
    
              result({"data" : "",
                "code" : "400",
                "status" : "Identity not authenticated"
              });
            }
    
    
    Markin Igor's avatar
    Markin Igor committed
            var success = extendPinCodeTtl(authenticationPublicKey);
    
    Markin Igor's avatar
    Markin Igor committed
            if(success === false) {
    
              result({"data" : "",
                "code" : "400",
                "status" : "Identity not authenticated"
              })
            }
    
    
            executeRestfulFunction("private", viamApi, viamApi.identityHasSession, null).then(executeResult => {
    
              result(executeResult);
            });
          });
        },
        marketingSignUpIdentificator(identificator, reference) {
          return new Penpal.Promise(result => {
    
    Markin Igor's avatar
    Markin Igor committed
            viamApi.setIdentity("marketingapppublickey");
    
            executeRestfulFunction("public", viamApi, viamApi.marketingSignUpIdentificator, null, identificator, reference).then(executeResult => {
    
    Markin Igor's avatar
    Markin Igor committed
              viamApi.setIdentity("");
    
    Markin Igor's avatar
    Markin Igor committed
              viamApi.setSessionData("", "");
    
              result(executeResult);
            });
          });
        },
        marketingGetIdentificatorProfile(identificator, pincode) {
          return new Penpal.Promise(result => {
    
    Markin Igor's avatar
    Markin Igor committed
            viamApi.setIdentity("marketingapppublickey");
    
            executeRestfulFunction("public", viamApi, viamApi.marketingGetIdentificatorProfile, null, identificator, pincode).then(executeResult => {
    
    Markin Igor's avatar
    Markin Igor committed
              viamApi.setIdentity("");
    
    Markin Igor's avatar
    Markin Igor committed
              viamApi.setSessionData("", "");
    
              result(executeResult);
            });
          });
        },
    
        marketingExecuteEventForIdentificator(identificator, pincode, event) {
    
          return new Penpal.Promise(result => {
    
    Markin Igor's avatar
    Markin Igor committed
            viamApi.setIdentity("marketingapppublickey");
    
            executeRestfulFunction("public", viamApi, viamApi.marketingExecuteEventForIdentificator, null, identificator, pincode, event).then(executeResult => {
    
    Markin Igor's avatar
    Markin Igor committed
              viamApi.setIdentity("");
    
    Markin Igor's avatar
    Markin Igor committed
              viamApi.setSessionData("", "");
    
              result(executeResult);
            });
          });
        },
        getCurrentlyAuthenticatedIdentity() {
    
          const { publicKey, x509Certificate } = window.currentlyAuthenticatedIdentity.authentication;
    
          return encodeResponse(
            "200",
            {
              authentication: {
                publicKey,
                x509Certificate
              }
            },
            "Currently authenticated identity"
          );
    
        stringToUtf8ByteArray(str) {
          return new Penpal.Promise(result => {
    
            result(stringToUtf8ByteArray(str));
          });
    
        },
        utf8ByteArrayToString(ba) {
          return new Penpal.Promise(result => {
    
            result(utf8ByteArrayToString(ba));
          });
    
        },
        stringToUtf8Base64(str) {
          return new Penpal.Promise(result => {
    
            result(stringToUtf8Base64(str));
          });
    
        },
        utf8Base64ToString(strBase64) {
          return new Penpal.Promise(result => {
    
            result(utf8Base64ToString(strBase64));
          });
    
        },
        base64ToByteArray(strBase64) {
          return new Penpal.Promise(result => {
    
            result(base64ToByteArray(strBase64));
          });
    
        },
        byteArrayToBase64(ba) {
          return new Penpal.Promise(result => {
    
            result(byteArrayToBase64(ba));
          });
    
    Alexey Lunin's avatar
    Alexey Lunin committed
    
        // Collabora APIs
    
        collaboraDiscovery() {
    
          return collaboraApi.discovery().then(apps => apps);
    
    Alexey Lunin's avatar
    Alexey Lunin committed
        // WOPI
    
        getPassports: async (resourceID, contentType) => {
    
    Alexey Lunin's avatar
    Alexey Lunin committed
          const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
    
    Alexey Lunin's avatar
    Alexey Lunin committed
          if (
            !authenticationPublicKey ||
            !window.loadedIdentities[authenticationPublicKey] ||
            !extendPinCodeTtl(authenticationPublicKey)
          ) {
            return encodeResponse("400", "", "Identity not authenticated");
          }
    
          const response = await wopiAPI.getPassports(resourceID, contentType);
    
    Alexey Lunin's avatar
    Alexey Lunin committed
          return response.data;
    
        wopiCreateDocument: async (passportUUID, path, contentType, title) => {
          const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
          if (
            !authenticationPublicKey ||
            !window.loadedIdentities[authenticationPublicKey] ||
            !extendPinCodeTtl(authenticationPublicKey)
          ) {
            return encodeResponse("400", "", "Identity not authenticated");
          }
    
          const config = {
            headers: {
              path,
              passportuuid: passportUUID,
              contentType,
              title
            }
          };
          const executeResult = await executeRestfulFunction("private", window.viamApi, window.viamApi.documentCreateDocument,
            config);
          if (executeResult.code !== "200") return executeResult;
          const resourceID = executeResult.data;
          const passports = await wopiAPI.getPassports(resourceID, contentType);
          return passports;
        },
    
    
        wopiPutFile: async (path, accessToken, file) => {
    
    Alexey Lunin's avatar
    Alexey Lunin committed
          const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
    
    Alexey Lunin's avatar
    Alexey Lunin committed
          if (
            !authenticationPublicKey ||
            !window.loadedIdentities[authenticationPublicKey] ||
            !extendPinCodeTtl(authenticationPublicKey)
          ) {
            return encodeResponse("400", "", "Identity not authenticated");
          }
    
    
          const response = await wopiAPI.putDocument(path, accessToken, file);
    
    Alexey Lunin's avatar
    Alexey Lunin committed
          return response.data;
    
    Markin Igor's avatar
    Markin Igor committed
    });
    
    connection.promise.then(parent => {
    
      iframeParent = parent;
    
    
      if (!navigator.cookieEnabled) {
        console.warn("Cookie disabled. Can't start library.");
        return;
      }
    
    
      window.addEventListener('storage', event => {
        if (event.key === "authenticatedIdentity" && event.newValue === null) {
    
    Markin Igor's avatar
    Markin Igor committed
          const publicKey = window.currentlyAuthenticatedIdentity.authentication.publicKey;
    
          window.currentlyLoadedIdentity = null;
          window.currentlyAuthenticatedIdentity = null;
    
    Markin Igor's avatar
    Markin Igor committed
          const event = createEvent("LogoutFromAnotherTab", "Logout", [publicKey]);
          parent.onEvent(event);
    
    Markin Igor's avatar
    Markin Igor committed
      const identities = localStorage.getItem("identities");
    
    Markin Igor's avatar
    Markin Igor committed
      console.log("Library loaded at: " + new Date().toISOString());
    
    
      if (identities === "" || identities === null) {
    
    Markin Igor's avatar
    Markin Igor committed
        localStorage.setItem("identities", JSON.stringify({}));
    
    Markin Igor's avatar
    Markin Igor committed
      if (
        localStorage.getItem("uuid") === null ||
        localStorage.getItem("token") === null ||
        localStorage.getItem("authenticatedIdentity") === null
      ) {
    
        const event = createEvent("", "NotAuthenticated");
        parent.onEvent(event);
    
    Markin Igor's avatar
    Markin Igor committed
        localStorage.removeItem("uuid");
        localStorage.removeItem("token");
    
    Markin Igor's avatar
    Markin Igor committed
        localStorage.removeItem("authenticatedIdentity");
    
        const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
        const pinCode = getPincode(authenticationPublicKey);
    
    Markin Igor's avatar
    Markin Igor committed
        if (pinCode === "" || pinCode === null) {
    
          loadIdentityInternal(authenticationPublicKey, "00000000").then(result => {
    
    Markin Igor's avatar
    Markin Igor committed
            if (result.code !== "200") {
              const event = createEvent(
                "CanNotGetPincodeForAuthenticatedIdentity",
                "IdentityNotLoaded",
    
                [authenticationPublicKey]
    
    Markin Igor's avatar
    Markin Igor committed
              );
              parent.onEvent(event);
    
          loadIdentityInternal(authenticationPublicKey, pinCode).then(result => {
    
    Markin Igor's avatar
    Markin Igor committed
            if (result.code !== "200") {
              const event = createEvent(
                "CanNotLoadIdentity",
                "ErrorDuringLoadingIdentity",
    
                [authenticationPublicKey]
    
    Markin Igor's avatar
    Markin Igor committed
              );
              parent.onEvent(event);
    
      let anynomousDeviceKeyEventsProcessing = false;
      let maxDeviceKeyAnonymousEventTime = 0;
    
      let eventsDeviceEventsProcessing = false;
      let maxDeviceKeyEventTime = 0;
    
      let eventsEntityEventsProcessing = false;
      let maxEntityEventTime = 0;
    
      let identityLoadedEvent = false;
      let identityAuthenticatedEvent = false;
    
    Alexey Lunin's avatar
    Alexey Lunin committed
      let previousLocalStorageUUID;
      let previousLocalStorageToken;
      let previousLocalStorageIdentity;
    
    
    Markin Igor's avatar
    Markin Igor committed
      setInterval(async function () {
        if (window.currentlyAuthenticatedIdentity) {
          const { authentication } = window.currentlyAuthenticatedIdentity;
          const pinCode = getPincode(authentication.publicKey);
          if (pinCode) {
            const identity = await getIdentityFromLocalStorage(authentication.publicKey, pinCode, false);
    
    Markin Igor's avatar
    Markin Igor committed
            window.currentlyLoadedIdentity = identity;
    
            if (!identityAuthenticatedEvent && identity) {
              const event = createEvent("IdentityAuthenticated", "Authenticated", [identity.authentication.publicKey]);
              parent.onEvent(event);
              identityAuthenticatedEvent = true;
            }
    
            const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
    
    Markin Igor's avatar
    Markin Igor committed
            if (authenticationPublicKey) {
              const result = await loadIdentityInternal(authenticationPublicKey, "00000000");
              if (result.code !== "200") {
                const event = createEvent("CanNotGetPincodeForAuthenticatedIdentity", "IdentityNotLoaded", [authenticationPublicKey]);
                parent.onEvent(event);
                clearPinCodeTtl(authenticationPublicKey);
                window.currentlyAuthenticatedIdentity = null;
              }
    
    Markin Igor's avatar
    Markin Igor committed
            identityAuthenticatedEvent = false;
    
    Markin Igor's avatar
    Markin Igor committed
            window.currentlyLoadedIdentity = null;
    
    Markin Igor's avatar
    Markin Igor committed
        if (window.currentlyLoadedIdentity) {
          const pinCode = getPincode(window.currentlyLoadedIdentity.authentication.publicKey);
          if (!pinCode) {
            if (!identityLoadedEvent) {
              const result = await loadIdentityInternal(window.currentlyLoadedIdentity.authentication.publicKey, "00000000");
    
              if (window.currentlyLoadedIdentity && result.code !== "200") {
    
    Markin Igor's avatar
    Markin Igor committed
                const event = createEvent("CanNotLoadPincodeForLoadedIdentity", "IdentityNotLoaded", [window.currentlyLoadedIdentity.authentication.publicKey]);
                parent.onEvent(event);
                identityLoadedEvent = true;
              }
    
    Markin Igor's avatar
    Markin Igor committed
            identityLoadedEvent = false;
    
    Markin Igor's avatar
    Markin Igor committed
        if (window.currentlyAuthenticatedIdentity) {
          const now = new Date().getTime();
          if (now - window.lastTimeGetProfile > 30000) {
            getProfileData(window.currentlyAuthenticatedIdentity);
    
    Markin Igor's avatar
    Markin Igor committed
            window.lastTimeGetProfile = now;
    
    Alexey Lunin's avatar
    Alexey Lunin committed
    
        const currentLocalStorageUUID = localStorage.getItem("uuid");
        const currentLocalStorageToken = localStorage.getItem("token");
        const currentLocalStorageIdentity = localStorage.getItem("authenticatedIdentity");
        if (
          !currentLocalStorageUUID && previousLocalStorageUUID ||
          !currentLocalStorageToken && previousLocalStorageToken ||
          !currentLocalStorageIdentity && previousLocalStorageIdentity
        ) {
          previousLocalStorageUUID = null;
          previousLocalStorageToken = null;
          previousLocalStorageIdentity = null;
    
          destroyAuthentication();
    
          const event = createEvent("", "LogoutExternal");
          parent.onEvent(event);
        } else {
          previousLocalStorageUUID = currentLocalStorageUUID;
          previousLocalStorageToken = currentLocalStorageToken;
          previousLocalStorageIdentity = currentLocalStorageIdentity;
        }
    
    Markin Igor's avatar
    Markin Igor committed
      }, 50);
    
      const getNewEventsWithoutSession = async () => {
        anynomousDeviceKeyEventsProcessing = true;
        try {
          const executeResult = await executeRestfulFunction("public", viamAnonymousApi, viamAnonymousApi.eventGetNewEventsWithoutSession, null, "devicekey");
          if(executeResult.code === "200") {
            const eventsLen = executeResult.data.length;
            let changedMaxDeviceKeyAnonymousEventTime = false;
            for (let i = 0; i < eventsLen; i++) {
              const event = executeResult.data[i];
              switch (event.type) {
                case "DeviceConfirmed" : {
                  await setIdentityInLocalStorage(window.currentlyLoadedIdentity);
                  parent.onEvent(event);
                  break;
    
                case "QRCodeUpdated" : {
    
                  const actionID = event["actionID"];
                  const QrCode = event["payloads"][1];
    
                  const eventCopy = JSON.parse(JSON.stringify(event));
    
    
                  QRCode.toDataURL(actionID + "," + QrCode, function (err, url) {
    
    Markin Igor's avatar
    Markin Igor committed
                    eventCopy["payloads"].push(url);
    
                  break;
                }
    
                case "KeyDeleted" : {
                  const authenticationPublicKey = localStorage.getItem("authenticatedIdentity");
                  clearPinCodeTtl(authenticationPublicKey);
                  localStorage.removeItem("uuid");
                  localStorage.removeItem("token");
                  localStorage.removeItem("authenticatedIdentity");
                  delete window.loadedIdentities[authenticationPublicKey];
                  window.currentlyLoadedIdentity = null;
                  window.currentlyAuthenticatedIdentity = null;
                  window.lastTimeGetProfile = 0;
    
                  destroyIdentityFromLocalStorage(authenticationPublicKey);
                  break;
                }
    
                default : {
    
              changedMaxDeviceKeyAnonymousEventTime = true;
              maxDeviceKeyAnonymousEventTime = Math.max(maxDeviceKeyAnonymousEventTime, event.stamp);
            }
    
            if(changedMaxDeviceKeyAnonymousEventTime) {
              await executeRestfulFunction("public", viamAnonymousApi, viamAnonymousApi.eventUpdateLastViewedWithoutSession,
                null, "devicekey", maxDeviceKeyAnonymousEventTime.toString());
            }
          }
        } catch (e) {
          console.warn(e);
        }
        anynomousDeviceKeyEventsProcessing = false;
      };
    
      const getNewDeviceEvents = async () => {
        eventsDeviceEventsProcessing = true;
        try {
          const executeResult = await executeRestfulFunction("private", viamApi, viamApi.eventGetNewEvents, null, "devicekey");
          if (executeResult.code === "200") {
            const eventsLen = executeResult.data.length;
            const changedMaxDeviceKeyEventTime = false;
            for (let i = 0; i < eventsLen; i++) {
              const event = executeResult.data[i];
              if (event.type === "QRCodeUpdated") {
                const actionID = event["actionID"];
                const QrCode = event["payloads"][1];
    
                const eventCopy = JSON.parse(JSON.stringify(event));
    
                QRCode.toDataURL(actionID + "," + QrCode, function (err, url) {
                  eventCopy["payloads"].push(url);
                  parent.onEvent(eventCopy);
                });
              } else {
                parent.onEvent(event);
    
              maxDeviceKeyEventTime = Math.max(maxDeviceKeyEventTime, event.stamp);
            }
            if(changedMaxDeviceKeyEventTime) {
              await executeRestfulFunction("private", viamApi, viamApi.eventUpdateLastViewed, null, "devicekey",
                maxDeviceKeyEventTime.toString());
    
        } catch (e) {
          console.warn(e);
    
        eventsDeviceEventsProcessing = false;
      };
    
      const getNewEntityEvents = async () => {
        eventsEntityEventsProcessing = true;
        try {
          const executeResult = await executeRestfulFunction("private", viamApi, viamApi.eventGetNewEvents, null, "entity");
    
          if (executeResult.code === "200") {
            const eventsLen = executeResult.data.length;
            let changedMaxEntityEventTime = false;
            for (let i = 0; i < eventsLen; i++) {
              const event = executeResult.data[i];
              if (event.type === "QRCodeUpdated") {
                const actionID = event["actionID"];
                const QrCode = event["payloads"][1];
    
                const eventCopy = JSON.parse(JSON.stringify(event));
    
                QRCode.toDataURL(actionID + "," + QrCode, function (err, url) {
                  eventCopy["payloads"].push(url);
                  parent.onEvent(eventCopy);
                });
    
    
              parent.onEvent(event);
              changedMaxEntityEventTime = true;
              maxEntityEventTime = Math.max(maxEntityEventTime, event.stamp);
            }
            if(changedMaxEntityEventTime) {
              await executeRestfulFunction("private", viamApi, viamApi.eventUpdateLastViewed, null, "entity",
                maxEntityEventTime.toString());
    
        } catch (e) {
          console.warn(e);
        }
        eventsEntityEventsProcessing = false;
      }
    
      setInterval(() => {
        if (window.currentlyLoadedIdentity && !anynomousDeviceKeyEventsProcessing && !window.currentlyAuthenticatedIdentity) {
          getNewEventsWithoutSession();
        }
    
        if (window.currentlyAuthenticatedIdentity) {
          // These functions has to be executed at the same time.
          !eventsDeviceEventsProcessing && getNewDeviceEvents();
          !eventsEntityEventsProcessing && getNewEntityEvents();