mirror of https://github.com/perkeep/perkeep.git
615 lines
16 KiB
Go
615 lines
16 KiB
Go
/*
|
|
Copyright 2013 The Perkeep Authors
|
|
|
|
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 (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"path/filepath"
|
|
"strings"
|
|
"time"
|
|
"unicode/utf8"
|
|
|
|
"perkeep.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 Perkeep 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.IsAnyZero() {
|
|
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{}
|
|
dec := json.NewDecoder(strings.NewReader(b.str))
|
|
dec.UseNumber()
|
|
err := dec.Decode(&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.IsAnyZero() {
|
|
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, isClaim := b.AsClaim()
|
|
if !isClaim {
|
|
return
|
|
}
|
|
|
|
if ClaimType(b.ss.ClaimType) == ShareClaim && b.ss.AuthType == ShareHaveRef && (b.ss.Target.Valid() || b.ss.Search != nil) {
|
|
return Share{c}, true
|
|
}
|
|
return s, false
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
// StaticSetMembers returns the refs in the "members" field if b is a valid
|
|
// "static-set" schema. Note that if it is a large static-set, the members are
|
|
// actually spread as subsets in "mergeSets". See StaticSetMergeSets.
|
|
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
|
|
}
|
|
|
|
// StaticSetMergeSets returns the refs of the static-sets in "mergeSets". These
|
|
// are the subsets of all the static-set members in the case of a large directory.
|
|
func (b *Blob) StaticSetMergeSets() []blob.Ref {
|
|
if b.Type() != "static-set" {
|
|
return nil
|
|
}
|
|
|
|
s := make([]blob.Ref, 0, len(b.ss.MergeSets))
|
|
for _, ref := range b.ss.MergeSets {
|
|
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 }
|
|
|
|
// ClaimDateString 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 a zero blob.Ref is
|
|
// returned.
|
|
func (c Claim) ModifiedPermanode() blob.Ref {
|
|
return c.b.ss.Permanode
|
|
}
|
|
|
|
// Target returns the blob referenced by the Share if it's
|
|
// a ShareClaim claim, or the object being deleted if it's a
|
|
// DeleteClaim claim.
|
|
// Otherwise a zero blob.Ref is returned.
|
|
func (c Claim) Target() blob.Ref {
|
|
return c.b.ss.Target
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
// IsExpired reports whether this share has expired.
|
|
func (s Share) IsExpired() bool {
|
|
t := time.Time(s.b.ss.Expires)
|
|
return !t.IsZero() && clockNow().After(t)
|
|
}
|
|
|
|
// A StaticFile is a Blob representing a file, symlink fifo or socket
|
|
// (or device file, when support for these is added).
|
|
type StaticFile struct {
|
|
b *Blob
|
|
}
|
|
|
|
// FileName returns the StaticFile's FileName if is not the empty string, otherwise it returns its FileNameBytes concatenated into a string.
|
|
func (sf StaticFile) FileName() string {
|
|
return sf.b.ss.FileNameString()
|
|
}
|
|
|
|
// AsStaticFile returns the Blob as a StaticFile if it represents
|
|
// one. Otherwise, it returns false in the boolean parameter and the
|
|
// zero value of StaticFile.
|
|
func (b *Blob) AsStaticFile() (sf StaticFile, ok bool) {
|
|
// TODO (marete) Add support for device files to
|
|
// Perkeep and change the implementation of StaticFile to
|
|
// reflect that.
|
|
t := b.ss.Type
|
|
if t == "file" || t == "symlink" || t == "fifo" || t == "socket" {
|
|
return StaticFile{b}, true
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// A StaticFIFO is a StaticFile that is also a fifo.
|
|
type StaticFIFO struct {
|
|
StaticFile
|
|
}
|
|
|
|
// A StaticSocket is a StaticFile that is also a socket.
|
|
type StaticSocket struct {
|
|
StaticFile
|
|
}
|
|
|
|
// A StaticSymlink is a StaticFile that is also a symbolic link.
|
|
type StaticSymlink struct {
|
|
// We name it `StaticSymlink' rather than just `Symlink' since
|
|
// a type called Symlink is already in schema.go.
|
|
StaticFile
|
|
}
|
|
|
|
// SymlinkTargetString returns the field symlinkTarget if is
|
|
// non-empty. Otherwise it returns the contents of symlinkTargetBytes
|
|
// concatenated as a string.
|
|
func (sl StaticSymlink) SymlinkTargetString() string {
|
|
return sl.StaticFile.b.ss.SymlinkTargetString()
|
|
}
|
|
|
|
// AsStaticSymlink returns the StaticFile as a StaticSymlink if the
|
|
// StaticFile represents a symlink. Othwerwise, it returns the zero
|
|
// value of StaticSymlink and false.
|
|
func (sf StaticFile) AsStaticSymlink() (s StaticSymlink, ok bool) {
|
|
if sf.b.ss.Type == "symlink" {
|
|
return StaticSymlink{sf}, true
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// AsStaticFIFO returns the StatifFile as a StaticFIFO if the
|
|
// StaticFile represents a fifo. Otherwise, it returns the zero value
|
|
// of StaticFIFO and false.
|
|
func (sf StaticFile) AsStaticFIFO() (fifo StaticFIFO, ok bool) {
|
|
if sf.b.ss.Type == "fifo" {
|
|
return StaticFIFO{sf}, true
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// AsStaticSocket returns the StaticFile as a StaticSocket if the
|
|
// StaticFile represents a socket. Otherwise, it returns the zero
|
|
// value of StaticSocket and false.
|
|
func (sf StaticFile) AsStaticSocket() (ss StaticSocket, ok bool) {
|
|
if sf.b.ss.Type == "socket" {
|
|
return StaticSocket{sf}, true
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// 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",
|
|
}}
|
|
}
|
|
|
|
// SetShareTarget sets the target of share claim.
|
|
// It panics if bb isn't a "share" claim type.
|
|
func (bb *Builder) SetShareTarget(t blob.Ref) *Builder {
|
|
if bb.Type() != "claim" || bb.ClaimType() != ShareClaim {
|
|
panic("called SetShareTarget on non-share")
|
|
}
|
|
bb.m["target"] = t.String()
|
|
return bb
|
|
}
|
|
|
|
// SetShareSearch sets the search of share claim.
|
|
// q is assumed to be of type *search.SearchQuery.
|
|
// It panics if bb isn't a "share" claim type.
|
|
func (bb *Builder) SetShareSearch(q SearchQuery) *Builder {
|
|
if bb.Type() != "claim" || bb.ClaimType() != ShareClaim {
|
|
panic("called SetShareSearch on non-share")
|
|
}
|
|
bb.m["search"] = q
|
|
return bb
|
|
}
|
|
|
|
// SetShareExpiration sets the expiration time on share claim.
|
|
// It panics if bb isn't a "share" claim type.
|
|
// If t is zero, the expiration is removed.
|
|
func (bb *Builder) SetShareExpiration(t time.Time) *Builder {
|
|
if bb.Type() != "claim" || bb.ClaimType() != ShareClaim {
|
|
panic("called SetShareExpiration on non-share")
|
|
}
|
|
if t.IsZero() {
|
|
delete(bb.m, "expires")
|
|
} else {
|
|
bb.m["expires"] = RFC3339FromTime(t)
|
|
}
|
|
return bb
|
|
}
|
|
|
|
func (bb *Builder) SetShareIsTransitive(b bool) *Builder {
|
|
if bb.Type() != "claim" || bb.ClaimType() != ShareClaim {
|
|
panic("called SetShareIsTransitive on non-share")
|
|
}
|
|
if !b {
|
|
delete(bb.m, "transitive")
|
|
} else {
|
|
bb.m["transitive"] = true
|
|
}
|
|
return bb
|
|
}
|
|
|
|
// 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.
|
|
// Calling SetSigner is unnecessary if using Sign.
|
|
func (bb *Builder) SetSigner(signer blob.Ref) *Builder {
|
|
bb.m["camliSigner"] = signer.String()
|
|
return bb
|
|
}
|
|
|
|
// Sign sets the blob builder's camliSigner field with SetSigner
|
|
// and returns the signed JSON using the provided signer.
|
|
func (bb *Builder) Sign(ctx context.Context, signer *Signer) (string, error) {
|
|
return bb.SignAt(ctx, signer, time.Time{})
|
|
}
|
|
|
|
// SignAt sets the blob builder's camliSigner field with SetSigner
|
|
// and returns the signed JSON using the provided signer.
|
|
// The provided sigTime is the time of the signature, used mostly
|
|
// for planned permanodes. If the zero value, the current time is used.
|
|
func (bb *Builder) SignAt(ctx context.Context, signer *Signer, sigTime time.Time) (string, error) {
|
|
switch bb.Type() {
|
|
case "permanode", "claim":
|
|
default:
|
|
return "", fmt.Errorf("can't sign camliType %q", bb.Type())
|
|
}
|
|
if sigTime.IsZero() {
|
|
sigTime = time.Now()
|
|
}
|
|
bb.SetClaimDate(sigTime)
|
|
return signer.SignJSON(ctx, bb.SetSigner(signer.pubref).Blob().JSON(), sigTime)
|
|
}
|
|
|
|
// 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 ""
|
|
}
|
|
|
|
// ClaimType returns the claimType value, or the empty string.
|
|
func (bb *Builder) ClaimType() ClaimType {
|
|
if s, ok := bb.m["claimType"].(string); ok {
|
|
return ClaimType(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"] = mixedArrayFromString(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"] = mixedArrayFromString(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
|
|
}
|
|
|
|
// CapCreationTime caps the "unixCtime" field to be less or equal than "unixMtime"
|
|
func (bb *Builder) CapCreationTime() *Builder {
|
|
ctime, ok := bb.m["unixCtime"].(string)
|
|
if !ok {
|
|
return bb
|
|
}
|
|
mtime, ok := bb.m["unixMtime"].(string)
|
|
if ok && ctime > mtime {
|
|
bb.m["unixCtime"] = mtime
|
|
}
|
|
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, json.Number:
|
|
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))
|
|
}
|