6.824 2015 Lecture 4: "Flat Datacenter Storage" Case Study
Note: These lecture notes were slightly modified from the ones posted on the 6.824
course website from Spring 2015.
Flat datacenter storage
Flat Datacenter Storage, Nightingale, Elson, Fan, Hofmann, Howell, Suzue, OSDI 2012
Why are we looking at this paper?
- Lab 2 wants to be like this when it grows up
- though details are all different
- fantastic performance -- world record cluster sort
- good systems paper -- details from apps all the way to network
What is FDS?
- a cluster storage system
- stores giant blobs -- 128-bit ID, multi-megabyte content
- clients and servers connected by network with high bisection bandwidth
- for big-data processing (like MapReduce)
- cluster of 1000s of computers processing data in parallel
High-level design -- a common pattern
- lots of clients
- lots of storage servers ("tractservers")
- lots of bandwidth between any two servers
- data is stored in blobs
- addressed by 128bit IDs
- further split into tracts
- numbered from 0
- 8MB sized
- partition the data
- master ("metadata server") controls partitioning
- replica groups for reliability
- tract table locator (TLT) stores a bunch entries
- in a
k
-replicated system, each entry has k
tractservers
- how to find where a tract
t
for blob b
is?
- compute TLT entry as
(h(b) + t) mod len(tlt)
- and you'll get a list of servers in that entry
- blob metadata is distributed and NOT stored in the TLT
- how to write a tract from a blob?
- look it up as described above
- send write to all servers in TLT entry
- only acknowledge write to client if all servers replied
- how to read a tract from a blob?
- look it up as described above
- send read to a random server in the TLT entry
Why is this high-level design useful?
- 1000s of disks of space
- store giant blobs, or many big blobs
- 1000s of servers/disks/arms of parallel throughput
- can expand over time -- reconfiguration
- large pool of storage servers for instant replacement after failure
Motivating app: MapReduce-style sort
- a mapper reads its split
1/M'th
of the input file (e.g., a tract)
- map emits a
<key, record>
for each record in split
- map partitions keys among
R
intermediate files (M*R
intermediate files in total)
- a reducer reads 1 of
R
intermediate files produced by each mapper
- reads
M
intermediate files (of 1/R
size)
- sorts its input
- produces
1/R'th
of the final sorted output file (R
blobs)
- FDS sort
- FDS sort does not store the intermediate files in FDS
- a client is both a mapper and reducer
- FDS sort is not locality-aware
- in mapreduce, master schedules workers on machine that are close to the data
- e.g., in same cluster
later versions of FDS sort uses more fine-grained work assignment
e.g., mapper doesn't get 1/N of the input file but something smaller
deals better with stragglers
The abstract's main claims are about performance.
- They set the world-record for disk-to-disk sorting in 2012 for MinuteSort
- 1,033 disks and 256 computers (136 tract servers, 120 clients)
- 1,401 Gbyte in 59.4s
Q: Does the abstract's 2 GByte/sec per client seem impressive?
- how fast can you read a file from Athena AFS? (abt 10 MB/sec)
- how fast can you read a typical hard drive?
- how fast can typical networks move data?
Q: Abstract claims recover from lost disk (92 GB) in 6.2 seconds
- that's 15 GByte / sec
- impressive?
- how is that even possible? that's 30x the speed of a disk!
- who might care about this metric?
What should we want to know from the paper?
- API?
- layout?
- finding data?
- add a server?
- replication?
- failure handling?
- failure model?
- consistent reads/writes? (i.e. does a read see latest write?)
- Not in FDS: "The current protocol for replication depends upon the
client to issue all writes to all replicas. This decision
means that FDS provides weak consistency guarantees
to clients. For example, if a client writes a tract to 1
of 3 replicas and then crashes, other clients reading
different replicas of that tract will observe differing state."
- "Writes are not guaranteed to be committed
in order of issue. Applications with ordering requirements
are responsible for issuing operations after previous
acknowledgments have been received, rather than
concurrently. FDS guarantees atomicity: a write is either
committed or failed completely"
- config mgr failure handling?
- good performance?
- useful for apps?
API
- Figure 1
- 128-bit blob IDs
- blobs have a length
- only whole-tract read and write -- 8 MB
Q: Why are 128-bit blob IDs a nice interface?
- Why not file names?
Q: Why do 8 MB tracts make sense?
- (Figure 3...)
Q: What kinds of client applications is the API aimed at?
- and not aimed at?
Layout: how do they spread data over the servers?
- Section 2.2
- break each blob into 8 MB tracts
- TLT maintained by metadata server
- has
n
entries
- for blob
b
and tract t
, i = (hash(b) + t) mod n
TLT[i]
contains list of tractservers w/ copy of the tract
- clients and servers all have copies of the latest TLT table
Example four-entry TLT with no replication:
0: S1
1: S2
2: S3
3: S4
suppose hash(27) = 2
then the tracts of blob 27 are laid out:
S1: 2 6
S2: 3 7
S3: 0 4 8
S4: 1 5 ...
FDS is "striping" blobs over servers at tract granularity
Q: hy have tracts at all? Why not store each blob on just one server?
- What kinds of apps will benefit from striping?
- What kinds of apps won't?
Q: How fast will a client be able to read a single tract?
Q: Where does the abstract's single-client 2 GB number come from?
Q: Why not the UNIX i-node approach?
- store an array per blob, indexed by tract #, yielding tractserver
- so you could make per-tract placement decisions
- e.g. write new tract to most lightly loaded server
Q: Why not hash(b + t)
?
Q: How many TLT entries should there be?
- how about
n = number of tractservers
?
- why do they claim this works badly? Section 2.2
The system needs to choose server pairs (or triplets &c) to put in TLT entries
- For replication
- Section 3.3
Q: How about:
0: S1 S2
1: S2 S1
2: S3 S4
3: S4 S3
...
- Why is this a bad idea?
- How long will repair take?
- What are the risks if two servers fail?
Q: why is the paper's n^2
scheme better?
Example:
0: S1 S2
1: S1 S3
2: S1 S4
3: S2 S1
4: S2 S3
5: S2 S4
...
- TLT with
n^2
entries, with every server pair occuring once
- How long will repair take?
- What are the risks if two servers fail?
Q: Why do they actually use a minimum replication level of 3?
- Same
n^2
table as before, third server is randomly chosen
- What effect on repair time?
- What effect on two servers failing?
- What if three disks fail?
Adding a tractserver
- To increase the amount of disk space / parallel throughput
- Metadata server picks some random TLT entries
- Substitutes new server for an existing server in those TLT entries
Extending a tract's size
- Newly created blobs have a length of 0 tracts
- Applications must extend a blob before writing past the end of it.
- The extend operation is atomic, is safe to execute concurrently with other
clients, and returns the new size of the blob as a result of the client’s call.
- A separate API tells the client the blob's current size.
- Extend operations for a blob are sent to the tractserver that owns that blob’s
metadata tract.
- The tractserver serializes it, atomically updates the metadata, and returns
the new size to each caller.
- If all writers follow this pattern, the extend operation provides a range of
tracts the caller may write without risk of conflict. Therefore, the extend
API is functionally equivalent to the Google File System's "atomic append."
- Space is allocated lazily on tractservers, so tracts claimed but not used do
not waste storage.
How do they maintain n^2
plus one arrangement as servers leave join?
Unclear.
Q: How long will adding a tractserver take?
Q: What about client write
's while tracts are being transferred?
- receiving tractserver may have copies from client(s) and from old srvr
- how does it know which is newest?
Q: What if a client reads/writes but has an old tract table?
Replication
- A writing client sends a copy to each tractserver in the TLT.
- A reading client asks one tractserver.
Q: Why don't they send writes through a primary?
- puts a lot of work on a primary? has to lookup and know TLT
- goal is not to have just one backup for a primary, it's to replicate and
strip data effectively across many disks
Q: What problems are they likely to have because of lack of primary?
- Why weren't these problems show-stoppers?
What happens after a tractserver fails?
- Metadata server stops getting heartbeat RPCs
- Picks random replacement for each TLT entry failed server was in
- New TLT gets a new version number
- Replacement servers fetch copies
Example of the tracts each server holds:
S1: 0 4 8 ...
S2: 0 1 ...
S3: 4 3 ...
S4: 8 2 ...
Q: Why not just pick one replacement server?
- it will have to take in a lot of writes for the lost data
=>
bad perf.
Q: How long will it take to copy all the tracts?
Q: If a tractserver's net breaks and is then repaired, might srvr serve old data?
Q: If a server crashes and reboots with disk intact, can contents be used?
- e.g. if it only missed a few writes?
- 3.2.1's "partial failure recovery"
- but won't it have already been replaced?
- how to know what writes it missed?
Q: When is it better to use 3.2.1's partial failure recovery?
What happens when the metadata server crashes?
Q: While metadata server is down, can the system proceed?
- yes, clients who have the TLT can go on
Q: Is there a backup metadata server?
- not in the paper, they said they might use Paxos for replication
- TODO: not clear why replicating the metadata server would lead to consistency
problems
Q: How does rebooted metadata server get a copy of the TLT?
- Eh, maybe it has it on disk?
- Maybe it just simply reconstructs it from all the heartbeats?
Q: Does their scheme seem correct?
- how does the metadata server know it has heard from all tractservers?
- it doesn't, it just adds servers as they send heartbeats
- how does it know all tractservers were up to date?
- TODO: Up to date with what?
Random issues
Q: Is the metadata server likely to be a bottleneck?
- hard to tell. what's the use case?
- if you have a client w/ that has memory to remember TLT
then he only contacts metadata server once and then
starts doing all of his reads/writes
- if you have a lot of clients joining the system, or coming back but
forgetting the TLT (because of lack of storage maybe), then the metadata
server would be in use heavily
- however, this won't affect the bandwidth the clients get once they
downloaded the TLT
Q: Why do they need the scrubber application mentioned in 2.3?
- why don't they delete the tracts when the blob is deleted?
- faster to do GC, rather than scheduling & executing deletes?
- can a blob be written after it is deleted?
- TODO: not sure, seems like yes, because the metadata for that
blob is in tract -1 and I don't think
WriteTract
checks the metadata
before every write, so you could maybe have races
Performance
Q: How do we know we're seeing "good" performance? What's the best you can expect?
- best you can expect is to take each disks bandwidth and have the system's
bandwidth be
# of disk * disk bandwidth
Q: Limiting resource for 2 GBps single-client?
- assuming this is end of 5.2
- 30 tractservers means maximum of 30 * 130MB/s = 3.9GBps
- so limiting resource is network bandwidth
Q: Figure 4a: Why starts low? Why goes up? Why levels off? Why does it level off at that particular performance?
- starts low because single client bandwidth is limited
- goes up b.c. as # of clients is increased each one adds more bandwidth to
the system
- levels off because at some point the client bandwidth > server's bandwidth
- why levels off at 32 GBps for
x
clients w/ 516 disks?
- Figure 3 suggests a 10,000 RPM disk can read 5MB chunks at around 130MB/s
- not clear from logarithmic scale graph what
x
is
10 < x < 50
(maybe 25 < x < 50
?)
516 disks * 130MB/s = 67 GBps
, so seems like best case performance
should've leveled off at more than 32 GBps?
- in reality not all disks are 130MB/s maybe? (only the 10,000rpm SAS onese
were that fast)
- in reality multiple disks on a single node might make that number smaller
maybe?
- anyway, something like
x=40
clients would have 40 * 10Gbps = 40 * 1.25GBps
= 50 Gbps
which is higher than the actual (claimed) bandwidth of the server
of 32 GBps
Q: Figure 4b shows random r/w as fast as sequential (Figure 4a). Is this what you'd expect?
- Yes. Random R/W requests for different tracts go to different servers, just
like sequential ones do
=
> no difference
Q: Why are writes slower than reads with replication in Figure 4c?
- A write is sent to all tract servers? Not over until all of them reply.
- w/ higher number of clients writing
=>
more work done by each server
- Paper says: "As expected, the write bandwidth is about onethird
of the read bandwidth since clients must send three
copies of each write"
- A read is sent to just one?
Q: Where does the 92 GB in 6.2 seconds come from?
- Table 1, 4th column
- that's 15 GB / second, both read and written
- 1000 disks, triple replicated, 128 servers?
- what's the limiting resource? disk? cpu? net?
How big is each sort bucket?
- i.e. is the sort of each bucket in-memory?
- 1400 GB total
- 128 compute servers
- between 12 and 96 GB of RAM each
- hmm, say 50 on average, so total RAM may be 6400 GB
- thus sort of each bucket is in memory, does not write passes to FDS
- thus total time is just four transfers of 1400 GB
- client limit:
128 * 2 GB/s = 256 GB / sec
- disk limit:
1000 * 50 MB/s = 50 GB / sec
- thus bottleneck is likely to be disk throughput