mirror of https://github.com/perkeep/perkeep.git
405 lines
10 KiB
Go
405 lines
10 KiB
Go
/*
|
|
Copyright 2013 Google 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 schema
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
"unicode/utf8"
|
|
|
|
"camlistore.org/pkg/blob"
|
|
)
|
|
|
|
// A MissingFieldError represents a missing JSON field in a schema blob.
|
|
type MissingFieldError string
|
|
|
|
func (e MissingFieldError) Error() string {
|
|
return fmt.Sprintf("schema: missing field %q", string(e))
|
|
}
|
|
|
|
// IsMissingField returns whether error is of type MissingFieldError.
|
|
func IsMissingField(err error) bool {
|
|
_, ok := err.(MissingFieldError)
|
|
return ok
|
|
}
|
|
|
|
// AnyBlob represents any type of schema blob.
|
|
type AnyBlob interface {
|
|
Blob() *Blob
|
|
}
|
|
|
|
// Buildable returns a Builder from a base.
|
|
type Buildable interface {
|
|
Builder() *Builder
|
|
}
|
|
|
|
// A Blob represents a Camlistore schema blob.
|
|
// It is immutable.
|
|
type Blob struct {
|
|
br blob.Ref
|
|
str string
|
|
ss *superset
|
|
}
|
|
|
|
// Type returns the blob's "camliType" field.
|
|
func (b *Blob) Type() string { return b.ss.Type }
|
|
|
|
// BlobRef returns the schema blob's blobref.
|
|
func (b *Blob) BlobRef() blob.Ref { return b.br }
|
|
|
|
// JSON returns the JSON bytes of the schema blob.
|
|
func (b *Blob) JSON() string { return b.str }
|
|
|
|
// Blob returns itself, so it satisifies the AnyBlob interface.
|
|
func (b *Blob) Blob() *Blob { return b }
|
|
|
|
// PartsSize returns the number of bytes represented by the "parts" field.
|
|
// TODO: move this off *Blob to a specialized type.
|
|
func (b *Blob) PartsSize() int64 {
|
|
n := int64(0)
|
|
for _, part := range b.ss.Parts {
|
|
n += int64(part.Size)
|
|
}
|
|
return n
|
|
}
|
|
|
|
// FileName returns the file, directory, or symlink's filename, or the empty string.
|
|
// TODO: move this off *Blob to a specialized type.
|
|
func (b *Blob) FileName() string {
|
|
return b.ss.FileNameString()
|
|
}
|
|
|
|
// ClaimDate returns the "claimDate" field.
|
|
// If there is no claimDate, the error will be a MissingFieldError.
|
|
func (b *Blob) ClaimDate() (time.Time, error) {
|
|
var ct time.Time
|
|
claimDate := b.ss.ClaimDate
|
|
if claimDate.IsZero() {
|
|
return ct, MissingFieldError("claimDate")
|
|
}
|
|
return claimDate.Time(), nil
|
|
}
|
|
|
|
// ByteParts returns the "parts" field. The caller owns the returned
|
|
// slice.
|
|
func (b *Blob) ByteParts() []BytesPart {
|
|
// TODO: move this method off Blob, and make the caller go
|
|
// through a (*Blob).ByteBackedBlob() comma-ok accessor first.
|
|
s := make([]BytesPart, len(b.ss.Parts))
|
|
for i, part := range b.ss.Parts {
|
|
s[i] = *part
|
|
}
|
|
return s
|
|
}
|
|
|
|
func (b *Blob) Builder() *Builder {
|
|
var m map[string]interface{}
|
|
err := json.Unmarshal([]byte(b.str), &m)
|
|
if err != nil {
|
|
panic("failed to decode previously-thought-valid Blob's JSON: " + err.Error())
|
|
}
|
|
return &Builder{m}
|
|
}
|
|
|
|
// AsClaim returns a Claim if the receiver Blob has all the required fields.
|
|
func (b *Blob) AsClaim() (c Claim, ok bool) {
|
|
if b.ss.Signer.Valid() && b.ss.Sig != "" && b.ss.ClaimType != "" && !b.ss.ClaimDate.IsZero() {
|
|
return Claim{b}, true
|
|
}
|
|
return
|
|
}
|
|
|
|
// AsShare returns a Share if the receiver Blob has all the required fields.
|
|
func (b *Blob) AsShare() (s Share, ok bool) {
|
|
c, ok := b.AsClaim()
|
|
if !ok {
|
|
return
|
|
}
|
|
if b.ss.ClaimType == claimTypeShare && b.ss.AuthType == ShareHaveRef && b.ss.Target.Valid() {
|
|
return Share{c}, true
|
|
}
|
|
return
|
|
}
|
|
|
|
// DirectoryEntries the "entries" field if valid and b's type is "directory".
|
|
func (b *Blob) DirectoryEntries() (br blob.Ref, ok bool) {
|
|
if b.Type() != "directory" {
|
|
return
|
|
}
|
|
return b.ss.Entries, true
|
|
}
|
|
|
|
func (b *Blob) StaticSetMembers() []blob.Ref {
|
|
if b.Type() != "static-set" {
|
|
return nil
|
|
}
|
|
s := make([]blob.Ref, 0, len(b.ss.Members))
|
|
for _, ref := range b.ss.Members {
|
|
if ref.Valid() {
|
|
s = append(s, ref)
|
|
}
|
|
}
|
|
return s
|
|
}
|
|
|
|
func (b *Blob) ShareAuthType() string {
|
|
s, ok := b.AsShare()
|
|
if !ok {
|
|
return ""
|
|
}
|
|
return s.AuthType()
|
|
}
|
|
|
|
func (b *Blob) ShareTarget() blob.Ref {
|
|
s, ok := b.AsShare()
|
|
if !ok {
|
|
return blob.Ref{}
|
|
}
|
|
return s.Target()
|
|
}
|
|
|
|
// ModTime returns the "unixMtime" field, or the zero time.
|
|
func (b *Blob) ModTime() time.Time { return b.ss.ModTime() }
|
|
|
|
// A Claim is a Blob that is signed.
|
|
type Claim struct {
|
|
b *Blob
|
|
}
|
|
|
|
// Blob returns the claim's Blob.
|
|
func (c Claim) Blob() *Blob { return c.b }
|
|
|
|
// ClaimDate returns the blob's "claimDate" field.
|
|
func (c Claim) ClaimDateString() string { return c.b.ss.ClaimDate.String() }
|
|
|
|
// ClaimType returns the blob's "claimType" field.
|
|
func (c Claim) ClaimType() string { return c.b.ss.ClaimType }
|
|
|
|
// Attribute returns the "attribute" field, if set.
|
|
func (c Claim) Attribute() string { return c.b.ss.Attribute }
|
|
|
|
// Value returns the "value" field, if set.
|
|
func (c Claim) Value() string { return c.b.ss.Value }
|
|
|
|
// ModifiedPermanode returns the claim's "permaNode" field, if it's
|
|
// a claim that modifies a permanode. Otherwise nil is returned.
|
|
func (c Claim) ModifiedPermanode() blob.Ref {
|
|
return c.b.ss.Permanode
|
|
}
|
|
|
|
// A Share is a claim for giving access to a user's blob(s).
|
|
// When returned from (*Blob).AsShare, it always represents
|
|
// a valid share with all required fields.
|
|
type Share struct {
|
|
Claim
|
|
}
|
|
|
|
// AuthType returns the AuthType of the Share.
|
|
func (s Share) AuthType() string {
|
|
return s.b.ss.AuthType
|
|
}
|
|
|
|
// Target returns the blob referenced by the Share.
|
|
func (s Share) Target() blob.Ref {
|
|
return s.b.ss.Target
|
|
}
|
|
|
|
// IsTransitive returns whether the Share transitively
|
|
// gives access to everything reachable from the referenced
|
|
// blob.
|
|
func (s Share) IsTransitive() bool {
|
|
return s.b.ss.Transitive
|
|
}
|
|
|
|
// A Builder builds a JSON blob.
|
|
// After mutating the Builder, call Blob to get the built blob.
|
|
type Builder struct {
|
|
m map[string]interface{}
|
|
}
|
|
|
|
// NewBuilder returns a new blob schema builder.
|
|
// The "camliVersion" field is set to "1" by default and the required
|
|
// "camliType" field is NOT set.
|
|
func NewBuilder() *Builder {
|
|
return &Builder{map[string]interface{}{
|
|
"camliVersion": "1",
|
|
}}
|
|
}
|
|
|
|
// SetRawStringField sets a raw string field in the underlying map.
|
|
func (bb *Builder) SetRawStringField(key, value string) *Builder {
|
|
bb.m[key] = value
|
|
return bb
|
|
}
|
|
|
|
// Blob builds the Blob. The builder continues to be usable after a call to Build.
|
|
func (bb *Builder) Blob() *Blob {
|
|
json, err := mapJSON(bb.m)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
ss, err := parseSuperset(strings.NewReader(json))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
h := blob.NewHash()
|
|
h.Write([]byte(json))
|
|
return &Blob{
|
|
str: json,
|
|
ss: ss,
|
|
br: blob.RefFromHash(h),
|
|
}
|
|
}
|
|
|
|
// Builder returns a clone of itself and satisifies the Buildable interface.
|
|
func (bb *Builder) Builder() *Builder {
|
|
return &Builder{clone(bb.m).(map[string]interface{})}
|
|
}
|
|
|
|
// JSON returns the JSON of the blob as built so far.
|
|
func (bb *Builder) JSON() (string, error) {
|
|
return mapJSON(bb.m)
|
|
}
|
|
|
|
// SetSigner sets the camliSigner field.
|
|
func (bb *Builder) SetSigner(signer blob.Ref) *Builder {
|
|
bb.m["camliSigner"] = signer.String()
|
|
return bb
|
|
}
|
|
|
|
// SetType sets the camliType field.
|
|
func (bb *Builder) SetType(t string) *Builder {
|
|
bb.m["camliType"] = t
|
|
return bb
|
|
}
|
|
|
|
// Type returns the camliType value.
|
|
func (bb *Builder) Type() string {
|
|
if s, ok := bb.m["camliType"].(string); ok {
|
|
return s
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// SetFileName sets the fileName or fileNameBytes field.
|
|
// The filename is truncated to just the base.
|
|
func (bb *Builder) SetFileName(name string) *Builder {
|
|
baseName := filepath.Base(name)
|
|
if utf8.ValidString(baseName) {
|
|
bb.m["fileName"] = baseName
|
|
} else {
|
|
bb.m["fileNameBytes"] = []uint8(baseName)
|
|
}
|
|
return bb
|
|
}
|
|
|
|
// SetSymlinkTarget sets bb to be of type "symlink" and sets the symlink's target.
|
|
func (bb *Builder) SetSymlinkTarget(target string) *Builder {
|
|
bb.SetType("symlink")
|
|
if utf8.ValidString(target) {
|
|
bb.m["symlinkTarget"] = target
|
|
} else {
|
|
bb.m["symlinkTargetBytes"] = []uint8(target)
|
|
}
|
|
return bb
|
|
}
|
|
|
|
// IsClaimType returns whether this blob builder is for a type
|
|
// which should be signed. (a "claim" or "permanode")
|
|
func (bb *Builder) IsClaimType() bool {
|
|
switch bb.Type() {
|
|
case "claim", "permanode":
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// SetClaimDate sets the "claimDate" on a claim.
|
|
// It is a fatal error to call SetClaimDate if the Map isn't of Type "claim".
|
|
func (bb *Builder) SetClaimDate(t time.Time) *Builder {
|
|
if !bb.IsClaimType() {
|
|
// This is a little gross, using panic here, but I
|
|
// don't want all callers to check errors. This is
|
|
// really a programming error, not a runtime error
|
|
// that would arise from e.g. random user data.
|
|
panic("SetClaimDate called on non-claim *Builder; camliType=" + bb.Type())
|
|
}
|
|
bb.m["claimDate"] = RFC3339FromTime(t)
|
|
return bb
|
|
}
|
|
|
|
// SetModTime sets the "unixMtime" field.
|
|
func (bb *Builder) SetModTime(t time.Time) *Builder {
|
|
bb.m["unixMtime"] = RFC3339FromTime(t)
|
|
return bb
|
|
}
|
|
|
|
// ModTime returns the "unixMtime" modtime field, if set.
|
|
func (bb *Builder) ModTime() (t time.Time, ok bool) {
|
|
s, ok := bb.m["unixMtime"].(string)
|
|
if !ok {
|
|
return
|
|
}
|
|
t, err := time.Parse(time.RFC3339, s)
|
|
if err != nil {
|
|
return
|
|
}
|
|
return t, true
|
|
}
|
|
|
|
// PopulateDirectoryMap sets the type of *Builder to "directory" and sets
|
|
// the "entries" field to the provided staticSet blobref.
|
|
func (bb *Builder) PopulateDirectoryMap(staticSetRef blob.Ref) *Builder {
|
|
bb.m["camliType"] = "directory"
|
|
bb.m["entries"] = staticSetRef.String()
|
|
return bb
|
|
}
|
|
|
|
// PartsSize returns the number of bytes represented by the "parts" field.
|
|
func (bb *Builder) PartsSize() int64 {
|
|
n := int64(0)
|
|
if parts, ok := bb.m["parts"].([]BytesPart); ok {
|
|
for _, part := range parts {
|
|
n += int64(part.Size)
|
|
}
|
|
}
|
|
return n
|
|
}
|
|
|
|
func clone(i interface{}) interface{} {
|
|
switch t := i.(type) {
|
|
case map[string]interface{}:
|
|
m2 := make(map[string]interface{})
|
|
for k, v := range t {
|
|
m2[k] = clone(v)
|
|
}
|
|
return m2
|
|
case string, int, int64, float64:
|
|
return t
|
|
case []interface{}:
|
|
s2 := make([]interface{}, len(t))
|
|
for i, v := range t {
|
|
s2[i] = clone(v)
|
|
}
|
|
return s2
|
|
}
|
|
panic(fmt.Sprintf("unsupported clone type %T", i))
|
|
}
|