2021-12-23 13:57:08 +03:00
/*
* Copyright ( C ) by Claudio Cambra < claudio . cambra @ nextcloud . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License
* for more details .
*/
# include "gui/tray/activitylistmodel.h"
# include "account.h"
# include "accountstate.h"
# include "accountmanager.h"
# include "syncenginetestutils.h"
# include "syncresult.h"
# include <QAbstractItemModelTester>
# include <QDesktopServices>
# include <QSignalSpy>
# include <QTest>
2022-01-04 17:28:26 +03:00
namespace {
2021-12-23 13:57:08 +03:00
constexpr auto startingId = 90000 ;
2022-01-04 17:28:26 +03:00
}
2021-12-23 13:57:08 +03:00
static QByteArray fake404Response = R " (
{ " ocs " : { " meta " : { " status " : " failure " , " statuscode " : 404 , " message " : " Invalid query, please check the syntax. API specifications are here: http: \ / \ /www.freedesktop.org \ /wiki \ /Specifications \ /open-collaboration-services. \n " } , " data " : [ ] } }
) " ;
static QByteArray fake400Response = R " (
{ " ocs " : { " meta " : { " status " : " failure " , " statuscode " : 400 , " message " : " Parameter is incorrect. \n " } , " data " : [ ] } }
) " ;
static QByteArray fake500Response = R " (
{ " ocs " : { " meta " : { " status " : " failure " , " statuscode " : 500 , " message " : " Internal Server Error. \n " } , " data " : [ ] } }
) " ;
class FakeRemoteActivityStorage
{
FakeRemoteActivityStorage ( ) = default ;
public :
static FakeRemoteActivityStorage * instance ( )
{
if ( ! _instance ) {
_instance = new FakeRemoteActivityStorage ( ) ;
_instance - > init ( ) ;
}
return _instance ;
}
static void destroy ( )
{
if ( _instance ) {
delete _instance ;
}
_instance = nullptr ;
}
void init ( )
{
if ( ! _activityData . isEmpty ( ) ) {
return ;
}
_metaSuccess = { { QStringLiteral ( " status " ) , QStringLiteral ( " ok " ) } , { QStringLiteral ( " statuscode " ) , 200 } ,
{ QStringLiteral ( " message " ) , QStringLiteral ( " OK " ) } } ;
initActivityData ( ) ;
}
void initActivityData ( )
{
// Insert activity data
for ( quint32 i = 0 ; i < = _numItemsToInsert ; i + + ) {
QJsonObject activity ;
activity . insert ( QStringLiteral ( " object_type " ) , " files " ) ;
activity . insert ( QStringLiteral ( " activity_id " ) , _startingId ) ;
activity . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " file " ) ) ;
activity . insert ( QStringLiteral ( " subject " ) , QStringLiteral ( " You created %1.txt " ) . arg ( i ) ) ;
activity . insert ( QStringLiteral ( " message " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " object_name " ) , QStringLiteral ( " %1.txt " ) . arg ( i ) ) ;
activity . insert ( QStringLiteral ( " datetime " ) , QDateTime : : currentDateTime ( ) . toString ( Qt : : ISODate ) ) ;
activity . insert ( QStringLiteral ( " icon " ) , QStringLiteral ( " http://example.de/apps/files/img/add-color.svg " ) ) ;
_activityData . push_back ( activity ) ;
2022-01-04 17:28:26 +03:00
_startingId + + ;
2021-12-23 13:57:08 +03:00
}
// Insert notification data
for ( quint32 i = 0 ; i < _numItemsToInsert ; i + + ) {
QJsonObject activity ;
activity . insert ( QStringLiteral ( " activity_id " ) , _startingId ) ;
activity . insert ( QStringLiteral ( " object_type " ) , " calendar " ) ;
activity . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " calendar-event " ) ) ;
2022-01-04 17:28:26 +03:00
activity . insert (
QStringLiteral ( " subject " ) , QStringLiteral ( " You created event %1 in calendar Events " ) . arg ( i ) ) ;
2021-12-23 13:57:08 +03:00
activity . insert ( QStringLiteral ( " message " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " object_name " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " datetime " ) , QDateTime : : currentDateTime ( ) . toString ( Qt : : ISODate ) ) ;
activity . insert ( QStringLiteral ( " icon " ) , QStringLiteral ( " http://example.de/core/img/places/calendar.svg " ) ) ;
2022-01-04 17:28:26 +03:00
QJsonArray actionsArray ;
QJsonObject secondaryAction ;
secondaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Dismiss " ) ) ;
secondaryAction . insert ( QStringLiteral ( " link " ) ,
QString ( QStringLiteral ( " http://cloud.example.de/remote.php/dav " )
+ QStringLiteral ( " ocs/v2.php/apps/notifications/api/v2/notifications " ) + QString : : number ( i ) ) ) ;
secondaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " DELETE " ) ) ;
secondaryAction . insert ( QStringLiteral ( " primary " ) , false ) ;
actionsArray . push_back ( secondaryAction ) ;
_activityData . push_back ( activity ) ;
_startingId + + ;
}
// Insert notification data
for ( quint32 i = 0 ; i < _numItemsToInsert ; i + + ) {
QJsonObject activity ;
activity . insert ( QStringLiteral ( " activity_id " ) , _startingId ) ;
activity . insert ( QStringLiteral ( " object_type " ) , " chat " ) ;
activity . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " chat " ) ) ;
activity . insert ( QStringLiteral ( " subject " ) , QStringLiteral ( " You have received %1's message " ) . arg ( i ) ) ;
activity . insert ( QStringLiteral ( " message " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " object_name " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " datetime " ) , QDateTime : : currentDateTime ( ) . toString ( Qt : : ISODate ) ) ;
activity . insert ( QStringLiteral ( " icon " ) , QStringLiteral ( " http://example.de/core/img/places/talk.svg " ) ) ;
QJsonArray actionsArray ;
QJsonObject primaryAction ;
primaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " View chat " ) ) ;
primaryAction . insert ( QStringLiteral ( " link " ) , QStringLiteral ( " http://cloud.example.de/call/9p4vjdzd " ) ) ;
primaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " WEB " ) ) ;
primaryAction . insert ( QStringLiteral ( " primary " ) , true ) ;
actionsArray . push_back ( primaryAction ) ;
QJsonObject secondaryAction ;
secondaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Dismiss " ) ) ;
secondaryAction . insert ( QStringLiteral ( " link " ) ,
QString ( QStringLiteral ( " http://cloud.example.de/remote.php/dav " )
+ QStringLiteral ( " ocs/v2.php/apps/notifications/api/v2/notifications " ) + QString : : number ( i ) ) ) ;
secondaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " DELETE " ) ) ;
secondaryAction . insert ( QStringLiteral ( " primary " ) , false ) ;
actionsArray . push_back ( secondaryAction ) ;
QJsonObject additionalAction ;
additionalAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Additional 1 " ) ) ;
additionalAction . insert ( QStringLiteral ( " link " ) , QStringLiteral ( " http://cloud.example.de/call/9p4vjdzd " ) ) ;
additionalAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " POST " ) ) ;
additionalAction . insert ( QStringLiteral ( " primary " ) , false ) ;
actionsArray . push_back ( additionalAction ) ;
additionalAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Additional 2 " ) ) ;
actionsArray . push_back ( additionalAction ) ;
activity . insert ( QStringLiteral ( " actions " ) , actionsArray ) ;
_activityData . push_back ( activity ) ;
_startingId + + ;
}
// Insert notification data
for ( quint32 i = 0 ; i < _numItemsToInsert ; i + + ) {
QJsonObject activity ;
activity . insert ( QStringLiteral ( " activity_id " ) , _startingId ) ;
activity . insert ( QStringLiteral ( " object_type " ) , " room " ) ;
activity . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " room " ) ) ;
activity . insert ( QStringLiteral ( " subject " ) , QStringLiteral ( " You have been invited into room%1 " ) . arg ( i ) ) ;
activity . insert ( QStringLiteral ( " message " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " object_name " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " datetime " ) , QDateTime : : currentDateTime ( ) . toString ( Qt : : ISODate ) ) ;
activity . insert ( QStringLiteral ( " icon " ) , QStringLiteral ( " http://example.de/core/img/places/talk.svg " ) ) ;
QJsonArray actionsArray ;
QJsonObject primaryAction ;
primaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " View chat " ) ) ;
primaryAction . insert ( QStringLiteral ( " link " ) , QStringLiteral ( " http://cloud.example.de/call/9p4vjdzd " ) ) ;
primaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " WEB " ) ) ;
primaryAction . insert ( QStringLiteral ( " primary " ) , true ) ;
actionsArray . push_back ( primaryAction ) ;
QJsonObject secondaryAction ;
secondaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Dismiss " ) ) ;
secondaryAction . insert ( QStringLiteral ( " link " ) ,
QString ( QStringLiteral ( " http://cloud.example.de/remote.php/dav " )
+ QStringLiteral ( " ocs/v2.php/apps/notifications/api/v2/notifications " ) + QString : : number ( i ) ) ) ;
secondaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " DELETE " ) ) ;
secondaryAction . insert ( QStringLiteral ( " primary " ) , false ) ;
actionsArray . push_back ( secondaryAction ) ;
activity . insert ( QStringLiteral ( " actions " ) , actionsArray ) ;
2021-12-23 13:57:08 +03:00
_activityData . push_back ( activity ) ;
2022-01-04 17:28:26 +03:00
_startingId + + ;
2021-12-23 13:57:08 +03:00
}
2022-01-04 17:28:26 +03:00
// Insert notification data
for ( quint32 i = 0 ; i < _numItemsToInsert ; i + + ) {
QJsonObject activity ;
activity . insert ( QStringLiteral ( " activity_id " ) , _startingId ) ;
activity . insert ( QStringLiteral ( " object_type " ) , " call " ) ;
activity . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " call " ) ) ;
activity . insert ( QStringLiteral ( " subject " ) , QStringLiteral ( " You have missed a %1's call " ) . arg ( i ) ) ;
activity . insert ( QStringLiteral ( " message " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " object_name " ) , QStringLiteral ( " " ) ) ;
activity . insert ( QStringLiteral ( " datetime " ) , QDateTime : : currentDateTime ( ) . toString ( Qt : : ISODate ) ) ;
activity . insert ( QStringLiteral ( " icon " ) , QStringLiteral ( " http://example.de/core/img/places/talk.svg " ) ) ;
QJsonArray actionsArray ;
QJsonObject primaryAction ;
primaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Call back " ) ) ;
primaryAction . insert ( QStringLiteral ( " link " ) , QStringLiteral ( " http://cloud.example.de/call/9p4vjdzd " ) ) ;
primaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " WEB " ) ) ;
primaryAction . insert ( QStringLiteral ( " primary " ) , true ) ;
actionsArray . push_back ( primaryAction ) ;
QJsonObject secondaryAction ;
secondaryAction . insert ( QStringLiteral ( " label " ) , QStringLiteral ( " Dismiss " ) ) ;
secondaryAction . insert ( QStringLiteral ( " link " ) ,
QString ( QStringLiteral ( " http://cloud.example.de/remote.php/dav " )
+ QStringLiteral ( " ocs/v2.php/apps/notifications/api/v2/notifications " ) + QString : : number ( i ) ) ) ;
secondaryAction . insert ( QStringLiteral ( " type " ) , QStringLiteral ( " DELETE " ) ) ;
secondaryAction . insert ( QStringLiteral ( " primary " ) , false ) ;
actionsArray . push_back ( secondaryAction ) ;
activity . insert ( QStringLiteral ( " actions " ) , actionsArray ) ;
_activityData . push_back ( activity ) ;
_startingId + + ;
}
_startingId - - ;
2021-12-23 13:57:08 +03:00
}
const QByteArray activityJsonData ( int sinceId , int limit )
{
QJsonArray data ;
2022-01-04 17:28:26 +03:00
const auto itFound = std : : find_if (
std : : cbegin ( _activityData ) , std : : cend ( _activityData ) , [ & sinceId ] ( const QJsonValue & currentActivityValue ) {
const auto currentActivityId =
currentActivityValue . toObject ( ) . value ( QStringLiteral ( " activity_id " ) ) . toInt ( ) ;
return currentActivityId = = sinceId ;
} ) ;
const int startIndex = itFound ! = std : : cend ( _activityData )
? static_cast < int > ( std : : distance ( std : : cbegin ( _activityData ) , itFound ) )
: - 1 ;
if ( startIndex > 0 ) {
for ( int dataIndex = startIndex , iteration = 0 ; dataIndex > = 0 & & iteration < limit ;
- - dataIndex , + + iteration ) {
if ( _activityData [ dataIndex ] . toObject ( ) . value ( QStringLiteral ( " activity_id " ) ) . toInt ( )
> sinceId - limit ) {
data . append ( _activityData [ dataIndex ] ) ;
}
2021-12-23 13:57:08 +03:00
}
}
QJsonObject root ;
QJsonObject ocs ;
ocs . insert ( QStringLiteral ( " data " ) , data ) ;
root . insert ( QStringLiteral ( " ocs " ) , ocs ) ;
return QJsonDocument ( root ) . toJson ( ) ;
}
2022-01-04 17:28:26 +03:00
QJsonValue activityById ( int id )
{
const auto itFound = std : : find_if (
std : : cbegin ( _activityData ) , std : : cend ( _activityData ) , [ & id ] ( const QJsonValue & currentActivityValue ) {
const auto currentActivityId =
currentActivityValue . toObject ( ) . value ( QStringLiteral ( " activity_id " ) ) . toInt ( ) ;
return currentActivityId = = id ;
} ) ;
if ( itFound ! = std : : cend ( _activityData ) ) {
return ( * itFound ) ;
}
return { } ;
}
int startingIdLast ( ) const { return _startingId ; }
2021-12-23 13:57:08 +03:00
private :
static FakeRemoteActivityStorage * _instance ;
QJsonArray _activityData ;
QVariantMap _metaSuccess ;
quint32 _numItemsToInsert = 30 ;
int _startingId = startingId ;
} ;
FakeRemoteActivityStorage * FakeRemoteActivityStorage : : _instance = nullptr ;
2022-01-04 17:28:26 +03:00
class TestingALM : public OCC : : ActivityListModel
{
Q_OBJECT
public :
TestingALM ( ) = default ;
void startFetchJob ( ) override
{
auto * job = new OCC : : JsonApiJob (
accountState ( ) - > account ( ) , QLatin1String ( " ocs/v2.php/apps/activity/api/v2/activity " ) , this ) ;
QObject : : connect ( this , & TestingALM : : activityJobStatusCode , this , & TestingALM : : slotProcessReceivedActivities ) ;
QObject : : connect ( job , & OCC : : JsonApiJob : : jsonReceived , this , & TestingALM : : activitiesReceived ) ;
QUrlQuery params ;
params . addQueryItem ( QLatin1String ( " since " ) , QString : : number ( currentItem ( ) ) ) ;
params . addQueryItem ( QLatin1String ( " limit " ) , QString : : number ( 50 ) ) ;
job - > addQueryParams ( params ) ;
job - > start ( ) ;
}
public slots :
void slotProcessReceivedActivities ( )
{
if ( rowCount ( ) > _numRowsPrev ) {
auto finalListCopy = finalList ( ) ;
for ( int i = _numRowsPrev ; i < rowCount ( ) ; + + i ) {
const auto modelIndex = index ( i , 0 ) ;
auto activity = finalListCopy . at ( modelIndex . row ( ) ) ;
if ( activity . _links . isEmpty ( ) ) {
const auto activityJsonObject = FakeRemoteActivityStorage : : instance ( ) - > activityById ( activity . _id ) ;
if ( ! activityJsonObject . isNull ( ) ) {
// because "_links" are normally populated within the notificationhandler.cpp, which we don't run as part of this unit test, we have to fill them here
// TODO: move the logic to populate "_links" to "activitylistmodel.cpp"
auto actions = activityJsonObject . toObject ( ) . value ( " actions " ) . toArray ( ) ;
foreach ( auto action , actions ) {
activity . _links . append ( OCC : : ActivityLink : : createFomJsonObject ( action . toObject ( ) ) ) ;
}
finalListCopy [ modelIndex . row ( ) ] = activity ;
}
}
}
setFinalList ( finalListCopy ) ;
}
_numRowsPrev = rowCount ( ) ;
emit activitiesProcessed ( ) ;
}
signals :
void activitiesProcessed ( ) ;
private :
int _numRowsPrev = 0 ;
} ;
2021-12-23 13:57:08 +03:00
class TestActivityListModel : public QObject
{
Q_OBJECT
public :
TestActivityListModel ( ) = default ;
~ TestActivityListModel ( ) override {
OCC : : AccountManager : : instance ( ) - > deleteAccount ( accountState . data ( ) ) ;
}
QScopedPointer < FakeQNAM > fakeQnam ;
OCC : : AccountPtr account ;
QScopedPointer < OCC : : AccountState > accountState ;
OCC : : Activity testNotificationActivity ;
static constexpr int searchResultsReplyDelay = 100 ;
private slots :
void initTestCase ( )
{
fakeQnam . reset ( new FakeQNAM ( { } ) ) ;
account = OCC : : Account : : create ( ) ;
account - > setCredentials ( new FakeCredentials { fakeQnam . data ( ) } ) ;
account - > setUrl ( QUrl ( ( " http://example.de " ) ) ) ;
accountState . reset ( new OCC : : AccountState ( account ) ) ;
fakeQnam - > setOverride ( [ this ] ( QNetworkAccessManager : : Operation op , const QNetworkRequest & req , QIODevice * device ) {
Q_UNUSED ( device ) ;
QNetworkReply * reply = nullptr ;
const auto urlQuery = QUrlQuery ( req . url ( ) ) ;
const auto format = urlQuery . queryItemValue ( QStringLiteral ( " format " ) ) ;
const auto since = urlQuery . queryItemValue ( QStringLiteral ( " since " ) ) . toInt ( ) ;
const auto limit = urlQuery . queryItemValue ( QStringLiteral ( " limit " ) ) . toInt ( ) ;
const auto path = req . url ( ) . path ( ) ;
if ( ! req . url ( ) . toString ( ) . startsWith ( accountState - > account ( ) - > url ( ) . toString ( ) ) ) {
reply = new FakeErrorReply ( op , req , this , 404 , fake404Response ) ;
}
if ( format ! = QStringLiteral ( " json " ) ) {
reply = new FakeErrorReply ( op , req , this , 400 , fake400Response ) ;
}
if ( path . startsWith ( QStringLiteral ( " /ocs/v2.php/apps/activity/api/v2/activity " ) ) ) {
reply = new FakePayloadReply ( op , req , FakeRemoteActivityStorage : : instance ( ) - > activityJsonData ( since , limit ) , searchResultsReplyDelay , fakeQnam . data ( ) ) ;
}
if ( ! reply ) {
return qobject_cast < QNetworkReply * > ( new FakeErrorReply ( op , req , this , 404 , QByteArrayLiteral ( " {error: \" Not found! \" } " ) ) ) ;
}
return reply ;
} ) ;
OCC : : AccountManager : : instance ( ) - > addAccount ( account ) ;
// Activity comparison is done by checking type, id, and accName
// We need an activity with these details, at least
testNotificationActivity . _accName = accountState - > account ( ) - > displayName ( ) ;
testNotificationActivity . _id = 1 ;
testNotificationActivity . _type = OCC : : Activity : : NotificationType ;
testNotificationActivity . _dateTime = QDateTime : : currentDateTime ( ) ;
} ;
// Test receiving activity from server
void testFetchingRemoteActivity ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
2022-01-04 17:28:26 +03:00
model . setCurrentItem ( FakeRemoteActivityStorage : : instance ( ) - > startingIdLast ( ) ) ;
2021-12-23 13:57:08 +03:00
model . startFetchJob ( ) ;
2022-01-04 17:28:26 +03:00
QSignalSpy activitiesJob ( & model , & TestingALM : : activitiesProcessed ) ;
2021-12-23 13:57:08 +03:00
QVERIFY ( activitiesJob . wait ( 3000 ) ) ;
QCOMPARE ( model . rowCount ( ) , 50 ) ;
} ;
// Test receiving activity from local user action
void testLocalSyncFileAction ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
OCC : : Activity activity ;
model . addSyncFileItemToActivityList ( activity ) ;
QCOMPARE ( model . rowCount ( ) , 1 ) ;
const auto index = model . index ( 0 , 0 ) ;
QVERIFY ( index . isValid ( ) ) ;
} ;
void testAddNotification ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
model . addNotificationToActivityList ( testNotificationActivity ) ;
QCOMPARE ( model . rowCount ( ) , 1 ) ;
const auto index = model . index ( 0 , 0 ) ;
QVERIFY ( index . isValid ( ) ) ;
} ;
void testAddError ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
OCC : : Activity activity ;
model . addErrorToActivityList ( activity ) ;
QCOMPARE ( model . rowCount ( ) , 1 ) ;
const auto index = model . index ( 0 , 0 ) ;
QVERIFY ( index . isValid ( ) ) ;
} ;
void testAddIgnoredFile ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
OCC : : Activity activity ;
activity . _folder = QStringLiteral ( " thingy " ) ;
activity . _file = QStringLiteral ( " test.txt " ) ;
model . addIgnoredFileToList ( activity ) ;
// We need to add another activity to the model for the combineActivityLists method to be called
model . addNotificationToActivityList ( testNotificationActivity ) ;
QCOMPARE ( model . rowCount ( ) , 2 ) ;
const auto index = model . index ( 0 , 0 ) ;
QVERIFY ( index . isValid ( ) ) ;
} ;
// Test removing activity from list
void testRemoveActivityWithRow ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
model . addNotificationToActivityList ( testNotificationActivity ) ;
QCOMPARE ( model . rowCount ( ) , 1 ) ;
model . removeActivityFromActivityList ( 0 ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
}
void testRemoveActivityWithActivity ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
model . addNotificationToActivityList ( testNotificationActivity ) ;
QCOMPARE ( model . rowCount ( ) , 1 ) ;
model . removeActivityFromActivityList ( testNotificationActivity ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
}
// Test getting the data from the model
void testData ( ) {
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
2022-01-04 17:28:26 +03:00
model . setCurrentItem ( FakeRemoteActivityStorage : : instance ( ) - > startingIdLast ( ) ) ;
2021-12-23 13:57:08 +03:00
model . startFetchJob ( ) ;
2022-01-04 17:28:26 +03:00
QSignalSpy activitiesJob ( & model , & TestingALM : : activitiesProcessed ) ;
2021-12-23 13:57:08 +03:00
QVERIFY ( activitiesJob . wait ( 3000 ) ) ;
QCOMPARE ( model . rowCount ( ) , 50 ) ;
model . addNotificationToActivityList ( testNotificationActivity ) ;
QCOMPARE ( model . rowCount ( ) , 51 ) ;
OCC : : Activity syncResultActivity ;
syncResultActivity . _id = 2 ;
syncResultActivity . _type = OCC : : Activity : : SyncResultType ;
syncResultActivity . _status = OCC : : SyncResult : : Error ;
syncResultActivity . _dateTime = QDateTime : : currentDateTime ( ) ;
syncResultActivity . _subject = QStringLiteral ( " Sample failed sync text " ) ;
syncResultActivity . _message = QStringLiteral ( " /path/to/thingy " ) ;
syncResultActivity . _link = QStringLiteral ( " /path/to/thingy " ) ;
syncResultActivity . _accName = accountState - > account ( ) - > displayName ( ) ;
model . addSyncFileItemToActivityList ( syncResultActivity ) ;
QCOMPARE ( model . rowCount ( ) , 52 ) ;
OCC : : Activity syncFileItemActivity ;
syncFileItemActivity . _id = 3 ;
syncFileItemActivity . _type = OCC : : Activity : : SyncFileItemType ; //client activity
syncFileItemActivity . _status = OCC : : SyncFileItem : : Success ;
syncFileItemActivity . _dateTime = QDateTime : : currentDateTime ( ) ;
syncFileItemActivity . _message = QStringLiteral ( " You created xyz.pdf " ) ;
syncFileItemActivity . _link = accountState - > account ( ) - > url ( ) ;
syncFileItemActivity . _accName = accountState - > account ( ) - > displayName ( ) ;
syncFileItemActivity . _file = QStringLiteral ( " xyz.pdf " ) ;
syncFileItemActivity . _fileAction = " " ;
model . addSyncFileItemToActivityList ( syncFileItemActivity ) ;
QCOMPARE ( model . rowCount ( ) , 53 ) ;
// Test all rows for things in common
for ( int i = 0 ; i < model . rowCount ( ) ; i + + ) {
const auto index = model . index ( i , 0 ) ;
2022-01-04 17:28:26 +03:00
auto text = index . data ( OCC : : ActivityListModel : : ActionTextRole ) . toString ( ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : ActionRole ) . canConvert < int > ( ) ) ;
const auto type = index . data ( OCC : : ActivityListModel : : ActionRole ) . toInt ( ) ;
2021-12-23 13:57:08 +03:00
QVERIFY ( type > = OCC : : Activity : : ActivityType ) ;
QVERIFY ( ! index . data ( OCC : : ActivityListModel : : AccountRole ) . toString ( ) . isEmpty ( ) ) ;
QVERIFY ( ! index . data ( OCC : : ActivityListModel : : ActionTextColorRole ) . toString ( ) . isEmpty ( ) ) ;
QVERIFY ( ! index . data ( OCC : : ActivityListModel : : ActionIconRole ) . toString ( ) . isEmpty ( ) ) ;
QVERIFY ( ! index . data ( OCC : : ActivityListModel : : PointInTimeRole ) . toString ( ) . isEmpty ( ) ) ;
2022-03-05 15:44:03 +03:00
QVERIFY ( index . data ( OCC : : ActivityListModel : : ObjectTypeRole ) . canConvert < int > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : ObjectNameRole ) . canConvert < QString > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : ObjectIdRole ) . canConvert < int > ( ) ) ;
2021-12-23 13:57:08 +03:00
QVERIFY ( index . data ( OCC : : ActivityListModel : : ActionsLinksRole ) . canConvert < QList < QVariant > > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : ActionTextRole ) . canConvert < QString > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : MessageRole ) . canConvert < QString > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : LinkRole ) . canConvert < QUrl > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : AccountConnectedRole ) . canConvert < bool > ( ) ) ;
QVERIFY ( index . data ( OCC : : ActivityListModel : : DisplayActions ) . canConvert < bool > ( ) ) ;
// Unfortunately, trying to check anything relating to filepaths causes a crash
// when the folder manager is invoked by the model to look for the relevant file
}
} ;
2022-01-04 17:28:26 +03:00
void tesActivityActionstData ( )
{
TestingALM model ;
model . setAccountState ( accountState . data ( ) ) ;
QAbstractItemModelTester modelTester ( & model ) ;
QCOMPARE ( model . rowCount ( ) , 0 ) ;
model . setCurrentItem ( FakeRemoteActivityStorage : : instance ( ) - > startingIdLast ( ) ) ;
int prevModelRowCount = model . rowCount ( ) ;
do {
prevModelRowCount = model . rowCount ( ) ;
model . startFetchJob ( ) ;
QSignalSpy activitiesJob ( & model , & TestingALM : : activitiesProcessed ) ;
QVERIFY ( activitiesJob . wait ( 3000 ) ) ;
for ( int i = prevModelRowCount ; i < model . rowCount ( ) ; i + + ) {
const auto index = model . index ( i , 0 ) ;
const auto actionsLinks = index . data ( OCC : : ActivityListModel : : ActionsLinksRole ) . toList ( ) ;
if ( ! actionsLinks . isEmpty ( ) ) {
const auto actionsLinksContextMenu =
index . data ( OCC : : ActivityListModel : : ActionsLinksContextMenuRole ) . toList ( ) ;
// context menu must be shorter than total action links
QVERIFY ( actionsLinks . isEmpty ( ) | | actionsLinksContextMenu . size ( ) < actionsLinks . size ( ) ) ;
// context menu must not contain the primary action
QVERIFY ( std : : find_if ( std : : begin ( actionsLinksContextMenu ) , std : : end ( actionsLinksContextMenu ) ,
[ ] ( const QVariant & entry ) { return entry . value < OCC : : ActivityLink > ( ) . _primary ; } )
= = std : : end ( actionsLinksContextMenu ) ) ;
const auto objectType = index . data ( OCC : : ActivityListModel : : ObjectTypeRole ) . toString ( ) ;
if ( ( objectType = = QStringLiteral ( " chat " ) | | objectType = = QStringLiteral ( " call " )
| | objectType = = QStringLiteral ( " room " ) ) ) {
const auto actionButtonsLinks =
index . data ( OCC : : ActivityListModel : : ActionsLinksForActionButtonsRole ) . toList ( ) ;
// both action links and buttons must contain a "WEB" verb element at the beginning
QVERIFY ( actionsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _verb = = QStringLiteral ( " WEB " ) ) ;
QVERIFY ( actionButtonsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _verb = = QStringLiteral ( " WEB " ) ) ;
// the first action button for chat must have image set
QVERIFY ( ! actionButtonsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _imageSource . isEmpty ( ) ) ;
QVERIFY ( ! actionButtonsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _imageSourceHovered . isEmpty ( ) ) ;
// logic for "chat" and other types of activities with multiple actions
if ( ( objectType = = QStringLiteral ( " chat " )
| | ( objectType ! = QStringLiteral ( " room " ) & & objectType ! = QStringLiteral ( " call " ) ) ) ) {
// button's label for "chat" must be renamed to "Reply"
QVERIFY ( actionButtonsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _label = = QObject : : tr ( " Reply " ) ) ;
if ( static_cast < quint32 > ( actionsLinks . size ( ) ) > OCC : : ActivityListModel : : maxActionButtons ( ) ) {
// in case total actions is longer than ActivityListModel::maxActionButtons, only one button must be present in a list of action buttons
QVERIFY ( actionButtonsLinks . size ( ) = = 1 ) ;
const auto actionButtonsAndContextMenuEntries = actionButtonsLinks + actionsLinksContextMenu ;
// in case total actions is longer than ActivityListModel::maxActionButtons, then a sum of action buttons and action menu entries must be equal to a total of action links
QVERIFY ( actionButtonsLinks . size ( ) + actionsLinksContextMenu . size ( ) = = actionsLinks . size ( ) ) ;
} else {
// in case a total of actions is less or equal to than ActivityListModel::maxActionButtons, then the length of action buttons must be greater than 1 and should contain "Mark as read" button at the end
QVERIFY ( actionButtonsLinks . size ( ) > 1 ) ;
QVERIFY ( actionButtonsLinks [ 1 ] . value < OCC : : ActivityLink > ( ) . _label
= = QObject : : tr ( " Mark as read " ) ) ;
}
} else if ( ( objectType = = QStringLiteral ( " call " ) ) ) {
QVERIFY (
actionButtonsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _label = = QStringLiteral ( " Call back " ) ) ;
}
} else {
QVERIFY ( actionsLinks [ 0 ] . value < OCC : : ActivityLink > ( ) . _label = = QStringLiteral ( " Dismiss " ) ) ;
}
}
}
} while ( prevModelRowCount < model . rowCount ( ) ) ;
} ;
2021-12-23 13:57:08 +03:00
} ;
QTEST_MAIN ( TestActivityListModel )
# include "testactivitylistmodel.moc"