6

I am using following code to fetch hierarchical data from Web SQL Database:

...
function getResult(query, data, callback){
    db.transaction(function(tx) {
        tx.executeSql(query, data, function(tx, result) {
        callback(result);
        });
    });
}

function findChildren(id){
    getResult("SELECT * FROM my_table WHERE parent_id=?", [id], function(result){
        for (var i = 0, item = null; i < result.rows.length; i++) {
            item = result.rows.item(i);
            data.push(item);
            findChildren(item.id);
        }
    });
}
var data = Array();
getResult("SELECT * FROM my_table WHERE name like ?", ["A"], function(result){
    for (var i = 0, item = null; i < result.rows.length; i++) {
        item = result.rows.item(i);
        data.push(item);
        findChildren(item.id);
    }
});
...

How can I detect if the execution has been completed?

Naveed
  • 1,191
  • 10
  • 22
  • 1
    Without a callback in `findChildren` there is no chance. Also, you are calling them in a loop, too, which complicates things. Have you looked into Promises? – Bergi Nov 19 '12 at 23:49
  • Well, what is the real problem? What are you trying to do? – phant0m Nov 20 '12 at 09:33
  • _Promises_ seems to be the solution, I am trying this out. – Naveed Nov 20 '12 at 09:41
  • @phant0m I push all the data in an array and want to use that at the end. – Naveed Nov 20 '12 at 09:42
  • 1
    As you haven't given much detail about the implementation it's difficult to say for sure, but it seems like you could do all this with a single more sophisticated SQL query - cutting out the current problem, reducing traffic, and improving performance. – tomfumb Nov 21 '12 at 00:05
  • I have added code in place of dots to make it more clear. – Naveed Nov 21 '12 at 07:55

1 Answers1

3

Use a callback for findChildren and a counter for open transactions:

function findChildren(id, callback){
    getResult("SELECT * FROM my_table WHERE parent_id=?", [id], function(result){
        var results = [],
            results.finished = 0;
            len = result.rows.length;
        for (var i = 0; i < len; i++) (function(i) {
            var item = result.rows.item(i);
            ...
            ...
            findChildren(item.id, function(result) {
                results[i] = result;
                if (++results.finished == len)
                    callback(results);
            });
        })(i);
    });
}

getResult("SELECT * FROM my_table WHERE name like ?", ["A"], function(result){
    var results = [],
        results.finished = 0, 
        len = result.rows.length;
    for (var i = 0; i < len; i++) (function(i) {
        var item = result.rows.item(i);
        ...
        ...
        findChildren(item.id, function(result) {
            results[i] = result;
            if (++results.finished == len) {
                // now results contains a nice tree of arrays with children ids
                // do something with it
            }
        });
    })(i);
});

Promises will abstract the counter and simplify the callback handling. Also, as your two queries are very similiar, you might want to abstract them in a common function, too.

Bergi
  • 630,263
  • 148
  • 957
  • 1,375