2009-08-20 11:56:10 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
|
|
* vim: sw=2 ts=2 et lcs=trail\:.,tab\:>~ :
|
|
|
|
* ***** 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 places test code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
2010-12-02 14:09:52 -08:00
|
|
|
* the Mozilla Foundation.
|
2009-08-20 11:56:10 -07:00
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Shawn Wilsher <me@shawnwilsher.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 ***** */
|
|
|
|
|
|
|
|
#include "TestHarness.h"
|
|
|
|
#include "nsMemory.h"
|
|
|
|
#include "nsThreadUtils.h"
|
|
|
|
#include "nsNetUtil.h"
|
|
|
|
#include "nsDocShellCID.h"
|
|
|
|
|
|
|
|
#include "nsToolkitCompsCID.h"
|
|
|
|
#include "nsINavHistoryService.h"
|
|
|
|
#include "nsIObserverService.h"
|
|
|
|
#include "mozilla/IHistory.h"
|
2010-07-13 18:00:33 -07:00
|
|
|
#include "mozIStorageConnection.h"
|
|
|
|
#include "mozIStorageStatement.h"
|
|
|
|
#include "nsPIPlacesDatabase.h"
|
2011-02-24 05:03:51 -08:00
|
|
|
#include "nsIObserver.h"
|
|
|
|
#include "prinrval.h"
|
|
|
|
|
|
|
|
#define TOPIC_FRECENCY_UPDATED "places-frecency-updated"
|
|
|
|
#define WAITFORTOPIC_TIMEOUT_SECONDS 5
|
2009-08-20 11:56:10 -07:00
|
|
|
|
|
|
|
using namespace mozilla;
|
|
|
|
|
|
|
|
static size_t gTotalTests = 0;
|
|
|
|
static size_t gPassedTests = 0;
|
|
|
|
|
|
|
|
#define do_check_true(aCondition) \
|
|
|
|
PR_BEGIN_MACRO \
|
|
|
|
gTotalTests++; \
|
|
|
|
if (aCondition) { \
|
|
|
|
gPassedTests++; \
|
|
|
|
} else { \
|
2010-12-18 19:30:46 -08:00
|
|
|
fail("%s | Expected true, got false at line %d", __FILE__, __LINE__); \
|
2009-08-20 11:56:10 -07:00
|
|
|
} \
|
|
|
|
PR_END_MACRO
|
|
|
|
|
|
|
|
#define do_check_false(aCondition) \
|
|
|
|
PR_BEGIN_MACRO \
|
|
|
|
gTotalTests++; \
|
|
|
|
if (!aCondition) { \
|
|
|
|
gPassedTests++; \
|
|
|
|
} else { \
|
2010-12-18 19:30:46 -08:00
|
|
|
fail("%s | Expected false, got true at line %d", __FILE__, __LINE__); \
|
2009-08-20 11:56:10 -07:00
|
|
|
} \
|
|
|
|
PR_END_MACRO
|
|
|
|
|
|
|
|
#define do_check_success(aResult) \
|
|
|
|
do_check_true(NS_SUCCEEDED(aResult))
|
|
|
|
|
2010-11-17 14:42:13 -08:00
|
|
|
#ifdef LINUX
|
|
|
|
// XXX Linux opt builds on tinderbox are orange due to linking with stdlib.
|
|
|
|
// This is sad and annoying, but it's a workaround that works.
|
|
|
|
#define do_check_eq(aExpected, aActual) \
|
|
|
|
do_check_true(aExpected == aActual)
|
|
|
|
#else
|
|
|
|
#include <sstream>
|
|
|
|
|
2010-12-02 14:09:52 -08:00
|
|
|
#define do_check_eq(aActual, aExpected) \
|
2010-11-08 11:42:08 -08:00
|
|
|
PR_BEGIN_MACRO \
|
|
|
|
gTotalTests++; \
|
2010-11-17 14:42:13 -08:00
|
|
|
if (aExpected == aActual) { \
|
2010-11-08 11:42:08 -08:00
|
|
|
gPassedTests++; \
|
|
|
|
} else { \
|
|
|
|
std::ostringstream temp; \
|
2010-12-18 19:30:46 -08:00
|
|
|
temp << __FILE__ << " | Expected '" << aExpected << "', got '"; \
|
|
|
|
temp << aActual <<"' at line " << __LINE__; \
|
2010-11-08 11:42:08 -08:00
|
|
|
fail(temp.str().c_str()); \
|
|
|
|
} \
|
|
|
|
PR_END_MACRO
|
2010-11-17 14:42:13 -08:00
|
|
|
#endif
|
2010-02-24 08:37:02 -08:00
|
|
|
|
2009-08-20 11:56:10 -07:00
|
|
|
struct Test
|
|
|
|
{
|
|
|
|
void (*func)(void);
|
|
|
|
const char* const name;
|
|
|
|
};
|
|
|
|
#define TEST(aName) \
|
|
|
|
{aName, #aName}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Runs the next text.
|
|
|
|
*/
|
|
|
|
void run_next_test();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To be used around asynchronous work.
|
|
|
|
*/
|
|
|
|
void do_test_pending();
|
|
|
|
void do_test_finished();
|
|
|
|
|
2011-02-24 05:03:51 -08:00
|
|
|
/**
|
|
|
|
* Spins current thread until a topic is received.
|
|
|
|
*/
|
|
|
|
class WaitForTopicSpinner : public nsIObserver
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
NS_DECL_ISUPPORTS
|
|
|
|
|
|
|
|
WaitForTopicSpinner(const char* const aTopic)
|
|
|
|
: mTopic(aTopic)
|
|
|
|
, mTopicReceived(false)
|
|
|
|
, mStartTime(PR_IntervalNow())
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void Spin() {
|
|
|
|
nsCOMPtr<nsIObserverService> observerService =
|
|
|
|
do_GetService(NS_OBSERVERSERVICE_CONTRACTID);
|
|
|
|
if (observerService) {
|
|
|
|
(void)observerService->AddObserver(this, mTopic, PR_FALSE);
|
|
|
|
|
|
|
|
while (!mTopicReceived) {
|
|
|
|
if (PR_IntervalNow() - mStartTime > WAITFORTOPIC_TIMEOUT_SECONDS * PR_USEC_PER_SEC) {
|
|
|
|
// Timed out waiting for the topic.
|
|
|
|
do_check_true(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
(void)NS_ProcessNextEvent();
|
|
|
|
}
|
|
|
|
|
|
|
|
(void)observerService->RemoveObserver(this, mTopic);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHOD Observe(nsISupports* aSubject,
|
|
|
|
const char* aTopic,
|
|
|
|
const PRUnichar* aData)
|
|
|
|
{
|
|
|
|
do_check_false(strcmp(aTopic, mTopic));
|
|
|
|
mTopicReceived = true;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const char* const mTopic;
|
|
|
|
bool mTopicReceived;
|
|
|
|
PRIntervalTime mStartTime;
|
|
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS1(
|
|
|
|
WaitForTopicSpinner,
|
|
|
|
nsIObserver
|
|
|
|
)
|
|
|
|
|
2009-08-20 11:56:10 -07:00
|
|
|
/**
|
|
|
|
* Adds a URI to the database.
|
|
|
|
*
|
|
|
|
* @param aURI
|
|
|
|
* The URI to add to the database.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
addURI(nsIURI* aURI)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsINavHistoryService> hist =
|
|
|
|
do_GetService(NS_NAVHISTORYSERVICE_CONTRACTID);
|
|
|
|
|
|
|
|
PRInt64 id;
|
|
|
|
nsresult rv = hist->AddVisit(aURI, PR_Now(), nsnull,
|
|
|
|
nsINavHistoryService::TRANSITION_LINK, PR_FALSE,
|
|
|
|
0, &id);
|
|
|
|
do_check_success(rv);
|
2011-02-24 05:03:51 -08:00
|
|
|
|
|
|
|
// Wait for frecency update.
|
|
|
|
nsRefPtr<WaitForTopicSpinner> spinner =
|
|
|
|
new WaitForTopicSpinner(TOPIC_FRECENCY_UPDATED);
|
|
|
|
spinner->Spin();
|
2009-08-20 11:56:10 -07:00
|
|
|
}
|
|
|
|
|
2010-07-13 18:00:33 -07:00
|
|
|
struct PlaceRecord
|
|
|
|
{
|
|
|
|
PRInt64 id;
|
|
|
|
PRInt32 hidden;
|
|
|
|
PRInt32 typed;
|
|
|
|
PRInt32 visitCount;
|
2010-12-02 14:09:52 -08:00
|
|
|
nsCString guid;
|
2010-07-13 18:00:33 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct VisitRecord
|
|
|
|
{
|
|
|
|
PRInt64 id;
|
|
|
|
PRInt64 lastVisitId;
|
|
|
|
PRInt32 transitionType;
|
|
|
|
};
|
|
|
|
|
2009-08-20 11:56:10 -07:00
|
|
|
already_AddRefed<IHistory>
|
|
|
|
do_get_IHistory()
|
|
|
|
{
|
|
|
|
nsCOMPtr<IHistory> history = do_GetService(NS_IHISTORY_CONTRACTID);
|
|
|
|
do_check_true(history);
|
|
|
|
return history.forget();
|
|
|
|
}
|
2010-07-13 18:00:33 -07:00
|
|
|
|
|
|
|
already_AddRefed<nsINavHistoryService>
|
|
|
|
do_get_NavHistory()
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsINavHistoryService> serv =
|
|
|
|
do_GetService(NS_NAVHISTORYSERVICE_CONTRACTID);
|
|
|
|
do_check_true(serv);
|
|
|
|
return serv.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<mozIStorageConnection>
|
|
|
|
do_get_db()
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsINavHistoryService> history = do_get_NavHistory();
|
|
|
|
nsCOMPtr<nsPIPlacesDatabase> database = do_QueryInterface(history);
|
|
|
|
do_check_true(database);
|
|
|
|
|
|
|
|
mozIStorageConnection* dbConn;
|
|
|
|
nsresult rv = database->GetDBConnection(&dbConn);
|
|
|
|
do_check_success(rv);
|
|
|
|
return dbConn;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the place record from the database.
|
|
|
|
*
|
|
|
|
* @param aURI The unique URI of the place we are looking up
|
|
|
|
* @param result Out parameter where the result is stored
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
do_get_place(nsIURI* aURI, PlaceRecord& result)
|
|
|
|
{
|
|
|
|
nsCOMPtr<mozIStorageConnection> dbConn = do_get_db();
|
|
|
|
nsCOMPtr<mozIStorageStatement> stmt;
|
|
|
|
|
|
|
|
nsCString spec;
|
|
|
|
nsresult rv = aURI->GetSpec(spec);
|
|
|
|
do_check_success(rv);
|
|
|
|
|
|
|
|
rv = dbConn->CreateStatement(NS_LITERAL_CSTRING(
|
2010-12-02 14:09:52 -08:00
|
|
|
"SELECT id, hidden, typed, visit_count, guid FROM moz_places "
|
2010-07-13 18:00:33 -07:00
|
|
|
"WHERE url=?1 "
|
|
|
|
), getter_AddRefs(stmt));
|
|
|
|
do_check_success(rv);
|
|
|
|
|
2010-08-09 08:59:59 -07:00
|
|
|
rv = stmt->BindUTF8StringByIndex(0, spec);
|
2010-07-13 18:00:33 -07:00
|
|
|
do_check_success(rv);
|
|
|
|
|
|
|
|
PRBool hasResults;
|
|
|
|
rv = stmt->ExecuteStep(&hasResults);
|
|
|
|
do_check_success(rv);
|
2010-11-12 14:24:25 -08:00
|
|
|
if (!hasResults) {
|
|
|
|
result.id = 0;
|
|
|
|
return;
|
|
|
|
}
|
2010-07-13 18:00:33 -07:00
|
|
|
|
|
|
|
rv = stmt->GetInt64(0, &result.id);
|
|
|
|
do_check_success(rv);
|
|
|
|
rv = stmt->GetInt32(1, &result.hidden);
|
|
|
|
do_check_success(rv);
|
|
|
|
rv = stmt->GetInt32(2, &result.typed);
|
|
|
|
do_check_success(rv);
|
|
|
|
rv = stmt->GetInt32(3, &result.visitCount);
|
|
|
|
do_check_success(rv);
|
2010-12-02 14:09:52 -08:00
|
|
|
rv = stmt->GetUTF8String(4, result.guid);
|
|
|
|
do_check_success(rv);
|
2010-07-13 18:00:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the most recent visit to a place.
|
|
|
|
*
|
|
|
|
* @param placeID ID from the moz_places table
|
|
|
|
* @param result Out parameter where visit is stored
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
do_get_lastVisit(PRInt64 placeId, VisitRecord& result)
|
|
|
|
{
|
|
|
|
nsCOMPtr<mozIStorageConnection> dbConn = do_get_db();
|
|
|
|
nsCOMPtr<mozIStorageStatement> stmt;
|
|
|
|
|
|
|
|
nsresult rv = dbConn->CreateStatement(NS_LITERAL_CSTRING(
|
|
|
|
"SELECT id, from_visit, visit_type FROM moz_historyvisits "
|
|
|
|
"WHERE place_id=?1 "
|
|
|
|
"LIMIT 1"
|
|
|
|
), getter_AddRefs(stmt));
|
|
|
|
do_check_success(rv);
|
|
|
|
|
2010-08-09 08:59:59 -07:00
|
|
|
rv = stmt->BindInt64ByIndex(0, placeId);
|
2010-07-13 18:00:33 -07:00
|
|
|
do_check_success(rv);
|
|
|
|
|
|
|
|
PRBool hasResults;
|
|
|
|
rv = stmt->ExecuteStep(&hasResults);
|
|
|
|
do_check_success(rv);
|
|
|
|
|
2010-11-12 14:24:25 -08:00
|
|
|
if (!hasResults) {
|
|
|
|
result.id = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-07-13 18:00:33 -07:00
|
|
|
rv = stmt->GetInt64(0, &result.id);
|
|
|
|
do_check_success(rv);
|
|
|
|
rv = stmt->GetInt64(1, &result.lastVisitId);
|
|
|
|
do_check_success(rv);
|
|
|
|
rv = stmt->GetInt32(2, &result.transitionType);
|
|
|
|
do_check_success(rv);
|
|
|
|
}
|