WIP
This commit is contained in:
parent
0150862216
commit
7954a85a2a
141
file.go
Normal file
141
file.go
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2015 Alex Yatskov <alex@foosoft.net>
|
||||||
|
* Author: Alex Yatskov <alex@foosoft.net>
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
* this software and associated documentation files (the "Software"), to deal in
|
||||||
|
* the Software without restriction, including without limitation the rights to
|
||||||
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||||
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
||||||
|
* subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||||
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||||
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package goldsmith
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
)
|
||||||
|
|
||||||
|
type file struct {
|
||||||
|
path string
|
||||||
|
meta map[string]interface{}
|
||||||
|
|
||||||
|
srcData *bytes.Reader
|
||||||
|
srcPath string
|
||||||
|
}
|
||||||
|
|
||||||
|
func newFileFromData(path string, srcData []byte) *file {
|
||||||
|
return &file{
|
||||||
|
path: path,
|
||||||
|
meta: make(map[string]interface{}),
|
||||||
|
srcData: bytes.NewReader(srcData),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func newFileFromPath(path, srcPath string) *file {
|
||||||
|
return &file{
|
||||||
|
path: path,
|
||||||
|
meta: make(map[string]interface{}),
|
||||||
|
srcPath: srcPath,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) rewind() {
|
||||||
|
if f.srcData != nil {
|
||||||
|
f.srcData.Seek(0, os.SEEK_SET)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) export(dstPath string) error {
|
||||||
|
if err := os.MkdirAll(path.Dir(dstPath), 0755); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := f.cache(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
f.rewind()
|
||||||
|
|
||||||
|
fh, err := os.Create(dstPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer fh.Close()
|
||||||
|
|
||||||
|
var buff [1024]byte
|
||||||
|
for {
|
||||||
|
count, err := f.Read(buff[:])
|
||||||
|
if err == io.EOF {
|
||||||
|
break
|
||||||
|
} else if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := fh.Write(buff[:count]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) cache() error {
|
||||||
|
if f.srcData != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := ioutil.ReadFile(f.srcPath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
f.srcData = bytes.NewReader(data)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) Path() string {
|
||||||
|
return f.path
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) Keys() (keys []string) {
|
||||||
|
for key := range f.meta {
|
||||||
|
keys = append(keys, key)
|
||||||
|
}
|
||||||
|
|
||||||
|
return keys
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) Value(key string, def interface{}) interface{} {
|
||||||
|
if value, ok := f.meta[key]; ok {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
return def
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) SetValue(key string, value interface{}) {
|
||||||
|
f.meta[key] = value
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *file) Read(p []byte) (int, error) {
|
||||||
|
if err := f.cache(); err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return f.srcData.Read(p)
|
||||||
|
}
|
113
goldsmith.go
113
goldsmith.go
@ -23,21 +23,14 @@
|
|||||||
package goldsmith
|
package goldsmith
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
type stage struct {
|
|
||||||
gs *goldsmith
|
|
||||||
input, output chan *File
|
|
||||||
err error
|
|
||||||
}
|
|
||||||
|
|
||||||
type goldsmith struct {
|
type goldsmith struct {
|
||||||
srcDir, dstDir string
|
srcDir, dstDir string
|
||||||
stages []*stage
|
stages []*stage
|
||||||
@ -45,6 +38,12 @@ type goldsmith struct {
|
|||||||
mtx sync.Mutex
|
mtx sync.Mutex
|
||||||
active int64
|
active int64
|
||||||
stalled int64
|
stalled int64
|
||||||
|
tainted bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gs *goldsmith) fault(s *stage, f *file, err error) {
|
||||||
|
log.Print("%s\t%s\t%s", s, f, err)
|
||||||
|
gs.tainted = true
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gs *goldsmith) queueFiles(target uint) {
|
func (gs *goldsmith) queueFiles(target uint) {
|
||||||
@ -70,15 +69,7 @@ func (gs *goldsmith) queueFiles(target uint) {
|
|||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
file := NewFile(relPath)
|
s.CopyFile(relPath, path)
|
||||||
|
|
||||||
var f *os.File
|
|
||||||
if f, file.Err = os.Open(path); file.Err == nil {
|
|
||||||
_, file.Err = file.Buff.ReadFrom(f)
|
|
||||||
f.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
s.AddFile(file)
|
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
@ -125,28 +116,16 @@ func (gs *goldsmith) cleanupFiles() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gs *goldsmith) exportFile(file *File) {
|
func (gs *goldsmith) exportFile(f *file) error {
|
||||||
defer func() {
|
defer atomic.AddInt64(&gs.active, -1)
|
||||||
file.Buff = *bytes.NewBuffer(nil)
|
|
||||||
atomic.AddInt64(&gs.active, -1)
|
|
||||||
}()
|
|
||||||
|
|
||||||
if file.Err != nil {
|
absPath := filepath.Join(gs.dstDir, f.path)
|
||||||
return
|
if err := f.export(absPath); err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
absPath := filepath.Join(gs.dstDir, file.Path)
|
gs.refFile(f.path)
|
||||||
if file.Err = os.MkdirAll(path.Dir(absPath), 0755); file.Err != nil {
|
return nil
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var f *os.File
|
|
||||||
if f, file.Err = os.Create(absPath); file.Err == nil {
|
|
||||||
defer f.Close()
|
|
||||||
if _, file.Err = f.Write(file.Buff.Bytes()); file.Err == nil {
|
|
||||||
gs.refFile(file.Path)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gs *goldsmith) refFile(path string) {
|
func (gs *goldsmith) refFile(path string) {
|
||||||
@ -170,7 +149,7 @@ func (gs *goldsmith) refFile(path string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (gs *goldsmith) newStage() *stage {
|
func (gs *goldsmith) newStage() *stage {
|
||||||
s := &stage{gs: gs, output: make(chan *File)}
|
s := &stage{gs: gs, output: make(chan *file)}
|
||||||
if len(gs.stages) > 0 {
|
if len(gs.stages) > 0 {
|
||||||
s.input = gs.stages[len(gs.stages)-1].output
|
s.input = gs.stages[len(gs.stages)-1].output
|
||||||
}
|
}
|
||||||
@ -190,53 +169,55 @@ func (gs *goldsmith) chain(s *stage, p Plugin) {
|
|||||||
var (
|
var (
|
||||||
wg sync.WaitGroup
|
wg sync.WaitGroup
|
||||||
mtx sync.Mutex
|
mtx sync.Mutex
|
||||||
batch []*File
|
batch []File
|
||||||
)
|
)
|
||||||
|
|
||||||
dispatch := func(f *File) {
|
dispatch := func(f *file) {
|
||||||
if fin == nil {
|
if fin == nil {
|
||||||
s.output <- f
|
s.output <- f
|
||||||
} else {
|
} else {
|
||||||
mtx.Lock()
|
mtx.Lock()
|
||||||
batch = append(batch, f)
|
batch = append(batch, f)
|
||||||
atomic.AddInt64(&s.gs.stalled, 1)
|
|
||||||
mtx.Unlock()
|
mtx.Unlock()
|
||||||
|
|
||||||
|
atomic.AddInt64(&s.gs.stalled, 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if init != nil {
|
if init != nil {
|
||||||
if s.err = init.Initialize(s); s.err != nil {
|
if err := init.Initialize(s); err != nil {
|
||||||
|
gs.fault(s, nil, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for file := range s.input {
|
for f := range s.input {
|
||||||
if file.Err != nil || accept != nil && !accept.Accept(file) {
|
if accept != nil && !accept.Accept(f) {
|
||||||
s.output <- file
|
s.output <- f
|
||||||
} else if proc == nil {
|
} else if proc == nil {
|
||||||
dispatch(file)
|
|
||||||
} else {
|
|
||||||
wg.Add(1)
|
|
||||||
go func(f *File) {
|
|
||||||
defer wg.Done()
|
|
||||||
if proc.Process(s, f) {
|
|
||||||
dispatch(f)
|
dispatch(f)
|
||||||
} else {
|
} else {
|
||||||
f.Buff = *bytes.NewBuffer(nil)
|
wg.Add(1)
|
||||||
atomic.AddInt64(&gs.active, -1)
|
go func(f *file) {
|
||||||
|
defer wg.Done()
|
||||||
|
if err := proc.Process(s, f); err != nil {
|
||||||
|
gs.fault(s, f, err)
|
||||||
}
|
}
|
||||||
}(file)
|
dispatch(f)
|
||||||
|
}(f)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
|
|
||||||
if fin != nil {
|
if fin != nil {
|
||||||
if s.err = fin.Finalize(s, batch); s.err == nil {
|
if err := fin.Finalize(s, batch); err != nil {
|
||||||
for _, file := range batch {
|
gs.fault(s, nil, err)
|
||||||
atomic.AddInt64(&s.gs.stalled, -1)
|
|
||||||
s.output <- file
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, f := range batch {
|
||||||
|
atomic.AddInt64(&s.gs.stalled, -1)
|
||||||
|
s.output <- f.(*file)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -246,26 +227,20 @@ func (gs *goldsmith) Chain(p Plugin) Goldsmith {
|
|||||||
return gs
|
return gs
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gs *goldsmith) Complete() ([]*File, []error) {
|
func (gs *goldsmith) Complete() bool {
|
||||||
s := gs.stages[len(gs.stages)-1]
|
s := gs.stages[len(gs.stages)-1]
|
||||||
|
|
||||||
var files []*File
|
for f := range s.output {
|
||||||
for file := range s.output {
|
gs.exportFile(f)
|
||||||
gs.exportFile(file)
|
|
||||||
files = append(files, file)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
gs.cleanupFiles()
|
gs.cleanupFiles()
|
||||||
|
|
||||||
var errs []error
|
tainted := gs.tainted
|
||||||
for _, s := range gs.stages {
|
|
||||||
if s.err != nil {
|
|
||||||
errs = append(errs, s.err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
gs.stages = nil
|
gs.stages = nil
|
||||||
gs.refs = nil
|
gs.refs = nil
|
||||||
|
gs.tainted = false
|
||||||
|
|
||||||
return files, errs
|
return tainted
|
||||||
}
|
}
|
||||||
|
25
stage.go
25
stage.go
@ -22,17 +22,30 @@
|
|||||||
|
|
||||||
package goldsmith
|
package goldsmith
|
||||||
|
|
||||||
import "sync/atomic"
|
import (
|
||||||
|
"io"
|
||||||
|
"sync/atomic"
|
||||||
|
)
|
||||||
|
|
||||||
|
type stage struct {
|
||||||
|
gs *goldsmith
|
||||||
|
input, output chan *file
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *stage) NewFile(path string, r io.Reader) File {
|
||||||
|
atomic.AddInt64(&s.gs.active, 1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *stage) CopyFile(dst, src string) File {
|
||||||
|
atomic.AddInt64(&s.gs.active, 1)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *stage) RefFile(path string) {
|
func (s *stage) RefFile(path string) {
|
||||||
s.gs.refFile(path)
|
s.gs.refFile(path)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *stage) AddFile(file *File) {
|
|
||||||
atomic.AddInt64(&s.gs.active, 1)
|
|
||||||
s.output <- file
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *stage) SrcDir() string {
|
func (s *stage) SrcDir() string {
|
||||||
return s.gs.srcDir
|
return s.gs.srcDir
|
||||||
}
|
}
|
||||||
|
35
types.go
35
types.go
@ -23,13 +23,13 @@
|
|||||||
package goldsmith
|
package goldsmith
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"io"
|
||||||
"runtime"
|
"runtime"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Goldsmith interface {
|
type Goldsmith interface {
|
||||||
Chain(p Plugin) Goldsmith
|
Chain(p Plugin) Goldsmith
|
||||||
Complete() ([]*File, []error)
|
Complete() bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(srcDir, dstDir string) Goldsmith {
|
func New(srcDir, dstDir string) Goldsmith {
|
||||||
@ -42,32 +42,29 @@ func NewThrottled(srcDir, dstDir string, targetFileCount uint) Goldsmith {
|
|||||||
return gs
|
return gs
|
||||||
}
|
}
|
||||||
|
|
||||||
type File struct {
|
type File interface {
|
||||||
Path string
|
Path() string
|
||||||
Meta map[string]interface{}
|
|
||||||
Buff bytes.Buffer
|
|
||||||
Err error
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewFile(path string) *File {
|
Keys() []string
|
||||||
return &File{
|
Value(key string, def interface{}) interface{}
|
||||||
Path: cleanPath(path),
|
SetValue(key string, value interface{})
|
||||||
Meta: make(map[string]interface{}),
|
|
||||||
}
|
Read(p []byte) (int, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type Context interface {
|
type Context interface {
|
||||||
|
NewFile(path string, r io.Reader) File
|
||||||
|
CopyFile(dst, src string) File
|
||||||
|
RefFile(path string)
|
||||||
|
|
||||||
SrcDir() string
|
SrcDir() string
|
||||||
DstDir() string
|
DstDir() string
|
||||||
|
|
||||||
AddFile(file *File)
|
|
||||||
RefFile(path string)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type Plugin interface{}
|
type Plugin interface{}
|
||||||
|
|
||||||
type Accepter interface {
|
type Accepter interface {
|
||||||
Accept(file *File) bool
|
Accept(file File) bool
|
||||||
}
|
}
|
||||||
|
|
||||||
type Initializer interface {
|
type Initializer interface {
|
||||||
@ -75,9 +72,9 @@ type Initializer interface {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type Finalizer interface {
|
type Finalizer interface {
|
||||||
Finalize(ctx Context, files []*File) error
|
Finalize(ctx Context, fs []File) error
|
||||||
}
|
}
|
||||||
|
|
||||||
type Processor interface {
|
type Processor interface {
|
||||||
Process(ctx Context, file *File) bool
|
Process(ctx Context, f File) error
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user