-
Notifications
You must be signed in to change notification settings - Fork 417
Scaling
With some modifications, you can make cube scale to thousands of inserts per second.
These are very rough notes, but perhaps useful to you.
Cube saves two kinds of records:
- Raw individual events. An event stream of 2400 records/s at 1kB each is 200 GB/day, 1.4 TB/week.
- Metrics aggregated over those events. For an event stream of any size, storing a year's worth of aggregates for 100 different metrics is 25 GB. (That's 10-second, 1-minute, 5-minute and hour aggregates, 68 bytes each)
- Collector: insert on events from receiving event
- Collector: update on metrics for invalidation (flush)
- Evaluator: find on metrics for metric calculation
- Evaluator: find on metrics for pyramidal calculation
- Evaluator: cursored find on events for calculateFlat
- Evaluator: save (upsert) on metrics from calculateFlat
- ensure time-based / ascending IDs
- Don't send old metrics
- Turn dates into mongo dates, not text strings
- Short field names
-
Capped events, uncapped metrics
- modify cube to construct uncapped collections for metrics, and capped collections for events
- change invalidation flusher to delete metrics, not update-with-invalid=true.
-
Calculation horizon -- Metrics older than the calculation horizon will not be recalculated from events.
-
Drop stale events on the floor
- drop events on the floor if they are older than the 'invalidation' horizon (which is shorter than the calculation horizon)
- Warn any time an invalidation extends more than 5 minutes into the past
-
Save metrics even if their value is zero
-
Use connection pool
-
Use Mongo 2.2+, and use separate databases for events and metrics
Currently, the event flush-er does a range-query update setting "i=false"; under heavy load, this is surprisingly hard on the database. We've observed multiple-second pauses for invalidation; I believe mongo has to hold a write lock open while it does a range query for the metrics to update. Metric updates, by contrast, are a reasonably cheap single-row save()
. (Side note: we tried uncapping the metrics collection and doing a remove()
; removes also lock the database, and it was no more efficient.)
Instead, the collector's flush operation should simply trigger a direct recalculation of the metric. This would halve the number of updates and significantly reduce thevariance of load.
The challenge is to a) not have everybody in sight run off recalculating metrics; b) not complicate the metric calculation code; c) ensure correctness; d) not introduce coupling.
proposal a: evaluator contains the queue
- add an api endpoint to the evaluator:
- endpoints: (method
DELETE
, path1.0/metric
), and methodGET
, path1.0/metric/refresh
. (I think this is the best match to HTTP semantics, but if anyone has strong opinions please advise.) - accepts start and stop; this calls deferred method on
metric.getter
and immediately returns202 (Accepted)
.
- endpoints: (method
- add code to
metric
to handle request - make the flusher call the evaluator rather than issue an update.
Good: simple; bad: requires the evaluator be running to handle invalidations.
proposal b: database contains the queue
- add a collection to track calculation requests. rather than updating a range of metrics, flusher inserts a single entry specifying a type + time range.
- add code to coalesce all pending calculation requests for a given type and update the metrics appropriately.
-
More ram helps
-
enable
--directoryperdb
-- with the separate -
(??hypothetical enable
noprealloc=true
. Collections that have heavy write loads will fill up very quickly, and cap from there.) -
enable
cpu=true
-
enable
logappend=true
-
do not enable
smallfiles
.
logpath=/ephemeral/log/mongod.log dbpath=/data
--slowms Specifies the threshold (in milliseconds) above which long-running queries will appear in the log, and in the system.profiler collection if profiling is enabled.
-
--journalCommitInterval
-- How often to group/batch commit (ms). Default 100ms -
--syncdelay arg
-- The number of seconds between data file flushes/syncs. Default 60 (seconds).- The MongoDB journal file flushes almost immediately, but data files are flushed lazily to optimize performance. The default is 60 seconds. A 0 setting means never, but is not recommended and should never be used with journaling enabled. On Linux, longer settings will likely be ineffective unless /proc/sys/vm/dirty_expire_centisecs is adjusted also. Generally, set syncdelay to 4x the desired dirty_expire_centiseconds value. If backgroundFlushing.average_ms is relatively large (>= 10,000 milliseconds), consider increasing this setting as well as dirty_expire_centiseconds. Under extremely high write volume situations, a higher syncdelay value may result in more journal files as they cannot be rotated (deleted) as quickly.
-
do use a journal
-
--rest
-- while in development, enables the REST interface.
To consider --
-
connection should be
{safe: false, connectTimeoutMS=XX}
-
mongo uses 1 thread per CPU connection.
Consider enabling swap. EC2
-
ulimit -- Set file descriptor limit (
-n
) and user process limit (-u
) to 4k+ (see etc/limits and ulimit) -
sysctl
- overcommit
- swappiness
-
Do not use large/huge virtual memory pages (the default is small pages which is what you want)
-
Use dmesg to see if box is behaving strangely
-
Ensure that readahead settings for the block devices that store the dbpath are acceptable Readahead
-
/proc/sys/vm/dirty_expire_centisecs
-
cube processes
- run with max memory size ulimit (
-m
) set to a finite value - (??hypothetical -- run other processes with NICE set so that mongo doesn't become target of OOM killer)
- run with max memory size ulimit (
- Ensure you are using NTP to minimize clock skew.
Note that you cannot shard with cube: capped collections do not work in a sharded database. Replica sets are fine; see above.
- "Exact values first. Sorted fields next. Ranged fields after that." See dex.
- use hints to force indexes
For Capped Collections, natural order is guaranteed to be the insertion order, making this a very efficient way to store and retrieve data in insertion order (much faster than say, indexing on a timestamp field). In addition to forward natural order, items may be retrieved in reverse natural order. For example, to return the 50 most recently inserted items (ordered most recent to less recent) from a capped collection, you would invoke: c=db.cappedCollection.find().sort({$natural:-1}).limit(50)
-
Use xfs, mounted with
"noatime"
, or ext4 file system, mounted with "noatime,data=writeback,nobarrier" -
use separate disks for journals, events and metrics
-
use the
--directoryperdb
-
use an SSD for metrics.
- don't need to use an SSD for events.
-
commodity drives are OK.
db.runCommand("journalLatencyTest") You can run this command on an idle system to get a baseline sync time for journaling. In addition, it is safe to run this command on a busy system to see the sync time on a busy system (which may be higher if the journal directory is on the same volume as the data files).
m1.large 232 7.68 .32 13 23 7.5 4 2 2 850 2 64 500 High
m1.xlarge 465 15.36 .64 13 23 15 8 4 2 1690 4 64 1000 High
m2.xlarge 327 10.80 .45 14 38 17.1 6.5 2 3.25 420 1 64 Moderate
m2.2xlarge 653 21.60 .90 14 38 34.2 13 4 3.25 850 2 64 High
m2.4xlarge 1307 43.20 1.80 14 38 68.4 26 8 3.25 1690 2 64 1000 High
hi1.4xlarge 2265 74.40 3.10 11 20 60.5 35 16 2.2 2048 ssd 2 64 10GB
-
faster cores is better than more cores+more total CPU.
- However I don't think you're CPU-bound so the m1.xlarge with provisioned EBS probably beats the m2.xlarge.
-
Provisioned IOPS for Amazon EBS
-
8 drives in a RAID-10 configuration
-
gives you 4x the storage of any one drive
$ for drive in /dev/md0 /dev/xvdh{1,2,3,4} ; do sudo blockdev --setra 128 $drive ; done
-
-
Put the oplog on the local drives
- If you put the journal on the local drives, it will not be present if the machine dies. It will persist through a reboot. Still, it may make sense if you are using a replica set.
-
do not put anything on the root partition.
EBS volumes are just fine. You have plenty of network throughput.
For an event stream of 2400 records/s @ 1kB/record,
- network throughput =~ 2.5 MB/s
- disk write =~ 2.5 MB/s
- mongo inserts =~ 2400 insert/s
- queries =~ (100 metrics, re-calculated 5x each)
- mongo network in =~ 2.5 MB/s
- mongo network out =~ k
- mongo vsize
- faults
- mongo db locked
- connections
- disk IO:
iostat -txm 5
- mongostat:
mongostat --host $(hostname)
- system:
htop
- network:
ifstat 5
insert query update delete getmore command flushes mapped vsize res faults locked db idx miss % qr|qw ar|aw netIn netOut conn time
0 0 0 0 0 1 0 464m 3.33g 45m 0 local:0.0% 0 0|0 0|0 62b 1k 1 17:58:22
0 0 0 0 0 1 0 464m 3.33g 45m 0 local:0.0% 0 0|0 0|0 62b 1k 1 17:58:23
inserts - # of inserts per second (* means replicated op)
query - # of queries per second
update - # of updates per second
delete - # of deletes per second
getmore - # of get mores (cursor batch) per second
command - # of commands per second, on a slave its local|replicated
flushes - # of fsync flushes per second
mapped - amount of data mmaped (total data size) megabytes
vsize - virtual size of process in megabytes
res - resident size of process in megabytes
faults - # of pages faults per sec
locked - name of and percent time for most locked database
idx miss - percent of btree page misses (sampled)
qr|qw - queue lengths for clients waiting (read|write)
ar|aw - active clients (read|write)
netIn - network traffic in - bits
netOut - network traffic out - bits
conn - number of open connections
-
backgroundFlushing.average_ms
-- if larger than 10_000 ms adjust mongo'ssyncdelay
and thedirty_expire_centiseconds
sysctl. -
virtualbytes - mappedbytes should be small
-
(??hypothetical run
db.runCommand({compact:'collectionname'})
to compact?) -
mongod --repair
-- will put a heavy load on your DB- you may see some 'yield' warnings when doing a repair; they're harmless
-
Dump stats on all databases and collections:
db.foo.stats() var mongo_stats = {} ; db._adminCommand("listDatabases").databases.forEach(function (d) {mdb = db.getSiblingDB(d.name); mongo_stats[d.name] = {} ; mongo_stats[d.name]["_stats"] = mdb.stats(); mdb.getCollectionNames().forEach(function(c) {mongo_stats[d.name][mdb[c].name] = mdb[c].stats(); })}); printjson(mongo_stats);
-
db.getProfilingLevel() db.setProfilingLevel(level) 0=off 1=record slow queries 2= record all queries mongod --profile=1 --slowms=15
db.system.profile.find( { info: /test.foo/ } )
db.system.profile.find().sort({$natural:-1})
-
nscanned
is much higher than nreturned, the database is scanning many objects to find the target objects. Consider creating an index to improve this. -
reslen
A large number of bytes returned (hundreds of kilobytes or more) causes slow performance. Consider passingfind()
a second parameter of the member names you require. -
fastmodinsert
-- better thanupsert
-
moved
-- Indicates the update moved the object on disk. bad. -
key updates
-- had to reindex. bad.
-
-
- start with
--rest
-- while in development, enables the REST interface.
- start with