Compare commits
No commits in common. "8545e098c12a2a4829aec3e9b45416fe175b4be8" and "c8b33572dae11ca054026000fefaa056af52a808" have entirely different histories.
8545e098c1
...
c8b33572da
|
@ -7,7 +7,10 @@ package api
|
|||
// Parameters:
|
||||
// uuid: string PhotoUID as returned by the API
|
||||
func (v1 *V1Client) GetPhoto(uuid string) (Photo, error) {
|
||||
photo := Photo{}
|
||||
photo := Photo{
|
||||
UUID: uuid,
|
||||
PhotoUID: uuid,
|
||||
}
|
||||
err := v1.GET("/api/v1/photos/%s", uuid).JSON(&photo)
|
||||
return photo, err
|
||||
}
|
||||
|
|
475
api/v1/types.go
|
@ -1,6 +1,7 @@
|
|||
package api
|
||||
|
||||
import (
|
||||
"database/sql"
|
||||
"time"
|
||||
)
|
||||
|
||||
|
@ -8,237 +9,335 @@ type Photos []Photo
|
|||
|
||||
// Photo represents a photo, all its properties, and link to all its images and sidecar files.
|
||||
type Photo struct {
|
||||
PhotoID int `json:"ID,omitempty"`
|
||||
TakenAt time.Time `json:"TakenAt"`
|
||||
TakenAtLocal time.Time `json:"TakenAtLocal"`
|
||||
TakenSrc string `json:"TakenSrc"`
|
||||
PhotoUID string `json:"UID"`
|
||||
PhotoType string `json:"Type"`
|
||||
TypeSrc string `json:"TypeSrc"`
|
||||
PhotoTitle string `json:"Title"`
|
||||
TitleSrc string `json:"TitleSrc"`
|
||||
PhotoDescription string `json:"Description"`
|
||||
DescriptionSrc string `json:"DescriptionSrc"`
|
||||
PhotoPath string `json:"Path"`
|
||||
PhotoName string `json:"Name"`
|
||||
OriginalName string `json:"OriginalName"`
|
||||
PhotoStack int8 `json:"Stack"`
|
||||
PhotoFavorite bool `json:"Favorite"`
|
||||
PhotoPrivate bool `json:"Private"`
|
||||
PhotoScan bool `json:"Scan"`
|
||||
PhotoPanorama bool `json:"Panorama"`
|
||||
TimeZone string `json:"TimeZone"`
|
||||
PlaceID string `json:"PlaceID"`
|
||||
PlaceSrc string `json:"PlaceSrc"`
|
||||
CellID string `json:"CellID"`
|
||||
CellAccuracy int `json:"CellAccuracy"`
|
||||
PhotoAltitude int `json:"Altitude"`
|
||||
PhotoLat float32 `json:"Lat"`
|
||||
PhotoLng float32 `json:"Lng"`
|
||||
PhotoCountry string `json:"Country"`
|
||||
PhotoYear int `json:"Year"`
|
||||
PhotoMonth int `json:"Month"`
|
||||
PhotoDay int `json:"Day"`
|
||||
PhotoIso int `json:"Iso"`
|
||||
PhotoExposure string `json:"Exposure"`
|
||||
PhotoFNumber float32 `json:"FNumber"`
|
||||
PhotoFocalLength int `json:"FocalLength"`
|
||||
PhotoQuality int `json:"Quality"`
|
||||
PhotoResolution int `json:"Resolution"`
|
||||
PhotoColor uint8 `json:"Color"`
|
||||
CameraID uint `json:"CameraID"`
|
||||
CameraSerial string `json:"CameraSerial"`
|
||||
CameraSrc string `json:"CameraSrc"`
|
||||
LensID uint `json:"LensID"`
|
||||
Details *Details `json:"Details"`
|
||||
Camera *Camera `json:"Camera"`
|
||||
Lens *Lens `json:"Lens"`
|
||||
Cell *Cell `json:"Cell"`
|
||||
Place *Place `json:"Place"`
|
||||
Files []File `json:"Files"`
|
||||
Labels []PhotoLabel `json:"Labels"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
EditedAt *time.Time `json:"EditedAt"`
|
||||
CheckedAt *time.Time `json:"CheckedAt"`
|
||||
DeletedAt *time.Time `json:"DeletedAt"`
|
||||
UUID string `gorm:"type:VARBINARY(42);index;" json:"DocumentID,omitempty" yaml:"DocumentID,omitempty"`
|
||||
TakenAt time.Time `gorm:"type:datetime;index:idx_photos_taken_uid;" json:"TakenAt" yaml:"TakenAt"`
|
||||
TakenAtLocal time.Time `gorm:"type:datetime;" yaml:"-"`
|
||||
TakenSrc string `gorm:"type:VARBINARY(8);" json:"TakenSrc" yaml:"TakenSrc,omitempty"`
|
||||
PhotoUID string `gorm:"type:VARBINARY(42);unique_index;index:idx_photos_taken_uid;" json:"UID" yaml:"UID"`
|
||||
PhotoType string `gorm:"type:VARBINARY(8);default:'image';" json:"Type" yaml:"Type"`
|
||||
TypeSrc string `gorm:"type:VARBINARY(8);" json:"TypeSrc" yaml:"TypeSrc,omitempty"`
|
||||
PhotoTitle string `gorm:"type:VARCHAR(255);" json:"Title" yaml:"Title"`
|
||||
TitleSrc string `gorm:"type:VARBINARY(8);" json:"TitleSrc" yaml:"TitleSrc,omitempty"`
|
||||
PhotoDescription string `gorm:"type:TEXT;" json:"Description" yaml:"Description,omitempty"`
|
||||
DescriptionSrc string `gorm:"type:VARBINARY(8);" json:"DescriptionSrc" yaml:"DescriptionSrc,omitempty"`
|
||||
PhotoPath string `gorm:"type:VARBINARY(500);index:idx_photos_path_name;" json:"Path" yaml:"-"`
|
||||
PhotoName string `gorm:"type:VARBINARY(255);index:idx_photos_path_name;" json:"Name" yaml:"-"`
|
||||
OriginalName string `gorm:"type:VARBINARY(755);" json:"OriginalName" yaml:"OriginalName,omitempty"`
|
||||
PhotoStack int8 `json:"Stack" yaml:"Stack,omitempty"`
|
||||
PhotoFavorite bool `json:"Favorite" yaml:"Favorite,omitempty"`
|
||||
PhotoPrivate bool `json:"Private" yaml:"Private,omitempty"`
|
||||
PhotoScan bool `json:"Scan" yaml:"Scan,omitempty"`
|
||||
PhotoPanorama bool `json:"Panorama" yaml:"Panorama,omitempty"`
|
||||
TimeZone string `gorm:"type:VARBINARY(64);" json:"TimeZone" yaml:"-"`
|
||||
PlaceID string `gorm:"type:VARBINARY(42);index;default:'zz'" json:"PlaceID" yaml:"-"`
|
||||
PlaceSrc string `gorm:"type:VARBINARY(8);" json:"PlaceSrc" yaml:"PlaceSrc,omitempty"`
|
||||
CellID string `gorm:"type:VARBINARY(42);index;default:'zz'" json:"CellID" yaml:"-"`
|
||||
CellAccuracy int `json:"CellAccuracy" yaml:"CellAccuracy,omitempty"`
|
||||
PhotoAltitude int `json:"Altitude" yaml:"Altitude,omitempty"`
|
||||
PhotoLat float32 `gorm:"type:FLOAT;index;" json:"Lat" yaml:"Lat,omitempty"`
|
||||
PhotoLng float32 `gorm:"type:FLOAT;index;" json:"Lng" yaml:"Lng,omitempty"`
|
||||
PhotoCountry string `gorm:"type:VARBINARY(2);index:idx_photos_country_year_month;default:'zz'" json:"Country" yaml:"-"`
|
||||
PhotoYear int `gorm:"index:idx_photos_country_year_month;" json:"Year" yaml:"Year"`
|
||||
PhotoMonth int `gorm:"index:idx_photos_country_year_month;" json:"Month" yaml:"Month"`
|
||||
PhotoDay int `json:"Day" yaml:"Day"`
|
||||
PhotoIso int `json:"Iso" yaml:"ISO,omitempty"`
|
||||
PhotoExposure string `gorm:"type:VARBINARY(64);" json:"Exposure" yaml:"Exposure,omitempty"`
|
||||
PhotoFNumber float32 `gorm:"type:FLOAT;" json:"FNumber" yaml:"FNumber,omitempty"`
|
||||
PhotoFocalLength int `json:"FocalLength" yaml:"FocalLength,omitempty"`
|
||||
PhotoQuality int `gorm:"type:SMALLINT" json:"Quality" yaml:"-"`
|
||||
PhotoResolution int `gorm:"type:SMALLINT" json:"Resolution" yaml:"-"`
|
||||
PhotoColor uint8 `json:"Color" yaml:"-"`
|
||||
CameraID uint `gorm:"index:idx_photos_camera_lens;default:1" json:"CameraID" yaml:"-"`
|
||||
CameraSerial string `gorm:"type:VARBINARY(255);" json:"CameraSerial" yaml:"CameraSerial,omitempty"`
|
||||
CameraSrc string `gorm:"type:VARBINARY(8);" json:"CameraSrc" yaml:"-"`
|
||||
LensID uint `gorm:"index:idx_photos_camera_lens;default:1" json:"LensID" yaml:"-"`
|
||||
Details *Details `gorm:"association_autoupdate:false;association_autocreate:false;association_save_reference:false" json:"Details" yaml:"Details"`
|
||||
Camera *Camera `gorm:"association_autoupdate:false;association_autocreate:false;association_save_reference:false" json:"Camera" yaml:"-"`
|
||||
Lens *Lens `gorm:"association_autoupdate:false;association_autocreate:false;association_save_reference:false" json:"Lens" yaml:"-"`
|
||||
Cell *Cell `gorm:"association_autoupdate:false;association_autocreate:false;association_save_reference:false" json:"Cell" yaml:"-"`
|
||||
Place *Place `gorm:"association_autoupdate:false;association_autocreate:false;association_save_reference:false" json:"Place" yaml:"-"`
|
||||
Keywords []Keyword `json:"-" yaml:"-"`
|
||||
Albums []Album `json:"-" yaml:"-"`
|
||||
Files []File `yaml:"-"`
|
||||
Labels []PhotoLabel `yaml:"-"`
|
||||
CreatedAt time.Time `yaml:"CreatedAt,omitempty"`
|
||||
UpdatedAt time.Time `yaml:"UpdatedAt,omitempty"`
|
||||
EditedAt *time.Time `yaml:"EditedAt,omitempty"`
|
||||
CheckedAt *time.Time `sql:"index" yaml:"-"`
|
||||
DeletedAt *time.Time `sql:"index" yaml:"DeletedAt,omitempty"`
|
||||
}
|
||||
|
||||
// Details stores additional metadata fields for each photo to improve search performance.
|
||||
type Details struct {
|
||||
PhotoID uint `json:"PhotoID"`
|
||||
Keywords string `json:"Keywords"`
|
||||
KeywordsSrc string `json:"KeywordsSrc"`
|
||||
Notes string `json:"Notes"`
|
||||
NotesSrc string `json:"NotesSrc"`
|
||||
Subject string `json:"Subject"`
|
||||
SubjectSrc string `json:"SubjectSrc"`
|
||||
Artist string `json:"Artist"`
|
||||
ArtistSrc string `json:"ArtistSrc"`
|
||||
Copyright string `json:"Copyright"`
|
||||
CopyrightSrc string `json:"CopyrightSrc"`
|
||||
License string `json:"License"`
|
||||
LicenseSrc string `json:"LicenseSrc"`
|
||||
Software string `json:"Software"`
|
||||
SoftwareSrc string `json:"SoftwareSrc"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
PhotoID uint `gorm:"primary_key;auto_increment:false" yaml:"-"`
|
||||
Keywords string `gorm:"type:TEXT;" json:"Keywords" yaml:"Keywords"`
|
||||
KeywordsSrc string `gorm:"type:VARBINARY(8);" json:"KeywordsSrc" yaml:"KeywordsSrc,omitempty"`
|
||||
Notes string `gorm:"type:TEXT;" json:"Notes" yaml:"Notes,omitempty"`
|
||||
NotesSrc string `gorm:"type:VARBINARY(8);" json:"NotesSrc" yaml:"NotesSrc,omitempty"`
|
||||
Subject string `gorm:"type:VARCHAR(255);" json:"Subject" yaml:"Subject,omitempty"`
|
||||
SubjectSrc string `gorm:"type:VARBINARY(8);" json:"SubjectSrc" yaml:"SubjectSrc,omitempty"`
|
||||
Artist string `gorm:"type:VARCHAR(255);" json:"Artist" yaml:"Artist,omitempty"`
|
||||
ArtistSrc string `gorm:"type:VARBINARY(8);" json:"ArtistSrc" yaml:"ArtistSrc,omitempty"`
|
||||
Copyright string `gorm:"type:VARCHAR(255);" json:"Copyright" yaml:"Copyright,omitempty"`
|
||||
CopyrightSrc string `gorm:"type:VARBINARY(8);" json:"CopyrightSrc" yaml:"CopyrightSrc,omitempty"`
|
||||
License string `gorm:"type:VARCHAR(255);" json:"License" yaml:"License,omitempty"`
|
||||
LicenseSrc string `gorm:"type:VARBINARY(8);" json:"LicenseSrc" yaml:"LicenseSrc,omitempty"`
|
||||
CreatedAt time.Time `yaml:"-"`
|
||||
UpdatedAt time.Time `yaml:"-"`
|
||||
}
|
||||
|
||||
// Camera model and make (as extracted from UpdateExif metadata)
|
||||
type Camera struct {
|
||||
ID uint `json:"ID"`
|
||||
CameraSlug string `json:"Slug"`
|
||||
CameraName string `json:"Name"`
|
||||
CameraMake string `json:"Make"`
|
||||
CameraModel string `json:"Model"`
|
||||
CameraType string `json:"Type,omitempty"`
|
||||
CameraDescription string `json:"Description,omitempty"`
|
||||
CameraNotes string `json:"Notes,omitempty"`
|
||||
ID uint `gorm:"primary_key" json:"ID" yaml:"ID"`
|
||||
CameraSlug string `gorm:"type:VARBINARY(255);unique_index;" json:"Slug" yaml:"-"`
|
||||
CameraName string `gorm:"type:VARCHAR(255);" json:"Name" yaml:"Name"`
|
||||
CameraMake string `gorm:"type:VARCHAR(255);" json:"Make" yaml:"Make,omitempty"`
|
||||
CameraModel string `gorm:"type:VARCHAR(255);" json:"Model" yaml:"Model,omitempty"`
|
||||
CameraType string `gorm:"type:VARCHAR(255);" json:"Type,omitempty" yaml:"Type,omitempty"`
|
||||
CameraDescription string `gorm:"type:TEXT;" json:"Description,omitempty" yaml:"Description,omitempty"`
|
||||
CameraNotes string `gorm:"type:TEXT;" json:"Notes,omitempty" yaml:"Notes,omitempty"`
|
||||
CreatedAt time.Time `json:"-" yaml:"-"`
|
||||
UpdatedAt time.Time `json:"-" yaml:"-"`
|
||||
DeletedAt *time.Time `sql:"index" json:"-" yaml:"-"`
|
||||
}
|
||||
|
||||
// Lens represents camera lens (as extracted from UpdateExif metadata)
|
||||
type Lens struct {
|
||||
ID uint `json:"ID"`
|
||||
LensSlug string `json:"Slug"`
|
||||
LensName string `json:"Name"`
|
||||
LensMake string `json:"Make"`
|
||||
LensModel string `json:"Model"`
|
||||
LensType string `json:"Type"`
|
||||
LensDescription string `json:"Description,omitempty"`
|
||||
LensNotes string `json:"Notes,omitempty"`
|
||||
ID uint `gorm:"primary_key" json:"ID" yaml:"ID"`
|
||||
LensSlug string `gorm:"type:VARBINARY(255);unique_index;" json:"Slug" yaml:"Slug,omitempty"`
|
||||
LensName string `gorm:"type:VARCHAR(255);" json:"Name" yaml:"Name"`
|
||||
LensMake string `gorm:"type:VARCHAR(255);" json:"Make" yaml:"Make,omitempty"`
|
||||
LensModel string `gorm:"type:VARCHAR(255);" json:"Model" yaml:"Model,omitempty"`
|
||||
LensType string `gorm:"type:VARCHAR(255);" json:"Type" yaml:"Type,omitempty"`
|
||||
LensDescription string `gorm:"type:TEXT;" json:"Description,omitempty" yaml:"Description,omitempty"`
|
||||
LensNotes string `gorm:"type:TEXT;" json:"Notes,omitempty" yaml:"Notes,omitempty"`
|
||||
CreatedAt time.Time `json:"-" yaml:"-"`
|
||||
UpdatedAt time.Time `json:"-" yaml:"-"`
|
||||
DeletedAt *time.Time `sql:"index" json:"-" yaml:"-"`
|
||||
}
|
||||
|
||||
// Cell represents a S2 cell with location data.
|
||||
type Cell struct {
|
||||
ID string `json:"ID"`
|
||||
CellName string `json:"Name"`
|
||||
CellStreet string `json:"Street"`
|
||||
CellPostcode string `json:"Postcode"`
|
||||
CellCategory string `json:"Category"`
|
||||
Place *Place `json:"Place"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
ID string `gorm:"type:VARBINARY(42);primary_key;auto_increment:false;" json:"ID" yaml:"ID"`
|
||||
CellName string `gorm:"type:VARCHAR(255);" json:"Name" yaml:"Name,omitempty"`
|
||||
CellCategory string `gorm:"type:VARCHAR(64);" json:"Category" yaml:"Category,omitempty"`
|
||||
PlaceID string `gorm:"type:VARBINARY(42);default:'zz'" json:"-" yaml:"PlaceID"`
|
||||
Place *Place `gorm:"PRELOAD:true" json:"Place" yaml:"-"`
|
||||
CreatedAt time.Time `json:"CreatedAt" yaml:"-"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
||||
}
|
||||
|
||||
// Place used to associate photos to places
|
||||
type Place struct {
|
||||
ID string `json:"PlaceID"`
|
||||
PlaceLabel string `json:"Label"`
|
||||
PlaceCity string `json:"City"`
|
||||
PlaceState string `json:"State"`
|
||||
PlaceCountry string `json:"Country"`
|
||||
PlaceKeywords string `json:"Keywords"`
|
||||
PlaceFavorite bool `json:"Favorite"`
|
||||
PhotoCount int `json:"PhotoCount"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
ID string `gorm:"type:VARBINARY(42);primary_key;auto_increment:false;" json:"PlaceID" yaml:"PlaceID"`
|
||||
PlaceLabel string `gorm:"type:VARBINARY(755);unique_index;" json:"Label" yaml:"Label"`
|
||||
PlaceCity string `gorm:"type:VARCHAR(255);" json:"City" yaml:"City,omitempty"`
|
||||
PlaceState string `gorm:"type:VARCHAR(255);" json:"State" yaml:"State,omitempty"`
|
||||
PlaceCountry string `gorm:"type:VARBINARY(2);" json:"Country" yaml:"Country,omitempty"`
|
||||
PlaceKeywords string `gorm:"type:VARCHAR(255);" json:"Keywords" yaml:"Keywords,omitempty"`
|
||||
PlaceFavorite bool `json:"Favorite" yaml:"Favorite,omitempty"`
|
||||
PhotoCount int `gorm:"default:1" json:"PhotoCount" yaml:"-"`
|
||||
CreatedAt time.Time `json:"CreatedAt" yaml:"-"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
||||
}
|
||||
|
||||
// Keyword used for full text search
|
||||
type Keyword struct {
|
||||
ID uint `gorm:"primary_key"`
|
||||
Keyword string `gorm:"type:VARCHAR(64);index;"`
|
||||
Skip bool
|
||||
}
|
||||
|
||||
// Album represents a photo album
|
||||
type Album struct {
|
||||
ID uint `json:"ID"`
|
||||
AlbumUID string `json:"UID"`
|
||||
AlbumSlug string `json:"Slug"`
|
||||
AlbumType string `json:"Type"`
|
||||
AlbumTitle string `json:"Title"`
|
||||
AlbumLocation string `json:"Location"`
|
||||
AlbumCategory string `json:"Category"`
|
||||
AlbumCaption string `json:"Caption"`
|
||||
AlbumDescription string `json:"Description"`
|
||||
AlbumNotes string `json:"Notes"`
|
||||
AlbumFilter string `json:"Filter"`
|
||||
AlbumOrder string `json:"Order"`
|
||||
AlbumTemplate string `json:"Template"`
|
||||
AlbumCountry string `json:"Country"`
|
||||
AlbumYear int `json:"Year"`
|
||||
AlbumMonth int `json:"Month"`
|
||||
AlbumDay int `json:"Day"`
|
||||
AlbumFavorite bool `json:"Favorite"`
|
||||
AlbumPrivate bool `json:"Private"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
DeletedAt *time.Time `json:"DeletedAt"`
|
||||
ID uint `gorm:"primary_key" json:"ID" yaml:"-"`
|
||||
AlbumUID string `gorm:"type:VARBINARY(42);unique_index;" json:"UID" yaml:"UID"`
|
||||
CoverUID string `gorm:"type:VARBINARY(42);" json:"CoverUID" yaml:"CoverUID,omitempty"`
|
||||
FolderUID string `gorm:"type:VARBINARY(42);index;" json:"FolderUID" yaml:"FolderUID,omitempty"`
|
||||
AlbumSlug string `gorm:"type:VARBINARY(255);index;" json:"Slug" yaml:"Slug"`
|
||||
AlbumPath string `gorm:"type:VARBINARY(500);index;" json:"Path" yaml:"-"`
|
||||
AlbumType string `gorm:"type:VARBINARY(8);default:'album';" json:"Type" yaml:"Type,omitempty"`
|
||||
AlbumTitle string `gorm:"type:VARCHAR(255);" json:"Title" yaml:"Title"`
|
||||
AlbumLocation string `gorm:"type:VARCHAR(255);" json:"Location" yaml:"Location,omitempty"`
|
||||
AlbumCategory string `gorm:"type:VARCHAR(255);index;" json:"Category" yaml:"Category,omitempty"`
|
||||
AlbumCaption string `gorm:"type:TEXT;" json:"Caption" yaml:"Caption,omitempty"`
|
||||
AlbumDescription string `gorm:"type:TEXT;" json:"Description" yaml:"Description,omitempty"`
|
||||
AlbumNotes string `gorm:"type:TEXT;" json:"Notes" yaml:"Notes,omitempty"`
|
||||
AlbumFilter string `gorm:"type:VARBINARY(1024);" json:"Filter" yaml:"Filter,omitempty"`
|
||||
AlbumOrder string `gorm:"type:VARBINARY(32);" json:"Order" yaml:"Order,omitempty"`
|
||||
AlbumTemplate string `gorm:"type:VARBINARY(255);" json:"Template" yaml:"Template,omitempty"`
|
||||
AlbumCountry string `gorm:"type:VARBINARY(2);index:idx_albums_country_year_month;default:'zz'" json:"Country" yaml:"Country,omitempty"`
|
||||
AlbumYear int `gorm:"index:idx_albums_country_year_month;" json:"Year" yaml:"Year,omitempty"`
|
||||
AlbumMonth int `gorm:"index:idx_albums_country_year_month;" json:"Month" yaml:"Month,omitempty"`
|
||||
AlbumDay int `json:"Day" yaml:"Day,omitempty"`
|
||||
AlbumFavorite bool `json:"Favorite" yaml:"Favorite,omitempty"`
|
||||
AlbumPrivate bool `json:"Private" yaml:"Private,omitempty"`
|
||||
CreatedAt time.Time `json:"CreatedAt" yaml:"CreatedAt,omitempty"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt" yaml:"UpdatedAt,omitempty"`
|
||||
DeletedAt *time.Time `sql:"index" json:"DeletedAt" yaml:"DeletedAt,omitempty"`
|
||||
Photos PhotoAlbums `gorm:"foreignkey:AlbumUID;association_foreignkey:AlbumUID" json:"-" yaml:"Photos,omitempty"`
|
||||
}
|
||||
|
||||
type PhotoAlbums []PhotoAlbum
|
||||
|
||||
// PhotoAlbum represents the many_to_many relation between Photo and Album
|
||||
type PhotoAlbum struct {
|
||||
PhotoUID string `json:"PhotoUID"`
|
||||
AlbumUID string `json:"AlbumUID"`
|
||||
Order int `json:"Order"`
|
||||
Hidden bool `json:"Hidden"`
|
||||
Missing bool `json:"Missing"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
Photo *Photo
|
||||
Album *Album
|
||||
PhotoUID string `gorm:"type:VARBINARY(42);primary_key;auto_increment:false" json:"PhotoUID" yaml:"UID"`
|
||||
AlbumUID string `gorm:"type:VARBINARY(42);primary_key;auto_increment:false;index" json:"AlbumUID" yaml:"-"`
|
||||
Order int `json:"Order" yaml:"Order,omitempty"`
|
||||
Hidden bool `json:"Hidden" yaml:"Hidden,omitempty"`
|
||||
Missing bool `json:"Missing" yaml:"Missing,omitempty"`
|
||||
CreatedAt time.Time `json:"CreatedAt" yaml:"CreatedAt,omitempty"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
||||
Photo *Photo `gorm:"PRELOAD:false" yaml:"-"`
|
||||
Album *Album `gorm:"PRELOAD:true" yaml:"-"`
|
||||
}
|
||||
|
||||
type Files []File
|
||||
|
||||
// File represents an image or sidecar file that belongs to a photo.
|
||||
type File struct {
|
||||
PhotoUID string `json:"PhotoUID"`
|
||||
FileUID string `json:"UID"`
|
||||
FileName string `json:"Name"`
|
||||
FileRoot string `json:"Root"`
|
||||
OriginalName string `json:"OriginalName"`
|
||||
FileHash string `json:"Hash"`
|
||||
FileSize int64 `json:"Size"`
|
||||
FileCodec string `json:"Codec"`
|
||||
FileType string `json:"Type"`
|
||||
FileMime string `json:"Mime"`
|
||||
FilePrimary bool `json:"Primary"`
|
||||
FileSidecar bool `json:"Sidecar"`
|
||||
FileMissing bool `json:"Missing"`
|
||||
FilePortrait bool `json:"Portrait"`
|
||||
FileVideo bool `json:"Video"`
|
||||
FileDuration time.Duration `json:"Duration"`
|
||||
FileWidth int `json:"Width"`
|
||||
FileHeight int `json:"Height"`
|
||||
FileOrientation int `json:"Orientation"`
|
||||
FileProjection string `json:"Projection,omitempty"`
|
||||
FileAspectRatio float32 `json:"AspectRatio"`
|
||||
FileMainColor string `json:"MainColor"`
|
||||
FileColors string `json:"Colors"`
|
||||
FileLuminance string `json:"Luminance"`
|
||||
FileDiff uint32 `json:"Diff"`
|
||||
FileChroma uint8 `json:"Chroma"`
|
||||
FileError string `json:"Error"`
|
||||
ModTime int64 `json:"ModTime"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
CreatedIn int64 `json:"CreatedIn"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
UpdatedIn int64 `json:"UpdatedIn"`
|
||||
DeletedAt *time.Time `json:"DeletedAt,omitempty"`
|
||||
ID uint `gorm:"primary_key" json:"-" yaml:"-"`
|
||||
Photo *Photo `json:"-" yaml:"-"`
|
||||
PhotoID uint `gorm:"index;" json:"-" yaml:"-"`
|
||||
PhotoUID string `gorm:"type:VARBINARY(42);index;" json:"PhotoUID" yaml:"PhotoUID"`
|
||||
InstanceID string `gorm:"type:VARBINARY(42);index;" json:"InstanceID,omitempty" yaml:"InstanceID,omitempty"`
|
||||
FileUID string `gorm:"type:VARBINARY(42);unique_index;" json:"UID" yaml:"UID"`
|
||||
FileName string `gorm:"type:VARBINARY(755);unique_index:idx_files_name_root;" json:"Name" yaml:"Name"`
|
||||
FileRoot string `gorm:"type:VARBINARY(16);default:'/';unique_index:idx_files_name_root;" json:"Root" yaml:"Root,omitempty"`
|
||||
OriginalName string `gorm:"type:VARBINARY(755);" json:"OriginalName" yaml:"OriginalName,omitempty"`
|
||||
FileHash string `gorm:"type:VARBINARY(128);index" json:"Hash" yaml:"Hash,omitempty"`
|
||||
FileSize int64 `json:"Size" yaml:"Size,omitempty"`
|
||||
FileCodec string `gorm:"type:VARBINARY(32)" json:"Codec" yaml:"Codec,omitempty"`
|
||||
FileType string `gorm:"type:VARBINARY(32)" json:"Type" yaml:"Type,omitempty"`
|
||||
FileMime string `gorm:"type:VARBINARY(64)" json:"Mime" yaml:"Mime,omitempty"`
|
||||
FilePrimary bool `json:"Primary" yaml:"Primary,omitempty"`
|
||||
FileSidecar bool `json:"Sidecar" yaml:"Sidecar,omitempty"`
|
||||
FileMissing bool `json:"Missing" yaml:"Missing,omitempty"`
|
||||
FilePortrait bool `json:"Portrait" yaml:"Portrait,omitempty"`
|
||||
FileVideo bool `json:"Video" yaml:"Video,omitempty"`
|
||||
FileDuration time.Duration `json:"Duration" yaml:"Duration,omitempty"`
|
||||
FileWidth int `json:"Width" yaml:"Width,omitempty"`
|
||||
FileHeight int `json:"Height" yaml:"Height,omitempty"`
|
||||
FileOrientation int `json:"Orientation" yaml:"Orientation,omitempty"`
|
||||
FileProjection string `gorm:"type:VARBINARY(16);" json:"Projection,omitempty" yaml:"Projection,omitempty"`
|
||||
FileAspectRatio float32 `gorm:"type:FLOAT;" json:"AspectRatio" yaml:"AspectRatio,omitempty"`
|
||||
FileMainColor string `gorm:"type:VARBINARY(16);index;" json:"MainColor" yaml:"MainColor,omitempty"`
|
||||
FileColors string `gorm:"type:VARBINARY(9);" json:"Colors" yaml:"Colors,omitempty"`
|
||||
FileLuminance string `gorm:"type:VARBINARY(9);" json:"Luminance" yaml:"Luminance,omitempty"`
|
||||
FileDiff uint32 `json:"Diff" yaml:"Diff,omitempty"`
|
||||
FileChroma uint8 `json:"Chroma" yaml:"Chroma,omitempty"`
|
||||
FileError string `gorm:"type:VARBINARY(512)" json:"Error" yaml:"Error,omitempty"`
|
||||
ModTime int64 `json:"ModTime" yaml:"-"`
|
||||
CreatedAt time.Time `json:"CreatedAt" yaml:"-"`
|
||||
CreatedIn int64 `json:"CreatedIn" yaml:"-"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
||||
UpdatedIn int64 `json:"UpdatedIn" yaml:"-"`
|
||||
DeletedAt *time.Time `sql:"index" json:"DeletedAt,omitempty" yaml:"-"`
|
||||
Share []FileShare `json:"-" yaml:"-"`
|
||||
Sync []FileSync `json:"-" yaml:"-"`
|
||||
}
|
||||
|
||||
// FileSync represents a one-to-many relation between File and Account for syncing with remote services.
|
||||
type FileSync struct {
|
||||
RemoteName string `gorm:"primary_key;auto_increment:false;type:VARBINARY(255)"`
|
||||
AccountID uint `gorm:"primary_key;auto_increment:false"`
|
||||
FileID uint `gorm:"index;"`
|
||||
RemoteDate time.Time
|
||||
RemoteSize int64
|
||||
Status string `gorm:"type:VARBINARY(16);"`
|
||||
Error string `gorm:"type:VARBINARY(512);"`
|
||||
Errors int
|
||||
File *File
|
||||
Account *Account
|
||||
CreatedAt time.Time
|
||||
UpdatedAt time.Time
|
||||
}
|
||||
|
||||
// FileShare represents a one-to-many relation between File and Account for pushing files to remote services.
|
||||
type FileShare struct {
|
||||
FileID uint `gorm:"primary_key;auto_increment:false"`
|
||||
AccountID uint `gorm:"primary_key;auto_increment:false"`
|
||||
RemoteName string `gorm:"primary_key;auto_increment:false;type:VARBINARY(255)"`
|
||||
Status string `gorm:"type:VARBINARY(16);"`
|
||||
Error string `gorm:"type:VARBINARY(512);"`
|
||||
Errors int
|
||||
File *File
|
||||
Account *Account
|
||||
CreatedAt time.Time
|
||||
UpdatedAt time.Time
|
||||
}
|
||||
|
||||
// PhotoLabel represents the many-to-many relation between Photo and label.
|
||||
// Labels are weighted by uncertainty (100 - confidence)
|
||||
type PhotoLabel struct {
|
||||
PhotoID uint `json:"PhotoID"`
|
||||
LabelID uint `json:"LabelID"`
|
||||
LabelSrc string `json:"LabelSrc""`
|
||||
Uncertainty int `json:"Uncertainty"`
|
||||
Photo *Photo `json:"Photo"`
|
||||
Label *Label `json:"Label"`
|
||||
PhotoID uint `gorm:"primary_key;auto_increment:false"`
|
||||
LabelID uint `gorm:"primary_key;auto_increment:false;index"`
|
||||
LabelSrc string `gorm:"type:VARBINARY(8);"`
|
||||
Uncertainty int `gorm:"type:SMALLINT"`
|
||||
Photo *Photo `gorm:"PRELOAD:false"`
|
||||
Label *Label `gorm:"PRELOAD:true"`
|
||||
}
|
||||
|
||||
// Label is used for photo, album and location categorization
|
||||
type Label struct {
|
||||
ID uint `json:"ID"`
|
||||
LabelUID string `json:"UID"`
|
||||
LabelSlug string `json:"Slug"`
|
||||
CustomSlug string `json:"CustomSlug"`
|
||||
LabelName string `json:"Name"`
|
||||
LabelPriority int `json:"Priority"`
|
||||
LabelFavorite bool `json:"Favorite"`
|
||||
LabelDescription string `json:"Description"`
|
||||
LabelNotes string `json:"Notes"`
|
||||
PhotoCount int `json:"PhotoCount"`
|
||||
LabelThumb string `json:"Thumb"`
|
||||
CreatedAt time.Time `json:"CreatedAt"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt"`
|
||||
ID uint `gorm:"primary_key" json:"ID" yaml:"-"`
|
||||
LabelUID string `gorm:"type:VARBINARY(42);unique_index;" json:"UID" yaml:"UID"`
|
||||
LabelSlug string `gorm:"type:VARBINARY(255);unique_index;" json:"Slug" yaml:"-"`
|
||||
CustomSlug string `gorm:"type:VARBINARY(255);index;" json:"CustomSlug" yaml:"-"`
|
||||
LabelName string `gorm:"type:VARCHAR(255);" json:"Name" yaml:"Name"`
|
||||
LabelPriority int `json:"Priority" yaml:"Priority,omitempty"`
|
||||
LabelFavorite bool `json:"Favorite" yaml:"Favorite,omitempty"`
|
||||
LabelDescription string `gorm:"type:TEXT;" json:"Description" yaml:"Description,omitempty"`
|
||||
LabelNotes string `gorm:"type:TEXT;" json:"Notes" yaml:"Notes,omitempty"`
|
||||
LabelCategories []*Label `gorm:"many2many:categories;association_jointable_foreignkey:category_id" json:"-" yaml:"-"`
|
||||
PhotoCount int `gorm:"default:1" json:"PhotoCount" yaml:"-"`
|
||||
CreatedAt time.Time `json:"CreatedAt" yaml:"-"`
|
||||
UpdatedAt time.Time `json:"UpdatedAt" yaml:"-"`
|
||||
DeletedAt *time.Time `sql:"index" json:"DeletedAt,omitempty" yaml:"-"`
|
||||
New bool `gorm:"-" json:"-" yaml:"-"`
|
||||
}
|
||||
|
||||
// FileInfos represents meta data about a file
|
||||
type FileInfos struct {
|
||||
FileWidth int
|
||||
FileHeight int
|
||||
FileOrientation int
|
||||
FileAspectRatio float32
|
||||
FileMainColor string
|
||||
FileColors string
|
||||
FileLuminance string
|
||||
FileDiff uint32
|
||||
FileChroma uint8
|
||||
}
|
||||
|
||||
// Account represents a remote service account for uploading, downloading or syncing media files.
|
||||
type Account struct {
|
||||
ID uint `gorm:"primary_key"`
|
||||
AccName string `gorm:"type:VARCHAR(255);"`
|
||||
AccOwner string `gorm:"type:VARCHAR(255);"`
|
||||
AccURL string `gorm:"type:VARBINARY(512);"`
|
||||
AccType string `gorm:"type:VARBINARY(255);"`
|
||||
AccKey string `gorm:"type:VARBINARY(255);"`
|
||||
AccUser string `gorm:"type:VARBINARY(255);"`
|
||||
AccPass string `gorm:"type:VARBINARY(255);"`
|
||||
AccError string `gorm:"type:VARBINARY(512);"`
|
||||
AccErrors int
|
||||
AccShare bool
|
||||
AccSync bool
|
||||
RetryLimit int
|
||||
SharePath string `gorm:"type:VARBINARY(500);"`
|
||||
ShareSize string `gorm:"type:VARBINARY(16);"`
|
||||
ShareExpires int
|
||||
SyncPath string `gorm:"type:VARBINARY(500);"`
|
||||
SyncStatus string `gorm:"type:VARBINARY(16);"`
|
||||
SyncInterval int
|
||||
SyncDate sql.NullTime `deepcopier:"skip"`
|
||||
SyncUpload bool
|
||||
SyncDownload bool
|
||||
SyncFilenames bool
|
||||
SyncRaw bool
|
||||
CreatedAt time.Time `deepcopier:"skip"`
|
||||
UpdatedAt time.Time `deepcopier:"skip"`
|
||||
DeletedAt *time.Time `deepcopier:"skip" sql:"index"`
|
||||
}
|
||||
|
|
|
@ -9,7 +9,7 @@ import (
|
|||
"net/url"
|
||||
"strings"
|
||||
|
||||
v1 "git.zio.sh/astra/photoprism-client-go/api/v1"
|
||||
v1 "github.com/astravexton/photoprism-client-go/api/v1"
|
||||
)
|
||||
|
||||
const (
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
ignore_*
|
|
@ -4,6 +4,8 @@ Here are good examples and working code snippets to start from.
|
|||
|
||||
A lot of these files are used for development and are subject to change.
|
||||
|
||||
For more complete sample code, see the integration testing suite in `/test`.
|
||||
|
||||
### Running the examples
|
||||
|
||||
Run the examples. Make sure to pass both the file you wish to run, as well as `common.go` to include the convenience functions.
|
||||
|
|
|
@ -1,9 +1,9 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
photoprism "git.zio.sh/astra/photoprism-client-go"
|
||||
"git.zio.sh/astra/photoprism-client-go/api/v1"
|
||||
"github.com/astravexton/logger"
|
||||
photoprism "github.com/astravexton/photoprism-client-go"
|
||||
"github.com/astravexton/photoprism-client-go/api/v1"
|
||||
)
|
||||
|
||||
func main() {
|
||||
|
|
|
@ -4,7 +4,7 @@ import (
|
|||
"fmt"
|
||||
"os"
|
||||
|
||||
photoprism "git.zio.sh/astra/photoprism-client-go"
|
||||
photoprism "github.com/astravexton/photoprism-client-go"
|
||||
|
||||
"github.com/astravexton/logger"
|
||||
)
|
||||
|
|
|
@ -3,8 +3,8 @@ package main
|
|||
import (
|
||||
"fmt"
|
||||
|
||||
photoprism "git.zio.sh/astra/photoprism-client-go"
|
||||
"github.com/astravexton/logger"
|
||||
photoprism "github.com/astravexton/photoprism-client-go"
|
||||
)
|
||||
|
||||
func main() {
|
||||
|
|
|
@ -1,8 +1,8 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
photoprism "git.zio.sh/astra/photoprism-client-go"
|
||||
"github.com/astravexton/logger"
|
||||
photoprism "github.com/astravexton/photoprism-client-go"
|
||||
)
|
||||
|
||||
func main() {
|
||||
|
|
|
@ -5,8 +5,8 @@ import (
|
|||
"io/ioutil"
|
||||
"path"
|
||||
|
||||
photoprism "git.zio.sh/astra/photoprism-client-go"
|
||||
"github.com/astravexton/logger"
|
||||
photoprism "github.com/astravexton/photoprism-client-go"
|
||||
)
|
||||
|
||||
func main() {
|
||||
|
|
9
go.mod
|
@ -1,5 +1,8 @@
|
|||
module git.zio.sh/astra/photoprism-client-go
|
||||
module github.com/astravexton/photoprism-client-go
|
||||
|
||||
go 1.18
|
||||
go 1.15
|
||||
|
||||
require github.com/astravexton/logger v0.2.2-0.20211216142523-c1e08a465709 // indirect
|
||||
require (
|
||||
github.com/astravexton/logger v0.2.1
|
||||
github.com/kris-nova/logger v0.2.1
|
||||
)
|
||||
|
|
12
go.sum
|
@ -1,8 +1,4 @@
|
|||
github.com/astravexton/logger v0.2.2-0.20211216142523-c1e08a465709 h1:S+dBCax7wdg3ikWx5t6cClPKQIerrmD404qAzWhO2V4=
|
||||
github.com/astravexton/logger v0.2.2-0.20211216142523-c1e08a465709/go.mod h1:6BlG5E0U3CImXLmDtGpEwdoc8PHiiDv0kbvJBCbfiMY=
|
||||
github.com/fatih/color v1.10.0/go.mod h1:ELkj/draVOlAH/xkhN6mQ50Qd0MPOk5AAr3maGEBuJM=
|
||||
github.com/kris-nova/lolgopher v0.0.0-20210112022122-73f0047e8b65/go.mod h1:V0HF/ZBlN86HqewcDC/cVxMmYDiRukWjSrgKLUAn9Js=
|
||||
github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
|
||||
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
|
||||
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
github.com/astravexton/logger v0.2.1 h1:nYxhV6oCHIJzUzaPhawbAc51EEvJowIViVsB5EbdeJQ=
|
||||
github.com/astravexton/logger v0.2.1/go.mod h1:6uxVhLlWlvZn48W8F52dsMXt4yztlrInpRhdWiJ9RXc=
|
||||
github.com/kris-nova/logger v0.2.1 h1:hbZusgXXXTSd0rNAMBBe/8lhjxXkqWs0+nzjwewCI+E=
|
||||
github.com/kris-nova/logger v0.2.1/go.mod h1:++9BgZujZd4v0ZTZCb5iPsaomXdZWyxotIAh1IiDm44=
|
||||
|
|
|
@ -0,0 +1,6 @@
|
|||
# Internal
|
||||
|
||||
This directory is used to store backend code while developing.
|
||||
|
||||
This directory should NEVER contain SDK code.
|
||||
|
|
@ -0,0 +1,17 @@
|
|||
# Sample App
|
||||
|
||||
This is bad code.
|
||||
|
||||
We shell exec the start/stop/create/destroy docker commands (poorly)
|
||||
and this is how the unit testing suite attempts to start/stop/create/destroy
|
||||
the local persistent store.
|
||||
|
||||
### Running The Sample App
|
||||
|
||||
```bash
|
||||
./pcreate # Will create the sample app running locally
|
||||
./pdestroy # Will destroy the sample app, but the data will persist regardles of running this command
|
||||
./pstop # Will stop the photoprism app from running/serving
|
||||
./plogs # Will tail the photoprism logs
|
||||
./pstart # Will start an already created, and then stopped Photoprism application
|
||||
```
|
|
@ -0,0 +1,91 @@
|
|||
package sampleapp
|
||||
|
||||
import (
|
||||
"path/filepath"
|
||||
"runtime"
|
||||
|
||||
"github.com/astravexton/logger"
|
||||
|
||||
"github.com/astravexton/photoprism-client-go"
|
||||
)
|
||||
|
||||
type SampleApplication struct {
|
||||
}
|
||||
|
||||
func New() *SampleApplication {
|
||||
app := &SampleApplication{}
|
||||
return app
|
||||
}
|
||||
|
||||
// These are the bash scripts that can be used
|
||||
// to start/stop the Photoprism test application
|
||||
var (
|
||||
CreateCommand = `pcreate`
|
||||
DestroyCommand = `pdestroy`
|
||||
LogsCommand = `plogs`
|
||||
StartCommand = `pstart`
|
||||
StopCommand = `pstop"`
|
||||
)
|
||||
|
||||
func (a *SampleApplication) Start() error {
|
||||
logger.Info("Starting Application...")
|
||||
script, err := NewScriptFromPath(filepath.Join(PrintWorkingDirectory(), StartCommand))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return script.Interpret()
|
||||
}
|
||||
|
||||
func (a *SampleApplication) Stop() error {
|
||||
logger.Info("Stopping Application...")
|
||||
script, err := NewScriptFromPath(filepath.Join(PrintWorkingDirectory(), StopCommand))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return script.Interpret()
|
||||
}
|
||||
|
||||
func (a *SampleApplication) Create() error {
|
||||
logger.Info("Create Application...")
|
||||
script, err := NewScriptFromPath(filepath.Join(PrintWorkingDirectory(), CreateCommand))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return script.Interpret()
|
||||
}
|
||||
|
||||
func (a *SampleApplication) Destroy() error {
|
||||
logger.Info("Destroying Application...")
|
||||
script, err := NewScriptFromPath(filepath.Join(PrintWorkingDirectory(), DestroyCommand))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return script.Interpret()
|
||||
}
|
||||
|
||||
func (a *SampleApplication) Logs() error {
|
||||
logger.Info("Logging Application...")
|
||||
script, err := NewScriptFromPath(filepath.Join(PrintWorkingDirectory(), LogsCommand))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return script.Interpret()
|
||||
}
|
||||
|
||||
func (a *SampleApplication) GetAuth() photoprism.ClientAuthenticator {
|
||||
return nil
|
||||
}
|
||||
|
||||
func PrintWorkingDirectory() string {
|
||||
_, filename, _, ok := runtime.Caller(1)
|
||||
if !ok {
|
||||
logger.Info("Unable to PWD")
|
||||
return ""
|
||||
}
|
||||
dir, err := filepath.Abs(filepath.Dir(filename))
|
||||
if err != nil {
|
||||
logger.Info("Unable to PWD: %v", err)
|
||||
return ""
|
||||
}
|
||||
return dir
|
||||
}
|
|
@ -0,0 +1,169 @@
|
|||
package sampleapp
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"os/exec"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
|
||||
"github.com/astravexton/logger"
|
||||
)
|
||||
|
||||
// Script is a set of commands delimited by newlines
|
||||
// Comments # and // are ignored.
|
||||
type Script struct {
|
||||
commands []string
|
||||
}
|
||||
|
||||
// NewScriptFromPath is used to build an executable script from a path of disk.
|
||||
func NewScriptFromPath(path string) (*Script, error) {
|
||||
path, err := filepath.Abs(path)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("unable to calculate fully qualified path for path: %s: %v", path, err)
|
||||
}
|
||||
|
||||
bytes, err := ioutil.ReadFile(path)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("unable to read from path %s: %v", path, err)
|
||||
}
|
||||
content := string(bytes)
|
||||
return NewScriptFromString(content), nil
|
||||
}
|
||||
|
||||
const (
|
||||
// IgnoreSpacesBash is the amount of (spaces - 1) that we see in common \\n delimited commands
|
||||
IgnoreSpacesBash string = " "
|
||||
|
||||
// IgnoreTabs is the tab character
|
||||
IgnoreTabs string = "\t"
|
||||
)
|
||||
|
||||
// NewScriptFromString is used to build an executable script from the content in string form.
|
||||
func NewScriptFromString(str string) *Script {
|
||||
script := &Script{}
|
||||
removeRuleF := func(str string, rs []string) string {
|
||||
for _, r := range rs {
|
||||
str = strings.Replace(str, r, "", -1)
|
||||
}
|
||||
return str
|
||||
}
|
||||
str = strings.Replace(str, "\\\n", "", -1)
|
||||
str = removeRuleF(str, []string{IgnoreSpacesBash, IgnoreTabs})
|
||||
spl := strings.Split(str, "\n")
|
||||
//logger.Info("Script lines: %d", len(spl))
|
||||
for _, line := range spl {
|
||||
script.commands = append(script.commands, line)
|
||||
}
|
||||
return script
|
||||
}
|
||||
|
||||
// Interpret is used to procedurally execute a script. The script will execute each line independently
|
||||
// and can error at any point in the executation path.
|
||||
func (s *Script) Interpret() error {
|
||||
//logger.Info("Running script...")
|
||||
for i, cmdStr := range s.commands {
|
||||
// Exec will hang for output
|
||||
// Ignore newlines
|
||||
// Ignore comments starting with #
|
||||
// Ignore comments starting with //
|
||||
if cmdStr == "\n" || cmdStr == "" || strings.HasPrefix(cmdStr, "#") || strings.HasPrefix(cmdStr, "//") {
|
||||
continue
|
||||
}
|
||||
logger.Info("Executing: [%s]", cmdStr)
|
||||
result, err := Exec(cmdStr)
|
||||
if err != nil {
|
||||
return fmt.Errorf("error executing running command [%s] on line [%d]\n%v\n", cmdStr, i+1, err)
|
||||
} else if result.exitCode != 0 {
|
||||
return fmt.Errorf("non zero exit code running command [%s] on line [%d]\n%s\n%s\n", cmdStr, i+1, result.Stdout(), result.Stderr())
|
||||
}
|
||||
// Here is where we log STDOUT from a "script"
|
||||
// Right now it is set to DEBUG which can be enabled by
|
||||
// setting logger.Level = 4
|
||||
logger.Debug(result.Stdout())
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
type ExecResult struct {
|
||||
stderr string
|
||||
stdout string
|
||||
exitCode int
|
||||
execErr exec.ExitError
|
||||
}
|
||||
|
||||
func (e *ExecResult) Stdout() string {
|
||||
return e.stdout
|
||||
}
|
||||
|
||||
func (e *ExecResult) Stderr() string {
|
||||
return e.stderr
|
||||
}
|
||||
|
||||
func (e *ExecResult) ExitCode() int {
|
||||
if e == nil {
|
||||
return 0
|
||||
}
|
||||
return e.exitCode
|
||||
}
|
||||
|
||||
func (e *ExecResult) ExecError() exec.ExitError {
|
||||
return e.execErr
|
||||
}
|
||||
|
||||
// Exec will take an arbitrary executable string
|
||||
// and hang until the command exits
|
||||
func Exec(str string) (*ExecResult, error) {
|
||||
//logger.Info("Exec [%s]", str)
|
||||
var cmdstr string
|
||||
var args []string
|
||||
var l int
|
||||
spl := strings.Split(str, " ")
|
||||
l = len(spl)
|
||||
if l == 1 {
|
||||
// <cmd>
|
||||
cmdstr = spl[0]
|
||||
} else if l > 1 {
|
||||
// <cmd> <arg>...
|
||||
cmdstr = spl[0]
|
||||
for i := 1; i < l; i++ {
|
||||
args = append(args, spl[i])
|
||||
}
|
||||
} else if l < 1 {
|
||||
return nil, fmt.Errorf("invalid Exec() string %s", str)
|
||||
}
|
||||
fqpcmd, err := exec.LookPath(cmdstr)
|
||||
if err != nil {
|
||||
logger.Debug("unable to find fully qualified path for executable %s: %v", cmdstr, err)
|
||||
}
|
||||
|
||||
//logger.Info("Command: %s", fqpcmd)
|
||||
//logger.Info("Args: %v", args)
|
||||
|
||||
stdoutBuffer := bytes.Buffer{}
|
||||
stderrBuffer := bytes.Buffer{}
|
||||
e := []string{fqpcmd}
|
||||
for _, arg := range args {
|
||||
e = append(e, arg)
|
||||
}
|
||||
cmd := exec.Command(e[0], e[1:]...)
|
||||
cmd.Stdout = &stdoutBuffer
|
||||
cmd.Stderr = &stderrBuffer
|
||||
result := &ExecResult{}
|
||||
err = cmd.Run()
|
||||
if err != nil {
|
||||
if eerr, ok := err.(*exec.ExitError); ok {
|
||||
result.stderr = stderrBuffer.String()
|
||||
result.stdout = stdoutBuffer.String()
|
||||
result.exitCode = eerr.ExitCode()
|
||||
result.execErr = *eerr
|
||||
return result, nil
|
||||
}
|
||||
return nil, fmt.Errorf("major error running command [%s]: %v", str, err)
|
||||
}
|
||||
result.stderr = stderrBuffer.String()
|
||||
result.stdout = stdoutBuffer.String()
|
||||
result.exitCode = 0
|
||||
return result, nil
|
||||
}
|
|
@ -0,0 +1,25 @@
|
|||
|
||||
#!/bin/bash
|
||||
####################################
|
||||
#####
|
||||
###
|
||||
##
|
||||
#
|
||||
#
|
||||
# Startup Script for the Application
|
||||
####################################
|
||||
echo "Creating [SampleApp]"
|
||||
|
||||
# TODO Nova
|
||||
# TODO Per edude03ontwitch we want to actually mount /photoprism/whatever
|
||||
docker run -d \
|
||||
--name photoprism \
|
||||
-p 8080:2342 \
|
||||
-e PHOTOPRISM_UPLOAD_NSFW="true" \
|
||||
-e PHOTOPRISM_ADMIN_PASSWORD="missy" \
|
||||
-v ${GOPATH}/src/github.com/astravexton/photoprism-client-go/sample-app/photoprism/import:/photoprism/import \
|
||||
-v ${GOPATH}/src/github.com/astravexton/photoprism-client-go/sample-app/photoprism/originals:/photoprism/originals \
|
||||
-v ${GOPATH}/src/github.com/astravexton/photoprism-client-go/sample-app/photoprism/storage:/photoprism/storage \
|
||||
photoprism/photoprism:latest
|
||||
|
||||
|
|
@ -0,0 +1,13 @@
|
|||
#!/bin/bash
|
||||
####################################
|
||||
#####
|
||||
###
|
||||
##
|
||||
#
|
||||
#
|
||||
# Startup Script for the Application
|
||||
####################################
|
||||
echo "Destroying [SampleApp]"
|
||||
docker stop photoprism
|
||||
docker rm photoprism
|
||||
|
After Width: | Height: | Size: 592 KiB |
After Width: | Height: | Size: 114 KiB |
After Width: | Height: | Size: 380 KiB |
After Width: | Height: | Size: 134 KiB |
After Width: | Height: | Size: 380 KiB |
After Width: | Height: | Size: 592 KiB |
After Width: | Height: | Size: 114 KiB |
After Width: | Height: | Size: 134 KiB |
|
@ -0,0 +1,17 @@
|
|||
UID: aqnzih81icziiyae
|
||||
Slug: february-2021
|
||||
Type: album
|
||||
Title: February 2021
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-04T03:17:32Z
|
||||
UpdatedAt: 2021-02-04T03:17:32Z
|
||||
Photos:
|
||||
- UID: pqnzigq156lndozm
|
||||
CreatedAt: 2021-02-04T03:17:40.83892969Z
|
||||
- UID: pqnzigq1jb1bibrz
|
||||
CreatedAt: 2021-02-04T03:17:40.846632301Z
|
||||
- UID: pqnzigq351j2fqgn
|
||||
CreatedAt: 2021-02-04T03:17:40.851856107Z
|
||||
- UID: pqnzigq3sidxb0j0
|
||||
CreatedAt: 2021-02-04T03:17:40.857083572Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe01hgxpo1hjih
|
||||
Slug: novaalbum
|
||||
Type: album
|
||||
Title: NovaAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:03:18Z
|
||||
UpdatedAt: 2021-02-11T23:03:18Z
|
||||
DeletedAt: 2021-02-11T23:14:45.771988424Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe0941aw0wz0rj
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:07:53Z
|
||||
UpdatedAt: 2021-02-11T23:07:53Z
|
||||
DeletedAt: 2021-02-11T23:14:36.400063828Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe0j0nog2wtd98
|
||||
Slug: february-2021
|
||||
Type: album
|
||||
Title: February 2021
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:13:49Z
|
||||
UpdatedAt: 2021-02-11T23:13:49Z
|
||||
DeletedAt: 2021-02-11T23:14:30.637709439Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe0o1v8rjprqy6
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:16:49Z
|
||||
UpdatedAt: 2021-02-11T23:16:49.091245891Z
|
||||
DeletedAt: 2021-02-11T23:18:32.10184988Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe0xu1149t43i3
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:22:42Z
|
||||
UpdatedAt: 2021-02-11T23:22:42.31692892Z
|
||||
DeletedAt: 2021-02-12T00:14:52.285222547Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe0zl3h1wak08f
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:23:46Z
|
||||
UpdatedAt: 2021-02-11T23:23:45.864716162Z
|
||||
DeletedAt: 2021-02-11T23:24:31.729202695Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe1231tb94a48k
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:25:16Z
|
||||
UpdatedAt: 2021-02-11T23:25:15.61908592Z
|
||||
DeletedAt: 2021-02-12T00:14:52.285222547Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe1981fbhzaou2
|
||||
Slug: novaalbum
|
||||
Type: album
|
||||
Title: NovaAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:29:33Z
|
||||
UpdatedAt: 2021-02-11T23:29:33Z
|
||||
DeletedAt: 2021-02-12T00:14:52.285222547Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe1bw1ywyawhlb
|
||||
Slug: novaalbum
|
||||
Type: album
|
||||
Title: NovaAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:31:08Z
|
||||
UpdatedAt: 2021-02-11T23:31:08Z
|
||||
DeletedAt: 2021-02-11T23:31:08.058407849Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe1cs1rrfaer9d
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:31:41Z
|
||||
UpdatedAt: 2021-02-11T23:31:40.791556222Z
|
||||
DeletedAt: 2021-02-11T23:31:40.798855984Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe1mn3669ed8vc
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:37:36Z
|
||||
UpdatedAt: 2021-02-11T23:37:35.931076271Z
|
||||
DeletedAt: 2021-02-11T23:37:35.938663557Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe1mnib00h33mh
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:37:36Z
|
||||
UpdatedAt: 2021-02-11T23:37:36Z
|
||||
DeletedAt: 2021-02-11T23:37:35.984813813Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe1su2yz12rn7r
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:41:19Z
|
||||
UpdatedAt: 2021-02-11T23:41:19Z
|
||||
DeletedAt: 2021-02-11T23:41:18.568153261Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe1su312pday8a
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:41:18Z
|
||||
UpdatedAt: 2021-02-11T23:41:18.505274668Z
|
||||
DeletedAt: 2021-02-11T23:41:18.517183797Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe1su3gj3q7mrk
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:41:19Z
|
||||
UpdatedAt: 2021-02-11T23:41:19Z
|
||||
DeletedAt: 2021-02-11T23:41:18.592298473Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2nw1mc3m0c95
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:59:57Z
|
||||
UpdatedAt: 2021-02-11T23:59:57Z
|
||||
DeletedAt: 2021-02-11T23:59:56.834221161Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2nw2mtjrlyuq
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:59:57Z
|
||||
UpdatedAt: 2021-02-11T23:59:57Z
|
||||
DeletedAt: 2021-02-11T23:59:56.78263971Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2nw310xqz9f1
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:59:57Z
|
||||
UpdatedAt: 2021-02-11T23:59:57Z
|
||||
DeletedAt: 2021-02-11T23:59:56.805849627Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe2nws4nefmwtt
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-11T23:59:57Z
|
||||
UpdatedAt: 2021-02-11T23:59:56.727315788Z
|
||||
DeletedAt: 2021-02-11T23:59:56.734742155Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2u021a4zjufp
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:03:37Z
|
||||
UpdatedAt: 2021-02-12T00:03:37Z
|
||||
DeletedAt: 2021-02-12T00:03:36.899233684Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2u0299hfp9uf
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:03:37Z
|
||||
UpdatedAt: 2021-02-12T00:03:37Z
|
||||
DeletedAt: 2021-02-12T00:03:36.927067016Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe2u03t1k09mbq
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:03:37Z
|
||||
UpdatedAt: 2021-02-12T00:03:36.846932177Z
|
||||
DeletedAt: 2021-02-12T00:03:36.854999187Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2u0zakwm0kh4
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:03:37Z
|
||||
UpdatedAt: 2021-02-12T00:03:37Z
|
||||
DeletedAt: 2021-02-12T00:03:36.955701902Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe2y1x2v9ulekw
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:06:02Z
|
||||
UpdatedAt: 2021-02-12T00:06:01.997666242Z
|
||||
DeletedAt: 2021-02-12T00:06:02.008962725Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2y233f9xc3me
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:06:02Z
|
||||
UpdatedAt: 2021-02-12T00:06:02Z
|
||||
DeletedAt: 2021-02-12T00:06:02.061930815Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2y23iah7q8dd
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:06:02Z
|
||||
UpdatedAt: 2021-02-12T00:06:02Z
|
||||
DeletedAt: 2021-02-12T00:06:02.107878556Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe2y23n8shfo26
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:06:02Z
|
||||
UpdatedAt: 2021-02-12T00:06:02Z
|
||||
DeletedAt: 2021-02-12T00:06:02.083118581Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe310mqejp7sjx
|
||||
Slug: novaalbum
|
||||
Type: album
|
||||
Title: NovaAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:07:48Z
|
||||
UpdatedAt: 2021-02-12T00:07:48Z
|
||||
DeletedAt: 2021-02-12T00:07:48.087397413Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe3di2oe0gj7eb
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:15:18Z
|
||||
UpdatedAt: 2021-02-12T00:15:18.326635885Z
|
||||
DeletedAt: 2021-02-12T00:15:18.337623732Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3di3f013g098
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:15:18Z
|
||||
UpdatedAt: 2021-02-12T00:15:18Z
|
||||
DeletedAt: 2021-02-12T00:15:18.387748516Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3di3rfrp4osq
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:15:18Z
|
||||
UpdatedAt: 2021-02-12T00:15:18Z
|
||||
DeletedAt: 2021-02-12T00:15:18.417353099Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3di9wl91qjy1
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:15:18Z
|
||||
UpdatedAt: 2021-02-12T00:15:18Z
|
||||
DeletedAt: 2021-02-12T00:15:18.441462643Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3go3alpu9g42
|
||||
Slug: february-2021-2
|
||||
Type: album
|
||||
Title: February 2021 (2)
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:17:12Z
|
||||
UpdatedAt: 2021-02-12T00:17:12Z
|
||||
DeletedAt: 2021-02-12T00:25:55.610343844Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3sj140onsu3u
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:24:20Z
|
||||
UpdatedAt: 2021-02-12T00:24:20Z
|
||||
DeletedAt: 2021-02-12T00:24:19.584898324Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe3sj2oim76rfl
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:24:19Z
|
||||
UpdatedAt: 2021-02-12T00:24:19.452079212Z
|
||||
DeletedAt: 2021-02-12T00:24:19.46454459Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3sj3d9vpwfs6
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:24:19Z
|
||||
UpdatedAt: 2021-02-12T00:24:19Z
|
||||
DeletedAt: 2021-02-12T00:24:19.513541437Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3sj5v7uxy0xi
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:24:20Z
|
||||
UpdatedAt: 2021-02-12T00:24:20Z
|
||||
DeletedAt: 2021-02-12T00:24:19.539899181Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe3uk1ztdf4ly1
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:25:32Z
|
||||
UpdatedAt: 2021-02-12T00:25:32.382246374Z
|
||||
DeletedAt: 2021-02-12T00:25:32.39451528Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3uk2ssh3wlmu
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:25:32Z
|
||||
UpdatedAt: 2021-02-12T00:25:32Z
|
||||
DeletedAt: 2021-02-12T00:25:32.501491328Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3uk3quu1oau5
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:25:32Z
|
||||
UpdatedAt: 2021-02-12T00:25:32Z
|
||||
DeletedAt: 2021-02-12T00:25:32.438255604Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe3ukef6k7pr1n
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:25:32Z
|
||||
UpdatedAt: 2021-02-12T00:25:32Z
|
||||
DeletedAt: 2021-02-12T00:25:32.457679012Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe43017x1x557a
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:30:36Z
|
||||
UpdatedAt: 2021-02-12T00:30:36.452565204Z
|
||||
DeletedAt: 2021-02-12T00:30:36.466024116Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe43038gpo8o62
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:30:37Z
|
||||
UpdatedAt: 2021-02-12T00:30:37Z
|
||||
DeletedAt: 2021-02-12T00:30:36.541728272Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4303e2k7fowx
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:30:36Z
|
||||
UpdatedAt: 2021-02-12T00:30:36Z
|
||||
DeletedAt: 2021-02-12T00:30:36.515616836Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe430hjepljox7
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:30:37Z
|
||||
UpdatedAt: 2021-02-12T00:30:37Z
|
||||
DeletedAt: 2021-02-12T00:30:36.590399562Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe46u1y7nllowc
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:32:55Z
|
||||
UpdatedAt: 2021-02-12T00:32:55Z
|
||||
DeletedAt: 2021-02-12T00:32:54.670452125Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe46u2eytps9ks
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:32:55Z
|
||||
UpdatedAt: 2021-02-12T00:32:54.621067369Z
|
||||
DeletedAt: 2021-02-12T00:32:54.629069306Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe46ugvoopsx2c
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:32:55Z
|
||||
UpdatedAt: 2021-02-12T00:32:55Z
|
||||
DeletedAt: 2021-02-12T00:32:54.738939626Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe46uh3251f3o5
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:32:55Z
|
||||
UpdatedAt: 2021-02-12T00:32:55Z
|
||||
DeletedAt: 2021-02-12T00:32:54.694577526Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4k41ws8kl303
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:40:52Z
|
||||
UpdatedAt: 2021-02-12T00:40:52Z
|
||||
DeletedAt: 2021-02-12T00:40:52.236143624Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe4k420c8937zq
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:40:52Z
|
||||
UpdatedAt: 2021-02-12T00:40:52.149620745Z
|
||||
DeletedAt: 2021-02-12T00:40:52.157633931Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4k423kseyeok
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:40:52Z
|
||||
UpdatedAt: 2021-02-12T00:40:52Z
|
||||
DeletedAt: 2021-02-12T00:40:52.205835621Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4k42wb7vdpy4
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:40:52Z
|
||||
UpdatedAt: 2021-02-12T00:40:52Z
|
||||
DeletedAt: 2021-02-12T00:40:52.346059642Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4m917yd0x5d8
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:42:10Z
|
||||
UpdatedAt: 2021-02-12T00:42:10Z
|
||||
DeletedAt: 2021-02-12T00:42:09.739048804Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe4m91dd475mc2
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:42:10Z
|
||||
UpdatedAt: 2021-02-12T00:42:09.68534495Z
|
||||
DeletedAt: 2021-02-12T00:42:09.697602792Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4m9204aigugh
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:42:10Z
|
||||
UpdatedAt: 2021-02-12T00:42:10Z
|
||||
DeletedAt: 2021-02-12T01:19:49.19521169Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe4m93kl5118yf
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T00:42:10Z
|
||||
UpdatedAt: 2021-02-12T00:42:10Z
|
||||
DeletedAt: 2021-02-12T00:42:09.764632667Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5s018umhsg2z
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:07:12Z
|
||||
UpdatedAt: 2021-02-12T01:07:12Z
|
||||
DeletedAt: 2021-02-12T01:07:12.278795475Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe5s02eypzy49n
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:07:12Z
|
||||
UpdatedAt: 2021-02-12T01:07:12.18606883Z
|
||||
DeletedAt: 2021-02-12T01:07:12.20368425Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5s02htbyx02a
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:07:12Z
|
||||
UpdatedAt: 2021-02-12T01:07:12Z
|
||||
DeletedAt: 2021-02-12T01:07:12.25672608Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5s0kcesnf038
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:07:12Z
|
||||
UpdatedAt: 2021-02-12T01:07:12Z
|
||||
DeletedAt: 2021-02-12T01:07:12.311501625Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5um2nxaam1jf
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:08:46Z
|
||||
UpdatedAt: 2021-02-12T01:08:46Z
|
||||
DeletedAt: 2021-02-12T01:08:46.441203314Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe5um3k2bjzdz3
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:08:46Z
|
||||
UpdatedAt: 2021-02-12T01:08:46.349964437Z
|
||||
DeletedAt: 2021-02-12T01:08:46.364581421Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5umo46cx9up1
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:08:46Z
|
||||
UpdatedAt: 2021-02-12T01:08:46Z
|
||||
DeletedAt: 2021-02-12T01:08:46.414981534Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5zd1dhchzyrt
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:11:37Z
|
||||
UpdatedAt: 2021-02-12T01:11:37Z
|
||||
DeletedAt: 2021-02-12T01:11:37.410053582Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe5zd1n8nyjezg
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:11:37Z
|
||||
UpdatedAt: 2021-02-12T01:11:37.317189151Z
|
||||
DeletedAt: 2021-02-12T01:11:37.337078186Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe5zdgv8bbtzbn
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:11:37Z
|
||||
UpdatedAt: 2021-02-12T01:11:37Z
|
||||
DeletedAt: 2021-02-12T01:11:37.38408389Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe63g15cp92acg
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:14:05Z
|
||||
UpdatedAt: 2021-02-12T01:14:04.617839652Z
|
||||
DeletedAt: 2021-02-12T01:14:04.630483844Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe63g1hvf4c37t
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:14:05Z
|
||||
UpdatedAt: 2021-02-12T01:14:05Z
|
||||
DeletedAt: 2021-02-12T01:14:04.683342935Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe63g1wp37dwns
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:14:05Z
|
||||
UpdatedAt: 2021-02-12T01:14:05Z
|
||||
DeletedAt: 2021-02-12T01:14:04.720807005Z
|
|
@ -0,0 +1,10 @@
|
|||
UID: aqoe66r266y7x2ct
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Description: An updated album description
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:16:03Z
|
||||
UpdatedAt: 2021-02-12T01:16:03.089952155Z
|
||||
DeletedAt: 2021-02-12T01:16:03.103617951Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe66r304v81itt
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:16:03Z
|
||||
UpdatedAt: 2021-02-12T01:16:03Z
|
||||
DeletedAt: 2021-02-12T01:16:03.147574525Z
|
|
@ -0,0 +1,9 @@
|
|||
UID: aqoe66r3bjgymp2o
|
||||
Slug: testalbum
|
||||
Type: album
|
||||
Title: TestAlbum
|
||||
Order: oldest
|
||||
Country: zz
|
||||
CreatedAt: 2021-02-12T01:16:03Z
|
||||
UpdatedAt: 2021-02-12T01:16:03Z
|
||||
DeletedAt: 2021-02-12T01:16:03.168746489Z
|