summaryrefslogtreecommitdiffstats
path: root/extensions/browser/quota_service.h
diff options
context:
space:
mode:
Diffstat (limited to 'extensions/browser/quota_service.h')
-rw-r--r--extensions/browser/quota_service.h242
1 files changed, 242 insertions, 0 deletions
diff --git a/extensions/browser/quota_service.h b/extensions/browser/quota_service.h
new file mode 100644
index 0000000..8dacb95
--- /dev/null
+++ b/extensions/browser/quota_service.h
@@ -0,0 +1,242 @@
+// Copyright 2013 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+// The QuotaService uses heuristics to limit abusive requests
+// made by extensions. In this model 'items' (e.g individual bookmarks) are
+// represented by a 'Bucket' that holds state for that item for one single
+// interval of time. The interval of time is defined as 'how long we need to
+// watch an item (for a particular heuristic) before making a decision about
+// quota violations'. A heuristic is two functions: one mapping input
+// arguments to a unique Bucket (the BucketMapper), and another to determine
+// if a new request involving such an item at a given time is a violation.
+
+#ifndef EXTENSIONS_BROWSER_QUOTA_SERVICE_H_
+#define EXTENSIONS_BROWSER_QUOTA_SERVICE_H_
+
+#include <list>
+#include <map>
+#include <string>
+
+#include "base/compiler_specific.h"
+#include "base/containers/hash_tables.h"
+#include "base/memory/scoped_ptr.h"
+#include "base/threading/non_thread_safe.h"
+#include "base/time/time.h"
+#include "base/timer/timer.h"
+#include "base/values.h"
+
+class ExtensionFunction;
+
+namespace extensions {
+class QuotaLimitHeuristic;
+class TestResetQuotaFunction;
+
+typedef std::list<QuotaLimitHeuristic*> QuotaLimitHeuristics;
+
+// The QuotaService takes care that calls to certain extension
+// functions do not exceed predefined quotas.
+//
+// The QuotaService needs to live entirely on one thread, i.e.
+// be created, called and destroyed on the same thread, due to its use
+// of a RepeatingTimer.
+class QuotaService : public base::NonThreadSafe {
+ public:
+ // Some concrete heuristics (declared below) that ExtensionFunctions can
+ // use to help the service make decisions about quota violations.
+ class TimedLimit;
+ class SustainedLimit;
+
+ QuotaService();
+ virtual ~QuotaService();
+
+ // Decide whether the invocation of |function| with argument |args| by the
+ // extension specified by |extension_id| results in a quota limit violation.
+ // Returns an error message representing the failure if quota was exceeded,
+ // or empty-string if the request is fine and can proceed.
+ std::string Assess(const std::string& extension_id,
+ ExtensionFunction* function,
+ const base::ListValue* args,
+ const base::TimeTicks& event_time);
+
+ private:
+ friend class extensions::TestResetQuotaFunction;
+ typedef std::string ExtensionId;
+ typedef std::string FunctionName;
+ // All QuotaLimitHeuristic instances in this map are owned by us.
+ typedef std::map<FunctionName, QuotaLimitHeuristics> FunctionHeuristicsMap;
+
+ // Purge resets all accumulated data (except |violation_errors_|) as if the
+ // service was just created. Called periodically so we don't consume an
+ // unbounded amount of memory while tracking quota. Yes, this could mean an
+ // extension gets away with murder if it is timed right, but the extensions
+ // we are trying to limit are ones that consistently violate, so we'll
+ // converge to the correct set.
+ void Purge();
+ void PurgeFunctionHeuristicsMap(FunctionHeuristicsMap* map);
+ base::RepeatingTimer<QuotaService> purge_timer_;
+
+ // Our quota tracking state for extensions that have invoked quota limited
+ // functions. Each extension is treated separately, so extension ids are the
+ // key for the mapping. As an extension invokes functions, the map keeps
+ // track of which functions it has invoked and the heuristics for each one.
+ // Each heuristic will be evaluated and ANDed together to get a final answer.
+ std::map<ExtensionId, FunctionHeuristicsMap> function_heuristics_;
+
+ // For now, as soon as an extension violates quota, we don't allow it to
+ // make any more requests to quota limited functions. This provides a quick
+ // lookup for these extensions that is only stored in memory.
+ typedef std::map<std::string, std::string> ViolationErrorMap;
+ ViolationErrorMap violation_errors_;
+
+ DISALLOW_COPY_AND_ASSIGN(QuotaService);
+};
+
+// A QuotaLimitHeuristic is two things: 1, A heuristic to map extension
+// function arguments to corresponding Buckets for each input arg, and 2) a
+// heuristic for determining if a new event involving a particular item
+// (represented by its Bucket) constitutes a quota violation.
+class QuotaLimitHeuristic {
+ public:
+ // Parameters to configure the amount of tokens allotted to individual
+ // Bucket objects (see Below) and how often they are replenished.
+ struct Config {
+ // The maximum number of tokens a bucket can contain, and is refilled to
+ // every epoch.
+ int64 refill_token_count;
+
+ // Specifies how frequently the bucket is logically refilled with tokens.
+ base::TimeDelta refill_interval;
+ };
+
+ // A Bucket is how the heuristic portrays an individual item (since quota
+ // limits are per item) and all associated state for an item that needs to
+ // carry through multiple calls to Apply. It "holds" tokens, which are
+ // debited and credited in response to new events involving the item being
+ // being represented. For convenience, instead of actually periodically
+ // refilling buckets they are just 'Reset' on-demand (e.g. when new events
+ // come in). So, a bucket has an expiration to denote it has becomes stale.
+ class Bucket {
+ public:
+ Bucket() : num_tokens_(0) {}
+ // Removes a token from this bucket, and returns true if the bucket had
+ // any tokens in the first place.
+ bool DeductToken() { return num_tokens_-- > 0; }
+
+ // Returns true if this bucket has tokens to deduct.
+ bool has_tokens() const { return num_tokens_ > 0; }
+
+ // Reset this bucket to specification (from internal configuration), to be
+ // valid from |start| until the first refill interval elapses and it needs
+ // to be reset again.
+ void Reset(const Config& config, const base::TimeTicks& start);
+
+ // The time at which the token count and next expiration should be reset,
+ // via a call to Reset.
+ const base::TimeTicks& expiration() { return expiration_; }
+
+ private:
+ base::TimeTicks expiration_;
+ int64 num_tokens_;
+ DISALLOW_COPY_AND_ASSIGN(Bucket);
+ };
+ typedef std::list<Bucket*> BucketList;
+
+ // A helper interface to retrieve the bucket corresponding to |args| from
+ // the set of buckets (which is typically stored in the BucketMapper itself)
+ // for this QuotaLimitHeuristic.
+ class BucketMapper {
+ public:
+ virtual ~BucketMapper() {}
+ // In most cases, this should simply extract item IDs from the arguments
+ // (e.g for bookmark operations involving an existing item). If a problem
+ // occurs while parsing |args|, the function aborts - buckets may be non-
+ // empty). The expectation is that invalid args and associated errors are
+ // handled by the ExtensionFunction itself so we don't concern ourselves.
+ virtual void GetBucketsForArgs(const base::ListValue* args,
+ BucketList* buckets) = 0;
+ };
+
+ // Maps all calls to the same bucket, regardless of |args|, for this
+ // QuotaLimitHeuristic.
+ class SingletonBucketMapper : public BucketMapper {
+ public:
+ SingletonBucketMapper() {}
+ virtual ~SingletonBucketMapper() {}
+ virtual void GetBucketsForArgs(const base::ListValue* args,
+ BucketList* buckets) OVERRIDE;
+
+ private:
+ Bucket bucket_;
+ DISALLOW_COPY_AND_ASSIGN(SingletonBucketMapper);
+ };
+
+ // Ownership of |map| is given to the new QuotaLimitHeuristic.
+ QuotaLimitHeuristic(const Config& config,
+ BucketMapper* map,
+ const std::string& name);
+ virtual ~QuotaLimitHeuristic();
+
+ // Determines if sufficient quota exists (according to the Apply
+ // implementation of a derived class) to perform an operation with |args|,
+ // based on the history of similar operations with similar arguments (which
+ // is retrieved using the BucketMapper).
+ bool ApplyToArgs(const base::ListValue* args,
+ const base::TimeTicks& event_time);
+
+ // Returns an error formatted according to this heuristic.
+ std::string GetError() const;
+
+ protected:
+ const Config& config() { return config_; }
+
+ // Determine if the new event occurring at |event_time| involving |bucket|
+ // constitutes a quota violation according to this heuristic.
+ virtual bool Apply(Bucket* bucket, const base::TimeTicks& event_time) = 0;
+
+ private:
+ friend class QuotaLimitHeuristicTest;
+
+ const Config config_;
+
+ // The mapper used in Map. Cannot be NULL.
+ scoped_ptr<BucketMapper> bucket_mapper_;
+
+ // The name of the heuristic for formatting error messages.
+ std::string name_;
+
+ DISALLOW_COPY_AND_ASSIGN(QuotaLimitHeuristic);
+};
+
+// A simple per-item heuristic to limit the number of events that can occur in
+// a given period of time; e.g "no more than 100 events in an hour".
+class QuotaService::TimedLimit : public QuotaLimitHeuristic {
+ public:
+ TimedLimit(const Config& config, BucketMapper* map, const std::string& name)
+ : QuotaLimitHeuristic(config, map, name) {}
+ virtual bool Apply(Bucket* bucket,
+ const base::TimeTicks& event_time) OVERRIDE;
+};
+
+// A per-item heuristic to limit the number of events that can occur in a
+// period of time over a sustained longer interval. E.g "no more than two
+// events per minute, sustained over 10 minutes".
+class QuotaService::SustainedLimit : public QuotaLimitHeuristic {
+ public:
+ SustainedLimit(const base::TimeDelta& sustain,
+ const Config& config,
+ BucketMapper* map,
+ const std::string& name);
+ virtual bool Apply(Bucket* bucket,
+ const base::TimeTicks& event_time) OVERRIDE;
+
+ private:
+ // Specifies how long exhaustion of buckets is allowed to continue before
+ // denying requests.
+ const int64 repeat_exhaustion_allowance_;
+ int64 num_available_repeat_exhaustions_;
+};
+
+} // namespace extensions
+
+#endif // EXTENSIONS_BROWSER_QUOTA_SERVICE_H_