linx-server/meta.go

219 lines
4.7 KiB
Go
Raw Permalink Normal View History

2015-09-27 22:17:12 -04:00
package main
import (
"archive/tar"
"archive/zip"
"compress/bzip2"
"compress/gzip"
"crypto/sha256"
"encoding/hex"
"encoding/json"
2015-09-27 22:17:12 -04:00
"errors"
"io"
"io/ioutil"
2015-09-27 22:17:12 -04:00
"os"
"path"
"sort"
"time"
2015-10-30 23:13:43 -04:00
"unicode"
"bitbucket.org/taruti/mimemagic"
"github.com/dchest/uniuri"
2015-09-27 22:17:12 -04:00
)
type MetadataJSON struct {
DeleteKey string `json:"delete_key"`
Sha256sum string `json:"sha256sum"`
Mimetype string `json:"mimetype"`
Size int64 `json:"size"`
Expiry int64 `json:"expiry"`
ArchiveFiles []string `json:"archive_files,omitempty"`
}
type Metadata struct {
DeleteKey string
Sha256sum string
Mimetype string
Size int64
Expiry time.Time
ArchiveFiles []string
}
var NotFoundErr = errors.New("File not found.")
var BadMetadata = errors.New("Corrupted metadata.")
2015-09-27 22:17:12 -04:00
func generateMetadata(fName string, exp time.Time, delKey string) (m Metadata, err error) {
file, err := os.Open(path.Join(Config.filesDir, fName))
fileInfo, err := os.Stat(path.Join(Config.filesDir, fName))
2015-09-27 22:17:12 -04:00
if err != nil {
return
2015-09-27 22:17:12 -04:00
}
defer file.Close()
m.Size = fileInfo.Size()
m.Expiry = exp
2015-09-27 22:17:12 -04:00
if delKey == "" {
m.DeleteKey = uniuri.NewLen(30)
} else {
m.DeleteKey = delKey
}
2015-09-27 22:17:12 -04:00
// Get first 512 bytes for mimetype detection
header := make([]byte, 512)
file.Read(header)
2015-09-27 22:17:12 -04:00
m.Mimetype = mimemagic.Match("", header)
2015-09-27 22:17:12 -04:00
2015-10-28 15:21:54 -04:00
if m.Mimetype == "" {
// Check if the file seems anything like text
2015-10-30 23:13:43 -04:00
if printable(header) {
2015-10-28 15:21:54 -04:00
m.Mimetype = "text/plain"
} else {
m.Mimetype = "application/octet-stream"
}
}
// Compute the sha256sum
hasher := sha256.New()
file.Seek(0, 0)
_, err = io.Copy(hasher, file)
if err == nil {
m.Sha256sum = hex.EncodeToString(hasher.Sum(nil))
2015-09-27 22:17:12 -04:00
}
file.Seek(0, 0)
// If archive, grab list of filenames
if m.Mimetype == "application/x-tar" {
tReadr := tar.NewReader(file)
for {
hdr, err := tReadr.Next()
if err == io.EOF || err != nil {
break
}
if hdr.Typeflag == tar.TypeDir || hdr.Typeflag == tar.TypeReg {
m.ArchiveFiles = append(m.ArchiveFiles, hdr.Name)
}
}
sort.Strings(m.ArchiveFiles)
} else if m.Mimetype == "application/x-gzip" {
gzf, err := gzip.NewReader(file)
if err == nil {
tReadr := tar.NewReader(gzf)
for {
hdr, err := tReadr.Next()
if err == io.EOF || err != nil {
break
}
if hdr.Typeflag == tar.TypeDir || hdr.Typeflag == tar.TypeReg {
m.ArchiveFiles = append(m.ArchiveFiles, hdr.Name)
}
}
sort.Strings(m.ArchiveFiles)
}
} else if m.Mimetype == "application/x-bzip" {
bzf := bzip2.NewReader(file)
tReadr := tar.NewReader(bzf)
for {
hdr, err := tReadr.Next()
if err == io.EOF || err != nil {
break
}
if hdr.Typeflag == tar.TypeDir || hdr.Typeflag == tar.TypeReg {
m.ArchiveFiles = append(m.ArchiveFiles, hdr.Name)
}
}
sort.Strings(m.ArchiveFiles)
} else if m.Mimetype == "application/zip" {
zf, err := zip.NewReader(file, m.Size)
if err == nil {
for _, f := range zf.File {
m.ArchiveFiles = append(m.ArchiveFiles, f.Name)
}
}
sort.Strings(m.ArchiveFiles)
2015-09-27 22:17:12 -04:00
}
return
2015-09-27 22:17:12 -04:00
}
func metadataWrite(filename string, metadata *Metadata) error {
file, err := os.Create(path.Join(Config.metaDir, filename))
if err != nil {
return err
}
defer file.Close()
2015-09-27 22:17:12 -04:00
mjson := MetadataJSON{}
mjson.DeleteKey = metadata.DeleteKey
mjson.Mimetype = metadata.Mimetype
mjson.ArchiveFiles = metadata.ArchiveFiles
mjson.Sha256sum = metadata.Sha256sum
mjson.Expiry = metadata.Expiry.Unix()
mjson.Size = metadata.Size
2015-09-27 22:17:12 -04:00
byt, err := json.Marshal(mjson)
2015-09-27 22:17:12 -04:00
if err != nil {
return err
2015-09-27 22:17:12 -04:00
}
_, err = file.Write(byt)
if err != nil {
return err
2015-09-27 22:17:12 -04:00
}
return nil
2015-09-27 22:17:12 -04:00
}
func metadataRead(filename string) (metadata Metadata, err error) {
b, err := ioutil.ReadFile(path.Join(Config.metaDir, filename))
if err != nil {
// Metadata does not exist, generate one
newMData, err := generateMetadata(filename, neverExpire, "")
if err != nil {
return metadata, err
}
metadataWrite(filename, &newMData)
b, err = ioutil.ReadFile(path.Join(Config.metaDir, filename))
if err != nil {
return metadata, BadMetadata
}
2015-09-27 22:17:12 -04:00
}
mjson := MetadataJSON{}
err = json.Unmarshal(b, &mjson)
2015-09-27 22:17:12 -04:00
if err != nil {
return metadata, BadMetadata
2015-09-27 22:17:12 -04:00
}
metadata.DeleteKey = mjson.DeleteKey
metadata.Mimetype = mjson.Mimetype
metadata.ArchiveFiles = mjson.ArchiveFiles
metadata.Sha256sum = mjson.Sha256sum
metadata.Expiry = time.Unix(mjson.Expiry, 0)
metadata.Size = mjson.Size
return
2015-09-27 22:17:12 -04:00
}
2015-10-30 23:13:43 -04:00
func printable(data []byte) bool {
for i, b := range data {
r := rune(b)
// A null terminator that's not at the beginning of the file
if r == 0 && i == 0 {
return false
} else if r == 0 && i < 0 {
continue
}
if r > unicode.MaxASCII {
return false
}
}
return true
}