goldsmith/file.go

149 lines
3.1 KiB
Go
Raw Normal View History

2015-12-18 04:14:39 +00:00
/*
* 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"
2016-01-13 04:12:50 +00:00
"path/filepath"
2015-12-18 04:14:39 +00:00
)
type file struct {
2015-12-18 07:06:28 +00:00
path string
2016-01-01 08:47:28 +00:00
Meta map[string]interface{}
2015-12-18 07:06:28 +00:00
2015-12-18 10:19:43 +00:00
reader *bytes.Reader
asset string
2015-12-18 04:14:39 +00:00
}
2016-01-13 04:12:50 +00:00
func (f *file) export(dstDir string) error {
dstPath := filepath.Join(dstDir, f.path)
2016-01-12 08:32:06 +00:00
if len(f.asset) > 0 && fileCached(f.asset, dstPath) {
return nil
}
2015-12-18 04:14:39 +00:00
if err := os.MkdirAll(path.Dir(dstPath), 0755); err != nil {
return err
}
2015-12-18 10:49:52 +00:00
fw, err := os.Create(dstPath)
2015-12-18 04:14:39 +00:00
if err != nil {
return err
}
2015-12-18 10:49:52 +00:00
defer fw.Close()
if f.reader == nil {
fr, err := os.Open(f.asset)
if err != nil {
return err
}
defer fr.Close()
if _, err := io.Copy(fw, fr); err != nil {
return err
}
} else {
2015-12-20 14:18:58 +00:00
if _, err := f.Seek(0, os.SEEK_SET); err != nil {
return err
}
2015-12-18 10:49:52 +00:00
if _, err := f.WriteTo(fw); err != nil {
return err
}
2015-12-18 04:14:39 +00:00
}
return nil
}
func (f *file) cache() error {
2015-12-18 10:19:43 +00:00
if f.reader != nil {
2015-12-18 04:14:39 +00:00
return nil
}
2015-12-18 10:19:43 +00:00
data, err := ioutil.ReadFile(f.asset)
2015-12-18 04:14:39 +00:00
if err != nil {
return err
}
2015-12-20 08:38:53 +00:00
f.reader = bytes.NewReader(data)
2015-12-18 04:14:39 +00:00
return nil
}
2015-12-18 04:37:32 +00:00
//
// File Implementation
//
2015-12-18 04:14:39 +00:00
func (f *file) Path() string {
return f.path
}
2016-06-06 05:02:09 +00:00
func (f *file) Dir() string {
return path.Dir(f.path)
}
2016-01-01 08:47:28 +00:00
func (f *file) Value(key string) (interface{}, bool) {
2015-12-29 12:08:41 +00:00
value, ok := f.Meta[key]
return value, ok
2015-12-18 04:14:39 +00:00
}
2016-01-01 08:47:28 +00:00
func (f *file) SetValue(key string, value interface{}) {
2015-12-29 12:08:41 +00:00
f.Meta[key] = value
}
func (f *file) CopyValues(src File) {
rf := src.(*file)
for name, value := range rf.Meta {
f.SetValue(name, value)
2015-12-19 12:14:16 +00:00
}
}
2015-12-18 04:14:39 +00:00
func (f *file) Read(p []byte) (int, error) {
if err := f.cache(); err != nil {
return 0, err
}
2015-12-18 10:19:43 +00:00
return f.reader.Read(p)
2015-12-18 07:06:28 +00:00
}
func (f *file) WriteTo(w io.Writer) (int64, error) {
if err := f.cache(); err != nil {
return 0, err
}
2015-12-18 10:19:43 +00:00
return f.reader.WriteTo(w)
2015-12-18 07:06:28 +00:00
}
2015-12-20 14:18:58 +00:00
func (f *file) Seek(offset int64, whence int) (int64, error) {
if f.reader == nil && offset == 0 && (whence == os.SEEK_SET || whence == os.SEEK_CUR) {
return 0, nil
}
if err := f.cache(); err != nil {
return 0, err
}
return f.reader.Seek(offset, whence)
}