зеркало из https://github.com/mozilla/pjs.git
Bug 333794 r=annie.sullivan Crash on shutdown if timer fires.
This commit is contained in:
Родитель
539e5c3c16
Коммит
94f8daf25c
|
@ -1,639 +0,0 @@
|
|||
//* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
||||
/* ***** BEGIN LICENSE BLOCK *****
|
||||
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||||
*
|
||||
* The contents of this file are subject to the Mozilla Public License Version
|
||||
* 1.1 (the "License"); you may not use this file except in compliance with
|
||||
* the License. You may obtain a copy of the License at
|
||||
* http://www.mozilla.org/MPL/
|
||||
*
|
||||
* Software distributed under the License is distributed on an "AS IS" basis,
|
||||
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||||
* for the specific language governing rights and limitations under the
|
||||
* License.
|
||||
*
|
||||
* The Original Code is Mozilla History System
|
||||
*
|
||||
* The Initial Developer of the Original Code is
|
||||
* Google Inc.
|
||||
* Portions created by the Initial Developer are Copyright (C) 2006
|
||||
* the Initial Developer. All Rights Reserved.
|
||||
*
|
||||
* Contributor(s):
|
||||
* Brett Wilson <brettw@gmail.com> (original author)
|
||||
*
|
||||
* Alternatively, the contents of this file may be used under the terms of
|
||||
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||||
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||||
* in which case the provisions of the GPL or the LGPL are applicable instead
|
||||
* of those above. If you wish to allow use of your version of this file only
|
||||
* under the terms of either the GPL or the LGPL, and not to allow others to
|
||||
* use your version of this file under the terms of the MPL, indicate your
|
||||
* decision by deleting the provisions above and replace them with the notice
|
||||
* and other provisions required by the GPL or the LGPL. If you do not delete
|
||||
* the provisions above, a recipient may use your version of this file under
|
||||
* the terms of any one of the MPL, the GPL or the LGPL.
|
||||
*
|
||||
* ***** END LICENSE BLOCK ***** */
|
||||
|
||||
/**
|
||||
* This class handles expiration of history for nsNavHistory. There is a 1-1
|
||||
* mapping between nsNavHistory class and a nsNavHistoryExpire class, the
|
||||
* code is separated for better understandability.
|
||||
*/
|
||||
|
||||
#include "nsNavHistory.h"
|
||||
#include "mozStorageHelper.h"
|
||||
#include "nsNetUtil.h"
|
||||
|
||||
struct nsNavHistoryExpireRecord {
|
||||
nsNavHistoryExpireRecord(mozIStorageStatement* statement);
|
||||
|
||||
PRInt64 visitID;
|
||||
PRInt64 pageID;
|
||||
PRTime visitDate;
|
||||
nsCString uri;
|
||||
PRInt64 faviconID;
|
||||
PRBool hidden;
|
||||
PRBool bookmarked;
|
||||
PRBool erased; // set to true if/when the history entry is erased
|
||||
};
|
||||
|
||||
// Number of things we'll expire at once. Runtime of expiration is approximately
|
||||
// linear with the number of things we expire at once. This number was picked so
|
||||
// we expire "several" things at once, but still run quickly. Just doing 3
|
||||
// expirations at once isn't much faster than 6 due to constant overhead of
|
||||
// running the query.
|
||||
#define EXPIRATION_COUNT_PER_RUN 6
|
||||
|
||||
// The time in ms to wait after AddURI to try expiration of pages. Short is
|
||||
// actually better. If expiration takes an unusually long period of time, it
|
||||
// will interfere with video playback in the browser, for example. Such a blip
|
||||
// is not likely to be noticable when the page has just appeared.
|
||||
#define PARTIAL_EXPIRATION_TIMEOUT 3500
|
||||
|
||||
// The time in ms to wait after the initial expiration run for additional ones
|
||||
#define SUBSEQUENT_EXIPRATION_TIMEOUT 20000
|
||||
|
||||
// Number of expirations we'll do after the most recent page is loaded before
|
||||
// stopping. We don't want to keep the computer chugging forever expiring
|
||||
// annotations if the user stopped using the browser.
|
||||
//
|
||||
// This current value of one prevents history expiration while the page is
|
||||
// being shown, because expiration may interfere with media playback.
|
||||
#define MAX_SEQUENTIAL_RUNS 1
|
||||
|
||||
|
||||
// nsNavHistoryExpire::nsNavHistoryExpire
|
||||
//
|
||||
// Warning: don't do anything with aHistory in the constructor, since
|
||||
// this is a member of the nsNavHistory, it is still being constructed
|
||||
// when this is called.
|
||||
|
||||
nsNavHistoryExpire::nsNavHistoryExpire(nsNavHistory* aHistory) :
|
||||
mHistory(aHistory),
|
||||
mSequentialRuns(0),
|
||||
mTimerSet(PR_FALSE),
|
||||
mAnyEmptyRuns(PR_FALSE),
|
||||
mNextExpirationTime(0),
|
||||
mAddCount(0),
|
||||
mExpiredItems(0),
|
||||
mExpireRuns(0)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::~nsNavHistoryExpire
|
||||
|
||||
nsNavHistoryExpire::~nsNavHistoryExpire()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::OnAddURI
|
||||
//
|
||||
// Called by history when a URI is added to history. This starts the timer
|
||||
// for when we are going to expire.
|
||||
//
|
||||
// The current time is passed in by the history service as an optimization.
|
||||
// The AddURI function has already computed the proper time, and getting the
|
||||
// time again from the OS is nontrivial.
|
||||
|
||||
void
|
||||
nsNavHistoryExpire::OnAddURI(PRTime aNow)
|
||||
{
|
||||
mAddCount ++;
|
||||
mSequentialRuns = 0;
|
||||
|
||||
if (mTimer && mTimerSet) {
|
||||
mTimer->Cancel();
|
||||
mTimerSet = PR_FALSE;
|
||||
}
|
||||
|
||||
if (mNextExpirationTime != 0 && aNow < mNextExpirationTime)
|
||||
return; // we know there's nothing to expire yet
|
||||
|
||||
StartTimer(PARTIAL_EXPIRATION_TIMEOUT);
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::OnQuit
|
||||
//
|
||||
// Here we check for some edge cases and fix them
|
||||
|
||||
void
|
||||
nsNavHistoryExpire::OnQuit()
|
||||
{
|
||||
mozIStorageConnection* connection = mHistory->GetStorageConnection();
|
||||
if (! connection) {
|
||||
NS_NOTREACHED("No connection");
|
||||
return;
|
||||
}
|
||||
|
||||
// Handle degenerate runs:
|
||||
ExpireForDegenerateRuns();
|
||||
|
||||
// vacuum up dangling items
|
||||
ExpireHistoryParanoid(connection);
|
||||
ExpireFaviconsParanoid(connection);
|
||||
ExpireAnnotationsParanoid(connection);
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ClearHistory
|
||||
//
|
||||
// Performance: ExpireItems sends notifications. We may want to disable this
|
||||
// for clear history cases. However, my initial tests show that the
|
||||
// notifications are not a significant part of clear history time.
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::ClearHistory()
|
||||
{
|
||||
PRBool keepGoing;
|
||||
|
||||
mozIStorageConnection* connection = mHistory->GetStorageConnection();
|
||||
NS_ENSURE_TRUE(connection, NS_ERROR_OUT_OF_MEMORY);
|
||||
|
||||
ExpireItems(0, &keepGoing);
|
||||
|
||||
ExpireHistoryParanoid(connection);
|
||||
ExpireFaviconsParanoid(connection);
|
||||
ExpireAnnotationsParanoid(connection);
|
||||
|
||||
ENUMERATE_WEAKARRAY(mHistory->mObservers, nsINavHistoryObserver,
|
||||
OnClearHistory())
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::OnExpirationChanged
|
||||
//
|
||||
// Called when the expiration length in days has changed. We clear any
|
||||
// next expiration time, meaning that we'll try to expire stuff next time,
|
||||
// and recompute the value if there's still nothing to expire.
|
||||
|
||||
void
|
||||
nsNavHistoryExpire::OnExpirationChanged()
|
||||
{
|
||||
mNextExpirationTime = 0;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::DoPartialExpiration
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::DoPartialExpiration()
|
||||
{
|
||||
mSequentialRuns ++;
|
||||
|
||||
PRBool keepGoing;
|
||||
ExpireItems(EXPIRATION_COUNT_PER_RUN, &keepGoing);
|
||||
|
||||
if (keepGoing && mSequentialRuns < MAX_SEQUENTIAL_RUNS)
|
||||
StartTimer(SUBSEQUENT_EXIPRATION_TIMEOUT);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ExpireItems
|
||||
//
|
||||
// Here, we try to expire aNumToExpire items and their associated data,
|
||||
// If we expired things and then stopped because we hit this limit,
|
||||
// aKeepGoing will be set indicating we should keep expiring. If we ran
|
||||
// out of things to expire, it will be unset indicating we should wait.
|
||||
//
|
||||
// As a special case, aNumToExpire can be 0 and we'll expire everything
|
||||
// in history.
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::ExpireItems(PRUint32 aNumToExpire, PRBool* aKeepGoing)
|
||||
{
|
||||
// mark how many times we've been able to run
|
||||
mExpireRuns ++;
|
||||
|
||||
mozIStorageConnection* connection = mHistory->GetStorageConnection();
|
||||
NS_ENSURE_TRUE(connection, NS_ERROR_OUT_OF_MEMORY);
|
||||
|
||||
// This transaction is important for performance. It makes the DB flush
|
||||
// everything to disk in one larger operation rather than many small ones.
|
||||
// Note that this transaction always commits.
|
||||
mozStorageTransaction transaction(connection, PR_TRUE);
|
||||
|
||||
*aKeepGoing = PR_TRUE;
|
||||
|
||||
PRInt64 expireTime;
|
||||
if (aNumToExpire == 0) {
|
||||
// special case: erase all history
|
||||
expireTime = 0;
|
||||
} else {
|
||||
expireTime = PR_Now() - GetExpirationTimeAgo();
|
||||
}
|
||||
|
||||
// find some visits to expire
|
||||
nsTArray<nsNavHistoryExpireRecord> expiredVisits;
|
||||
nsresult rv = FindVisits(expireTime, aNumToExpire, connection,
|
||||
expiredVisits);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// if we didn't find the as many things to expire as we could have, then
|
||||
// we should note the next time we need to expire.
|
||||
if (expiredVisits.Length() < aNumToExpire) {
|
||||
*aKeepGoing = PR_FALSE;
|
||||
ComputeNextExpirationTime(connection);
|
||||
|
||||
if (expiredVisits.Length() == 0) {
|
||||
// Nothing to expire. Set the flag so we know we don't have to do any
|
||||
// work on shutdown.
|
||||
mAnyEmptyRuns = PR_TRUE;
|
||||
return NS_OK;
|
||||
}
|
||||
}
|
||||
mExpiredItems += expiredVisits.Length();
|
||||
|
||||
rv = EraseVisits(connection, expiredVisits);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
rv = EraseHistory(connection, expiredVisits);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// send observer messages
|
||||
nsCOMPtr<nsIURI> uri;
|
||||
for (PRUint32 i = 0; i < expiredVisits.Length(); i ++) {
|
||||
rv = NS_NewURI(getter_AddRefs(uri), expiredVisits[i].uri);
|
||||
if (NS_FAILED(rv)) continue;
|
||||
|
||||
// FIXME bug 325241 provide a way to observe hidden elements
|
||||
if (expiredVisits[i].hidden) continue;
|
||||
|
||||
ENUMERATE_WEAKARRAY(mHistory->mObservers, nsINavHistoryObserver,
|
||||
OnPageExpired(uri, expiredVisits[i].visitDate,
|
||||
expiredVisits[i].erased));
|
||||
}
|
||||
|
||||
// don't worry about errors here, it doesn't affect out ability to continue
|
||||
EraseFavicons(connection, expiredVisits);
|
||||
EraseAnnotations(connection, expiredVisits);
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpireRecord::nsNavHistoryExpireRecord
|
||||
//
|
||||
// Statement should be the one created in FindVisits. The parameters must
|
||||
// agree.
|
||||
|
||||
nsNavHistoryExpireRecord::nsNavHistoryExpireRecord(
|
||||
mozIStorageStatement* statement)
|
||||
{
|
||||
visitID = statement->AsInt64(0);
|
||||
pageID = statement->AsInt64(1);
|
||||
visitDate = statement->AsInt64(2);
|
||||
statement->GetUTF8String(3, uri);
|
||||
faviconID = statement->AsInt64(4);
|
||||
hidden = (statement->AsInt32(5) > 0);
|
||||
bookmarked = (statement->AsInt32(6) > 0);
|
||||
erased = PR_FALSE;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::FindVisits
|
||||
//
|
||||
// aExpireThreshold is the time at which we will delete visits before.
|
||||
// If it is zero, we will not use a threshold and will match everything.
|
||||
//
|
||||
// aNumToExpire is the maximum number of visits to find. If it is 0, then
|
||||
// we will get all matching visits.
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::FindVisits(PRTime aExpireThreshold, PRUint32 aNumToExpire,
|
||||
mozIStorageConnection* aConnection,
|
||||
nsTArray<nsNavHistoryExpireRecord>& aRecords)
|
||||
{
|
||||
nsresult rv;
|
||||
|
||||
// get info for expiring visits, special case no threshold so there is no
|
||||
// SQL parameter
|
||||
nsCOMPtr<mozIStorageStatement> selectStatement;
|
||||
nsCString sql;
|
||||
sql.AssignLiteral("SELECT "
|
||||
"v.visit_id, v.page_id, v.visit_date, h.url, h.favicon, h.hidden, b.item_child "
|
||||
"FROM moz_historyvisit v LEFT JOIN moz_history h ON v.page_id = h.id "
|
||||
"LEFT OUTER JOIN moz_bookmarks b on v.page_id = b.item_child");
|
||||
if (aExpireThreshold != 0)
|
||||
sql.AppendLiteral(" WHERE visit_date < ?1");
|
||||
rv = aConnection->CreateStatement(sql, getter_AddRefs(selectStatement));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
if (aExpireThreshold != 0) {
|
||||
rv = selectStatement->BindInt64Parameter(0, aExpireThreshold);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
}
|
||||
|
||||
PRBool hasMore = PR_FALSE;
|
||||
while (NS_SUCCEEDED(selectStatement->ExecuteStep(&hasMore)) && hasMore &&
|
||||
(aNumToExpire == 0 || aRecords.Length() < aNumToExpire)) {
|
||||
nsNavHistoryExpireRecord record(selectStatement);
|
||||
aRecords.AppendElement(record);
|
||||
}
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::EraseVisits
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::EraseVisits(mozIStorageConnection* aConnection,
|
||||
const nsTArray<nsNavHistoryExpireRecord>& aRecords)
|
||||
{
|
||||
nsCOMPtr<mozIStorageStatement> deleteStatement;
|
||||
nsresult rv = aConnection->CreateStatement(NS_LITERAL_CSTRING(
|
||||
"DELETE FROM moz_historyvisit WHERE visit_id = ?1"),
|
||||
getter_AddRefs(deleteStatement));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
PRUint32 i;
|
||||
for (i = 0; i < aRecords.Length(); i ++) {
|
||||
deleteStatement->BindInt64Parameter(0, aRecords[i].visitID);
|
||||
rv = deleteStatement->Execute();
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
}
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::EraseHistory
|
||||
//
|
||||
// This erases records in moz_history when there are no more visits.
|
||||
// We need to be careful not to delete bookmarks and place:URIs.
|
||||
//
|
||||
// This will modify the input by setting the erased flag on each of the
|
||||
// array elements according to whether the history item was erased or not.
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::EraseHistory(mozIStorageConnection* aConnection,
|
||||
nsTArray<nsNavHistoryExpireRecord>& aRecords)
|
||||
{
|
||||
nsCOMPtr<mozIStorageStatement> deleteStatement;
|
||||
nsresult rv = aConnection->CreateStatement(NS_LITERAL_CSTRING(
|
||||
"DELETE FROM moz_history WHERE id = ?1"),
|
||||
getter_AddRefs(deleteStatement));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
nsCOMPtr<mozIStorageStatement> selectStatement;
|
||||
rv = aConnection->CreateStatement(NS_LITERAL_CSTRING(
|
||||
"SELECT page_id FROM moz_historyvisit WHERE page_id = ?1"),
|
||||
getter_AddRefs(selectStatement));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
for (PRUint32 i = 0; i < aRecords.Length(); i ++) {
|
||||
if (aRecords[i].bookmarked)
|
||||
continue; // don't delete bookmarked entries
|
||||
if (StringBeginsWith(aRecords[i].uri, NS_LITERAL_CSTRING("place:")))
|
||||
continue; // don't delete "place" URIs
|
||||
|
||||
// check that there are no visits
|
||||
rv = selectStatement->BindInt64Parameter(0, aRecords[i].pageID);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
PRBool hasVisit = PR_FALSE;
|
||||
rv = selectStatement->ExecuteStep(&hasVisit);
|
||||
selectStatement->Reset();
|
||||
if (hasVisit) continue;
|
||||
|
||||
aRecords[i].erased = PR_TRUE;
|
||||
rv = deleteStatement->BindInt64Parameter(0, aRecords[i].pageID);
|
||||
rv = deleteStatement->Execute();
|
||||
}
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::EraseFavicons
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::EraseFavicons(mozIStorageConnection* aConnection,
|
||||
const nsTArray<nsNavHistoryExpireRecord>& aRecords)
|
||||
{
|
||||
// see if this favicon still has an entry
|
||||
nsCOMPtr<mozIStorageStatement> selectStatement;
|
||||
nsresult rv = aConnection->CreateStatement(NS_LITERAL_CSTRING(
|
||||
"SELECT id FROM moz_history where favicon = ?1"),
|
||||
getter_AddRefs(selectStatement));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
// delete a favicon
|
||||
nsCOMPtr<mozIStorageStatement> deleteStatement;
|
||||
rv = aConnection->CreateStatement(NS_LITERAL_CSTRING(
|
||||
"DELETE FROM moz_favicon WHERE id = ?1"),
|
||||
getter_AddRefs(deleteStatement));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
|
||||
for (PRUint32 i = 0; i < aRecords.Length(); i ++) {
|
||||
if (! aRecords[i].erased)
|
||||
continue; // main entry not expired
|
||||
if (aRecords[i].faviconID == 0)
|
||||
continue; // no favicon
|
||||
selectStatement->BindInt64Parameter(0, aRecords[i].faviconID);
|
||||
|
||||
// see if there are any history entries and skip if so
|
||||
PRBool hasEntry;
|
||||
if (NS_SUCCEEDED(selectStatement->ExecuteStep(&hasEntry)) && hasEntry) {
|
||||
selectStatement->Reset();
|
||||
continue; // favicon still referenced
|
||||
}
|
||||
selectStatement->Reset();
|
||||
|
||||
// delete the favicon, ignoring errors. We could have the same favicon
|
||||
// referenced twice in our list, and we'd try to delete it twice.
|
||||
deleteStatement->BindInt64Parameter(0, aRecords[i].faviconID);
|
||||
deleteStatement->Execute();
|
||||
}
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::EraseAnnotations
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::EraseAnnotations(mozIStorageConnection* aConnection,
|
||||
const nsTArray<nsNavHistoryExpireRecord>& aRecords)
|
||||
{
|
||||
// FIXME bug 319455 expire annotations
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ExpireHistoryParanoid
|
||||
//
|
||||
// Deletes any dangling history entries that aren't associated with any
|
||||
// visits or bookmarks. Also, special case "place:" URIs.
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::ExpireHistoryParanoid(mozIStorageConnection* aConnection)
|
||||
{
|
||||
// delete history entries with no visits that are not bookmarked
|
||||
// also never delete any "place:" URIs (see function header comment)
|
||||
nsresult rv = aConnection->ExecuteSimpleSQL(NS_LITERAL_CSTRING(
|
||||
"DELETE FROM moz_history WHERE id IN (SELECT id FROM moz_history h "
|
||||
"LEFT OUTER JOIN moz_historyvisit v ON h.id = v.page_id "
|
||||
"LEFT OUTER JOIN moz_bookmarks b ON h.id = b.item_child "
|
||||
"WHERE v.visit_id IS NULL "
|
||||
"AND b.item_child IS NULL "
|
||||
"AND SUBSTR(url,0,6) <> 'place:')"));
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ExpireFaviconsParanoid
|
||||
//
|
||||
// Deletes any dangling favicons that aren't associated with any pages.
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::ExpireFaviconsParanoid(mozIStorageConnection* aConnection)
|
||||
{
|
||||
return aConnection->ExecuteSimpleSQL(NS_LITERAL_CSTRING(
|
||||
"DELETE FROM moz_favicon WHERE id IN "
|
||||
"(SELECT f.id FROM moz_favicon f "
|
||||
"LEFT OUTER JOIN moz_history h ON f.id = h.favicon "
|
||||
"WHERE h.favicon IS NULL)"));
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ExpireAnnotationsParanoid
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::ExpireAnnotationsParanoid(mozIStorageConnection* aConnection)
|
||||
{
|
||||
// FIXME bug 319455 expire annotations
|
||||
// Also remember to expire unused names in moz_anno_name
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ExpireForDegenerateRuns
|
||||
//
|
||||
// This checks for potentiall degenerate runs. For example, a tinderbox
|
||||
// loads many web pages quickly and we'll never have a chance to expire.
|
||||
// Particularly crazy users might also do this. If we detect this, then we
|
||||
// want to force some expiration so history doesn't keep increasing.
|
||||
//
|
||||
// Returns true if we did anything.
|
||||
|
||||
PRBool
|
||||
nsNavHistoryExpire::ExpireForDegenerateRuns()
|
||||
{
|
||||
// If there were any times that we didn't have anything to expire, this is
|
||||
// not a degenerate run.
|
||||
if (mAnyEmptyRuns)
|
||||
return PR_FALSE;
|
||||
|
||||
// If very few URIs were added this run, or we expired more items than we
|
||||
// added, don't worry about it
|
||||
if (mAddCount < 10 || mAddCount < mExpiredItems)
|
||||
return PR_FALSE;
|
||||
|
||||
// This run looks suspicious, try to expire up to the number of items
|
||||
// we may have missed this session.
|
||||
PRBool keepGoing;
|
||||
ExpireItems(mAddCount - mExpiredItems, &keepGoing);
|
||||
return PR_TRUE;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::ComputeNextExpirationTime
|
||||
//
|
||||
// This computes mNextExpirationTime. See that var in the header file.
|
||||
// It is passed the number of microseconds that things expire in.
|
||||
|
||||
void
|
||||
nsNavHistoryExpire::ComputeNextExpirationTime(
|
||||
mozIStorageConnection* aConnection)
|
||||
{
|
||||
mNextExpirationTime = 0;
|
||||
|
||||
nsCOMPtr<mozIStorageStatement> statement;
|
||||
nsresult rv = aConnection->CreateStatement(NS_LITERAL_CSTRING(
|
||||
"SELECT MIN(visit_date) FROM moz_historyvisit"),
|
||||
getter_AddRefs(statement));
|
||||
NS_ASSERTION(NS_SUCCEEDED(rv), "Could not create statement");
|
||||
if (NS_FAILED(rv)) return;
|
||||
|
||||
PRBool hasMore;
|
||||
rv = statement->ExecuteStep(&hasMore);
|
||||
if (NS_FAILED(rv) || ! hasMore)
|
||||
return; // no items, we'll leave mNextExpirationTime = 0 and try to expire
|
||||
// again next time
|
||||
|
||||
PRTime minTime = statement->AsInt64(0);
|
||||
mNextExpirationTime = minTime + GetExpirationTimeAgo();
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::StartTimer
|
||||
|
||||
nsresult
|
||||
nsNavHistoryExpire::StartTimer(PRUint32 aMilleseconds)
|
||||
{
|
||||
if (! mTimer)
|
||||
mTimer = do_CreateInstance("@mozilla.org/timer;1");
|
||||
NS_ENSURE_STATE(mTimer); // returns on error
|
||||
nsresult rv = mTimer->InitWithFuncCallback(TimerCallback, this,
|
||||
aMilleseconds,
|
||||
nsITimer::TYPE_ONE_SHOT);
|
||||
NS_ENSURE_SUCCESS(rv, rv);
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::TimerCallback
|
||||
|
||||
void // static
|
||||
nsNavHistoryExpire::TimerCallback(nsITimer* aTimer, void* aClosure)
|
||||
{
|
||||
nsNavHistoryExpire* that = NS_STATIC_CAST(nsNavHistoryExpire*, aClosure);
|
||||
that->mTimerSet = PR_FALSE;
|
||||
that->DoPartialExpiration();
|
||||
}
|
||||
|
||||
|
||||
// nsNavHistoryExpire::GetExpirationTimeAgo
|
||||
|
||||
PRTime
|
||||
nsNavHistoryExpire::GetExpirationTimeAgo()
|
||||
{
|
||||
PRInt64 expireDays = mHistory->mExpireDays;
|
||||
|
||||
// Prevent Int64 overflow for people that type in huge numbers.
|
||||
// This number is 2^63 / 24 / 60 / 60 / 1000000 (reversing the math below)
|
||||
const PRInt64 maxDays = 106751991;
|
||||
if (expireDays > maxDays)
|
||||
expireDays = maxDays;
|
||||
|
||||
// compute how long ago to expire from
|
||||
const PRInt64 secsPerDay = 24*60*60;
|
||||
const PRInt64 usecsPerSec = 1000000;
|
||||
const PRInt64 usecsPerDay = secsPerDay * usecsPerSec;
|
||||
return expireDays * usecsPerDay;
|
||||
}
|
Загрузка…
Ссылка в новой задаче