parent
c87f259820
commit
feb337098d
@ -0,0 +1,66 @@ |
||||
/* |
||||
* Minio Cloud Storage, (C) 2016 Minio, Inc. |
||||
* |
||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||
* you may not use this file except in compliance with the License. |
||||
* You may obtain a copy of the License at |
||||
* |
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
* |
||||
* Unless required by applicable law or agreed to in writing, software |
||||
* distributed under the License is distributed on an "AS IS" BASIS, |
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
* See the License for the specific language governing permissions and |
||||
* limitations under the License. |
||||
*/ |
||||
|
||||
package main |
||||
|
||||
import "sync" |
||||
|
||||
// AppendFile - append data buffer at path.
|
||||
func (e erasure) AppendFile(volume, path string, dataBuffer []byte) (n int64, err error) { |
||||
// Split the input buffer into data and parity blocks.
|
||||
var blocks [][]byte |
||||
blocks, err = e.ReedSolomon.Split(dataBuffer) |
||||
if err != nil { |
||||
return 0, err |
||||
} |
||||
|
||||
// Encode parity blocks using data blocks.
|
||||
err = e.ReedSolomon.Encode(blocks) |
||||
if err != nil { |
||||
return 0, err |
||||
} |
||||
|
||||
var wg = &sync.WaitGroup{} |
||||
var wErrs = make([]error, len(e.storageDisks)) |
||||
// Write encoded data to quorum disks in parallel.
|
||||
for index, disk := range e.storageDisks { |
||||
if disk == nil { |
||||
continue |
||||
} |
||||
wg.Add(1) |
||||
// Write encoded data in routine.
|
||||
go func(index int, disk StorageAPI) { |
||||
defer wg.Done() |
||||
// Pick the block from the distribution.
|
||||
blockIndex := e.distribution[index] - 1 |
||||
n, wErr := disk.AppendFile(volume, path, blocks[blockIndex]) |
||||
if wErr != nil { |
||||
wErrs[index] = wErr |
||||
return |
||||
} |
||||
if n != int64(len(blocks[blockIndex])) { |
||||
wErrs[index] = errUnexpected |
||||
return |
||||
} |
||||
wErrs[index] = nil |
||||
}(index, disk) |
||||
} |
||||
|
||||
// Wait for all the appends to finish.
|
||||
wg.Wait() |
||||
|
||||
return int64(len(dataBuffer)), nil |
||||
} |
@ -1,186 +0,0 @@ |
||||
/* |
||||
* Minio Cloud Storage, (C) 2016 Minio, Inc. |
||||
* |
||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||
* you may not use this file except in compliance with the License. |
||||
* You may obtain a copy of the License at |
||||
* |
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
* |
||||
* Unless required by applicable law or agreed to in writing, software |
||||
* distributed under the License is distributed on an "AS IS" BASIS, |
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
* See the License for the specific language governing permissions and |
||||
* limitations under the License. |
||||
*/ |
||||
|
||||
package main |
||||
|
||||
import ( |
||||
"io" |
||||
"sync" |
||||
) |
||||
|
||||
// cleanupCreateFileOps - cleans up all the temporary files and other
|
||||
// temporary data upon any failure.
|
||||
func (e erasure) cleanupCreateFileOps(volume, path string, writers []io.WriteCloser) { |
||||
// Close and remove temporary writers.
|
||||
for _, writer := range writers { |
||||
if err := safeCloseAndRemove(writer); err != nil { |
||||
errorIf(err, "Failed to close writer.") |
||||
} |
||||
} |
||||
} |
||||
|
||||
// WriteErasure reads predefined blocks, encodes them and writes to configured storage disks.
|
||||
func (e erasure) writeErasure(volume, path string, reader *io.PipeReader, wcloser *waitCloser) { |
||||
// Release the block writer upon function return.
|
||||
defer wcloser.release() |
||||
|
||||
writers := make([]io.WriteCloser, len(e.storageDisks)) |
||||
|
||||
var wwg = &sync.WaitGroup{} |
||||
var errs = make([]error, len(e.storageDisks)) |
||||
|
||||
// Initialize all writers.
|
||||
for index, disk := range e.storageDisks { |
||||
if disk == nil { |
||||
continue |
||||
} |
||||
wwg.Add(1) |
||||
go func(index int, disk StorageAPI) { |
||||
defer wwg.Done() |
||||
writer, err := disk.CreateFile(volume, path) |
||||
if err != nil { |
||||
errs[index] = err |
||||
return |
||||
} |
||||
writers[index] = writer |
||||
}(index, disk) |
||||
} |
||||
|
||||
wwg.Wait() // Wait for all the create file to finish in parallel.
|
||||
for _, err := range errs { |
||||
if err == nil { |
||||
continue |
||||
} |
||||
e.cleanupCreateFileOps(volume, path, writers) |
||||
reader.CloseWithError(err) |
||||
return |
||||
} |
||||
|
||||
// Allocate 4MiB block size buffer for reading.
|
||||
dataBuffer := make([]byte, erasureBlockSize) |
||||
for { |
||||
// Read up to allocated block size.
|
||||
n, err := io.ReadFull(reader, dataBuffer) |
||||
if err != nil { |
||||
// Any unexpected errors, close the pipe reader with error.
|
||||
if err != io.ErrUnexpectedEOF && err != io.EOF { |
||||
// Remove all temp writers.
|
||||
e.cleanupCreateFileOps(volume, path, writers) |
||||
reader.CloseWithError(err) |
||||
return |
||||
} |
||||
} |
||||
// At EOF break out.
|
||||
if err == io.EOF { |
||||
break |
||||
} |
||||
if n > 0 { |
||||
// Split the input buffer into data and parity blocks.
|
||||
var dataBlocks [][]byte |
||||
dataBlocks, err = e.ReedSolomon.Split(dataBuffer[0:n]) |
||||
if err != nil { |
||||
// Remove all temp writers.
|
||||
e.cleanupCreateFileOps(volume, path, writers) |
||||
reader.CloseWithError(err) |
||||
return |
||||
} |
||||
|
||||
// Encode parity blocks using data blocks.
|
||||
err = e.ReedSolomon.Encode(dataBlocks) |
||||
if err != nil { |
||||
// Remove all temp writers upon error.
|
||||
e.cleanupCreateFileOps(volume, path, writers) |
||||
reader.CloseWithError(err) |
||||
return |
||||
} |
||||
|
||||
var wg = &sync.WaitGroup{} |
||||
var wErrs = make([]error, len(writers)) |
||||
// Write encoded data to quorum disks in parallel.
|
||||
for index, writer := range writers { |
||||
if writer == nil { |
||||
continue |
||||
} |
||||
wg.Add(1) |
||||
// Write encoded data in routine.
|
||||
go func(index int, writer io.Writer) { |
||||
defer wg.Done() |
||||
// Pick the block from the distribution.
|
||||
encodedData := dataBlocks[e.distribution[index]-1] |
||||
_, wErr := writers[index].Write(encodedData) |
||||
if wErr != nil { |
||||
wErrs[index] = wErr |
||||
return |
||||
} |
||||
wErrs[index] = nil |
||||
}(index, writer) |
||||
} |
||||
wg.Wait() |
||||
|
||||
// Cleanup and return on first non-nil error.
|
||||
for _, wErr := range wErrs { |
||||
if wErr == nil { |
||||
continue |
||||
} |
||||
// Remove all temp writers upon error.
|
||||
e.cleanupCreateFileOps(volume, path, writers) |
||||
reader.CloseWithError(wErr) |
||||
return |
||||
} |
||||
} |
||||
} |
||||
|
||||
// Close all writers and metadata writers in routines.
|
||||
for _, writer := range writers { |
||||
if writer == nil { |
||||
continue |
||||
} |
||||
// Safely wrote, now rename to its actual location.
|
||||
if err := writer.Close(); err != nil { |
||||
// Remove all temp writers upon error.
|
||||
e.cleanupCreateFileOps(volume, path, writers) |
||||
reader.CloseWithError(err) |
||||
return |
||||
} |
||||
} |
||||
|
||||
// Close the pipe reader and return.
|
||||
reader.Close() |
||||
return |
||||
} |
||||
|
||||
// CreateFile - create a file.
|
||||
func (e erasure) CreateFile(volume, path string) (writeCloser io.WriteCloser, err error) { |
||||
// Input validation.
|
||||
if !isValidVolname(volume) { |
||||
return nil, errInvalidArgument |
||||
} |
||||
if !isValidPath(path) { |
||||
return nil, errInvalidArgument |
||||
} |
||||
|
||||
// Initialize pipe for data pipe line.
|
||||
pipeReader, pipeWriter := io.Pipe() |
||||
|
||||
// Initialize a new wait closer, implements both Write and Close.
|
||||
wcloser := newWaitCloser(pipeWriter) |
||||
|
||||
// Start erasure encoding in routine, reading data block by block from pipeReader.
|
||||
go e.writeErasure(volume, path, pipeReader, wcloser) |
||||
|
||||
// Return the writer, caller should start writing to this.
|
||||
return wcloser, nil |
||||
} |
Loading…
Reference in new issue