1 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
3 /* This Source Code Form is subject to the terms of the Mozilla Public
4 * License, v. 2.0. If a copy of the MPL was not distributed with this
5 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
7 #include "StorageActivityService.h"
9 #include "mozilla/ipc/BackgroundChild.h"
10 #include "mozilla/ipc/BackgroundUtils.h"
11 #include "mozilla/ipc/PBackgroundChild.h"
12 #include "mozilla/ipc/PBackgroundSharedTypes.h"
13 #include "mozilla/BasePrincipal.h"
14 #include "mozilla/SchedulerGroup.h"
15 #include "mozilla/Services.h"
16 #include "mozilla/StaticPtr.h"
18 #include "nsComponentManagerUtils.h"
19 #include "nsIMutableArray.h"
20 #include "nsIObserverService.h"
21 #include "nsIPrincipal.h"
23 #include "nsSupportsPrimitives.h"
26 // This const is used to know when origin activities should be purged because
27 // too old. This value should be in sync with what the UI needs.
28 #define TIME_MAX_SECS 86400 /* 24 hours */
30 namespace mozilla::dom
{
32 static StaticRefPtr
<StorageActivityService
> gStorageActivityService
;
33 static bool gStorageActivityShutdown
= false;
36 void StorageActivityService::SendActivity(nsIPrincipal
* aPrincipal
) {
37 MOZ_ASSERT(NS_IsMainThread());
39 if (!aPrincipal
|| BasePrincipal::Cast(aPrincipal
)->Kind() !=
40 BasePrincipal::eContentPrincipal
) {
41 // Only content principals.
45 RefPtr
<StorageActivityService
> service
= GetOrCreate();
46 if (NS_WARN_IF(!service
)) {
50 service
->SendActivityInternal(aPrincipal
);
54 void StorageActivityService::SendActivity(
55 const mozilla::ipc::PrincipalInfo
& aPrincipalInfo
) {
56 if (aPrincipalInfo
.type() !=
57 mozilla::ipc::PrincipalInfo::TContentPrincipalInfo
) {
58 // only content principal.
62 RefPtr
<Runnable
> r
= NS_NewRunnableFunction(
63 "StorageActivityService::SendActivity", [aPrincipalInfo
]() {
64 MOZ_ASSERT(NS_IsMainThread());
67 mozilla::ipc::PrincipalInfoToPrincipal(aPrincipalInfo
);
69 if (principalOrErr
.isOk()) {
70 nsCOMPtr
<nsIPrincipal
> principal
= principalOrErr
.unwrap();
71 StorageActivityService::SendActivity(principal
);
74 "Could not obtain principal from "
75 "mozilla::ipc::PrincipalInfoToPrincipal");
79 SchedulerGroup::Dispatch(TaskCategory::Other
, r
.forget());
83 void StorageActivityService::SendActivity(const nsACString
& aOrigin
) {
84 MOZ_ASSERT(XRE_IsParentProcess());
87 origin
.Assign(aOrigin
);
89 RefPtr
<Runnable
> r
= NS_NewRunnableFunction(
90 "StorageActivityService::SendActivity", [origin
]() {
91 MOZ_ASSERT(NS_IsMainThread());
93 RefPtr
<StorageActivityService
> service
= GetOrCreate();
94 if (NS_WARN_IF(!service
)) {
98 service
->SendActivityInternal(origin
);
101 if (NS_IsMainThread()) {
104 SchedulerGroup::Dispatch(TaskCategory::Other
, r
.forget());
109 already_AddRefed
<StorageActivityService
> StorageActivityService::GetOrCreate() {
110 MOZ_ASSERT(NS_IsMainThread());
112 if (!gStorageActivityService
&& !gStorageActivityShutdown
) {
113 RefPtr
<StorageActivityService
> service
= new StorageActivityService();
115 nsCOMPtr
<nsIObserverService
> obs
= mozilla::services::GetObserverService();
116 if (NS_WARN_IF(!obs
)) {
121 obs
->AddObserver(service
, NS_XPCOM_SHUTDOWN_OBSERVER_ID
, true);
122 if (NS_WARN_IF(NS_FAILED(rv
))) {
126 gStorageActivityService
= service
;
129 RefPtr
<StorageActivityService
> service
= gStorageActivityService
;
130 return service
.forget();
133 StorageActivityService::StorageActivityService() {
134 MOZ_ASSERT(NS_IsMainThread());
137 StorageActivityService::~StorageActivityService() {
138 MOZ_ASSERT(NS_IsMainThread());
142 void StorageActivityService::SendActivityInternal(nsIPrincipal
* aPrincipal
) {
143 MOZ_ASSERT(NS_IsMainThread());
144 MOZ_ASSERT(aPrincipal
);
145 MOZ_ASSERT(BasePrincipal::Cast(aPrincipal
)->Kind() ==
146 BasePrincipal::eContentPrincipal
);
148 if (!XRE_IsParentProcess()) {
149 SendActivityToParent(aPrincipal
);
153 nsAutoCString origin
;
154 nsresult rv
= aPrincipal
->GetOrigin(origin
);
155 if (NS_WARN_IF(NS_FAILED(rv
))) {
159 SendActivityInternal(origin
);
162 void StorageActivityService::SendActivityInternal(const nsACString
& aOrigin
) {
163 MOZ_ASSERT(XRE_IsParentProcess());
165 mActivities
.InsertOrUpdate(aOrigin
, PR_Now());
169 void StorageActivityService::SendActivityToParent(nsIPrincipal
* aPrincipal
) {
170 MOZ_ASSERT(NS_IsMainThread());
171 MOZ_ASSERT(!XRE_IsParentProcess());
173 ::mozilla::ipc::PBackgroundChild
* actor
=
174 ::mozilla::ipc::BackgroundChild::GetOrCreateForCurrentThread();
175 if (NS_WARN_IF(!actor
)) {
179 mozilla::ipc::PrincipalInfo principalInfo
;
181 mozilla::ipc::PrincipalToPrincipalInfo(aPrincipal
, &principalInfo
);
182 if (NS_WARN_IF(NS_FAILED(rv
))) {
186 actor
->SendStorageActivity(principalInfo
);
190 StorageActivityService::Observe(nsISupports
* aSubject
, const char* aTopic
,
191 const char16_t
* aData
) {
192 MOZ_ASSERT(NS_IsMainThread());
193 MOZ_ASSERT(!strcmp(aTopic
, NS_XPCOM_SHUTDOWN_OBSERVER_ID
));
197 nsCOMPtr
<nsIObserverService
> obs
= mozilla::services::GetObserverService();
199 obs
->RemoveObserver(this, NS_XPCOM_SHUTDOWN_OBSERVER_ID
);
202 gStorageActivityShutdown
= true;
203 gStorageActivityService
= nullptr;
207 void StorageActivityService::MaybeStartTimer() {
208 MOZ_ASSERT(NS_IsMainThread());
211 nsresult rv
= NS_NewTimerWithCallback(
212 getter_AddRefs(mTimer
), this, 1000 * 5 * 60 /* any 5 minutes */,
213 nsITimer::TYPE_REPEATING_SLACK
, nullptr);
215 NS_WARNING("Could not start StorageActivityService timer");
220 void StorageActivityService::MaybeStopTimer() {
221 MOZ_ASSERT(NS_IsMainThread());
230 StorageActivityService::Notify(nsITimer
* aTimer
) {
231 MOZ_ASSERT(NS_IsMainThread());
232 MOZ_ASSERT(mTimer
== aTimer
);
234 uint64_t now
= PR_Now();
236 for (auto iter
= mActivities
.Iter(); !iter
.Done(); iter
.Next()) {
237 if ((now
- iter
.UserData()) / PR_USEC_PER_SEC
> TIME_MAX_SECS
) {
242 // If no activities, let's stop the timer.
243 if (mActivities
.Count() == 0) {
251 StorageActivityService::GetName(nsACString
& aName
) {
252 aName
.AssignLiteral("StorageActivityService");
257 StorageActivityService::GetActiveOrigins(PRTime aFrom
, PRTime aTo
,
258 nsIArray
** aRetval
) {
259 uint64_t now
= PR_Now();
260 if (((now
- aFrom
) / PR_USEC_PER_SEC
) > TIME_MAX_SECS
|| aFrom
>= aTo
) {
261 return NS_ERROR_INVALID_ARG
;
265 nsCOMPtr
<nsIMutableArray
> devices
=
266 do_CreateInstance(NS_ARRAY_CONTRACTID
, &rv
);
267 if (NS_WARN_IF(NS_FAILED(rv
))) {
271 for (const auto& activityEntry
: mActivities
) {
272 if (activityEntry
.GetData() >= aFrom
&& activityEntry
.GetData() <= aTo
) {
273 RefPtr
<BasePrincipal
> principal
=
274 BasePrincipal::CreateContentPrincipal(activityEntry
.GetKey());
275 MOZ_ASSERT(principal
);
277 rv
= devices
->AppendElement(principal
);
278 if (NS_WARN_IF(NS_FAILED(rv
))) {
284 devices
.forget(aRetval
);
289 StorageActivityService::MoveOriginInTime(nsIPrincipal
* aPrincipal
,
291 if (!XRE_IsParentProcess()) {
292 return NS_ERROR_FAILURE
;
295 nsAutoCString origin
;
296 nsresult rv
= aPrincipal
->GetOrigin(origin
);
297 if (NS_WARN_IF(NS_FAILED(rv
))) {
301 mActivities
.InsertOrUpdate(origin
, aWhen
/ PR_USEC_PER_SEC
);
306 StorageActivityService::TestOnlyReset() {
311 NS_INTERFACE_MAP_BEGIN(StorageActivityService
)
312 NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports
, nsIStorageActivityService
)
313 NS_INTERFACE_MAP_ENTRY(nsIStorageActivityService
)
314 NS_INTERFACE_MAP_ENTRY(nsIObserver
)
315 NS_INTERFACE_MAP_ENTRY(nsITimerCallback
)
316 NS_INTERFACE_MAP_ENTRY(nsINamed
)
317 NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference
)
320 NS_IMPL_ADDREF(StorageActivityService
)
321 NS_IMPL_RELEASE(StorageActivityService
)
323 } // namespace mozilla::dom