Promise returns undefined after switching tabs only

Promise returns undefined after switching tabs only

我正在制作一个 chrome 扩展程序,允许用户在 YouTube 视频上做笔记。笔记使用 IndexedDB 存储。我 运行 遇到一个问题,如果我切换到另一个选项卡然后再切换回来,承诺 returns 未定义。首先,我使用的大部分代码是为了让问题更容易理解。

// background.js

chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
if(request.message === 'get_notes') {
    let getNotes_request = get_records(request.payload)

        getNotes_request.then(res => { // This is where the error occurs, so the above function returns undefined
            chrome.runtime.sendMessage({
                message: 'getNotes_success',
                payload: res
            })
        })
    }
});

function create_database() {
    const request = self.indexedDB.open('MyTestDB');

    request.onerror = function(event) {
        console.log("Problem opening DB.");
    }

    request.onupgradeneeded = function(event) {
        db = event.target.result;

        let objectStore = db.createObjectStore('notes', {
            keypath: "id", autoIncrement: true
        });
        objectStore.createIndex("videoID, videoTime", ["videoID", "videoTime"], {unique: false});
    
        objectStore.transaction.oncomplete = function(event) {
            console.log("ObjectStore Created.");
        }
    }

    request.onsuccess = function(event) {
        db = event.target.result;
        console.log("DB Opened.")
    // Functions to carry out if successfully opened:
    
    // insert_records(notes); This is only done when for the first run, so I will have some notes to use for checking and debugging. The notes are in the form of an array.
    }

}

function get_records(vidID) {
    if(db) {
        const get_transaction = db.transaction("notes", "readonly");
        const objectStore = get_transaction.objectStore("notes");
        const myIndex = objectStore.index("videoID, videoTime");
        console.log("Pre-promise reached!");

        return new Promise((resolve, reject) => {
            get_transaction.oncomplete = function() {
                console.log("All Get Transactions Complete!");
            }

            get_transaction.onerror = function() {
                console.log("Problem Getting Notes!");
            }

            let request = myIndex.getAll(IDBKeyRange.bound([vidID], [vidID, []]));

            request.onsuccess = function(event) {
                console.log(event.target.result);
                resolve(event.target.result);
            }
        });

    }
}

create_database();

现在 popup.js 代码:

chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
    if (request.message === 'getNotes_success') {
        notesTable.innerHTML = ""; // Clear the table body
        if (request.payload) {
            // Code to display the notes.
        } else {
            // Display a message to add notes or refresh the table.
        }
    }
}

chrome.tabs.query({active: true, currentWindow: true}, (tabs) => {
    site = tabs[0].url; // Variables declared earlier, not shown here

    // Valid YouTube video page
    if (isYTVid.test(site)) { // isYTVid is a previously declared regex string

        refNotesbtn.click(); // Auto-click the refresh notes button
    }

refNotesbtn.addEventListener('click', function() {

    videoID = get_videoID();

    chrome.runtime.sendMessage({
        message: 'get_notes',
        payload: videoID
    });
});

我现在的问题是上面显示注释的代码大部分时间都可以正常工作,但是如果我切换到另一个选项卡,然后切换回 YouTube 选项卡并打开扩展程序,检索notes returns 未定义,并显示未找到注释的消息。如果我点击按钮刷新笔记,它们会正确显示。如果插入、编辑或删除功能(此处未显示)发生此错误,可能会给用户带来重大问题,因此我想在继续之前解决它。

我注意到当错误发生时,“Pre-promise reached!” message也不显示,所以是get_notes函数根本没有触发,还是触发后的问题?为代码墙道歉,感谢您的帮助。

由于 db 是异步发现的,因此您需要将其缓存为 Promise-wrapped(例如 const dbPromise = ...)而不是原始的 db。然后,您可以使用 dbPromise.then(db => {...})(或 async/await 等价物)可靠地访问 db

我会这样写。代码中有大量解释性注释。

// background.js

// cache for Promise-wrapped `db` object
let dbPromise = null;

// listener (top-level code)
chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
    if(request.message === 'get_notes') {
        if(!dbPromise) {
            dbPromise = create_database()
            .catch(error => {
                console.log(error);
                throw error;
            });
        } else {
            // do nothing ... dbPromise is already cached.
        }
        dbPromise // read the cached promise and chain .then().then() .
        .then(db => get_records(db, request.payload)) // this is where you were struggling.
        .then(res => {
            chrome.runtime.sendMessage({
                'message': 'getNotes_success',
                'payload': res
            });
        })
        .catch(error => { // catch to avoid unhandled error exception
            console.log(error); // don't re-throw
        });
    } else {
        // do nothing ... presumably.
    }
});

function create_database() {
    return new Promise((resolve, reject) => {
        const request = self.indexedDB.open('MyTestDB'); // scope of self?
        request.onerror = function(event) {
            reject(new Error('Problem opening DB.'));
        }
        request.onupgradeneeded = function(event) {
            let db = event.target.result;
            let objectStore = db.createObjectStore('notes', {
                'keypath': 'id', 
                'autoIncrement': true
            });
            objectStore.createIndex('videoID, videoTime', ['videoID', 'videoTime'], {'unique': false});
            objectStore.transaction.oncomplete = function(event) {
                console.log('ObjectStore Created.');
                resolve(db);
            }
        }
        request.onsuccess = function(event) {
            let db = event.target.result;
            console.log('DB Opened.')
            // Functions to carry out if successfully opened:
            
            // insert_records(notes); This is only done when for the first run, so I will have some notes to use for checking and debugging. The notes are in the form of an array.
            resolve(db);
        }
    });
}

function get_records(db, vidID) {
    // first a couple of low-lwvel utility functions to help keep the high-level code clean.
    function getTransaction = function() {
        const transaction = db.transaction('notes', 'readonly');
        transaction.oncomplete = function() {
            console.log('All Get Transactions Complete!');
        }
        transaction.onerror = function() {
            console.log('Problem Getting Notes!');
        }
        return transaction;
    };
    function getAllAsync = function(transaction) {
        return new Promise((resolve, reject) {
            const objectStore = transaction.objectStore('notes');
            let request = objectStore.index('videoID, videoTime').getAll(IDBKeyRange.bound([vidID], [vidID, []]));
            request.onsuccess = function(event) {
                // console.log(event.target.result);
                resolve(event.target.result);
            }
            request.onerror = function(event) { // presumably
                reject(new Error('getAll request failed'));
            }
        }); 
    };
    return getAllAsync(getTransaction(db));
}

我唯一不确定的部分是 request.onupgradeneededrequest.onsuccess 之间的相互作用。我假设这些事件中的一个或另一个会触发。如果它们按顺序触发(?),那么代码可能需要略有不同。

我建议如下。创建一个简单的承诺 returning 辅助函数以连接到 indexedDB。

function open(name, version, onupgradeneeded) {
  return new Promise((resolve, reject) => {
    const request = indexedDB.open(name, version);
    request.onupgradeneeded = onupgradeneeded;
    request.onerror = event => reject(event.target.error);
    request.onsuccess = event => resolve(event.target.result);
  });
}

修改 get_records 以接受数据库参数作为其第一个参数,始终 return 承诺,并在承诺执行器回调函数中执行所有工作。

function get_records(db, vidID) {
  return new Promise((resolve, reject) => {
    const tx = db.transaction("notes", "readonly");
    const store = tx.objectStore("notes");
    const index = store.index("videoID, videoTime");
    let request = index.getAll(IDBKeyRange.bound([vidID], [vidID, []]));
    request.onsuccess = function(event) {
      console.log(event.target.result);
      resolve(event.target.result);
    };
    request.onerror = event => reject(event.target.error);
  });
}

有事就写作业:

function uponSomethingHappeningSomewhere() {
  open('mydb', 1, myOnUpgradeneeded).then(db => {
    const videoId = 1234;
    return get_records(db, videoId);
  }).then(results => {
    return chrome.runtime.sendMessage({
      'message': 'getNotes_success',
      'payload': res
    });
  }).catch(console.error);
}