2020-08-05 15:08:59 +00:00
package main
import (
"errors"
"fmt"
"io/ioutil"
"log"
2020-10-22 16:42:48 +00:00
"mime/multipart"
2020-08-05 15:08:59 +00:00
"os"
"path/filepath"
2020-10-02 15:31:59 +00:00
"strings"
2020-08-05 15:08:59 +00:00
2020-09-29 15:04:22 +00:00
"github.com/bouncepaw/mycorrhiza/history"
2020-10-30 13:25:48 +00:00
"github.com/bouncepaw/mycorrhiza/markup"
2020-11-18 13:07:53 +00:00
"github.com/bouncepaw/mycorrhiza/user"
2020-10-03 16:56:56 +00:00
"github.com/bouncepaw/mycorrhiza/util"
2020-08-05 15:08:59 +00:00
)
func init ( ) {
2020-10-30 13:25:48 +00:00
markup . HyphaExists = func ( hyphaName string ) bool {
2020-08-05 15:08:59 +00:00
_ , hyphaExists := HyphaStorage [ hyphaName ]
return hyphaExists
}
2020-10-30 13:25:48 +00:00
markup . HyphaAccess = func ( hyphaName string ) ( rawText , binaryBlock string , err error ) {
2020-08-05 15:08:59 +00:00
if hyphaData , ok := HyphaStorage [ hyphaName ] ; ok {
rawText , err = FetchTextPart ( hyphaData )
if hyphaData . binaryPath != "" {
binaryBlock = binaryHtmlBlock ( hyphaName , hyphaData )
}
} else {
err = errors . New ( "Hypha " + hyphaName + " does not exist" )
}
return
}
2020-11-26 18:41:26 +00:00
markup . HyphaIterate = IterateHyphaNamesWith
2020-08-05 15:08:59 +00:00
}
2020-11-18 13:07:53 +00:00
// GetHyphaData finds a hypha addressed by `hyphaName` and returns its `hyphaData`. `hyphaData` is set to a zero value if this hypha does not exist. `isOld` is false if this hypha does not exist.
func GetHyphaData ( hyphaName string ) ( hyphaData * HyphaData , isOld bool ) {
hyphaData , isOld = HyphaStorage [ hyphaName ]
if hyphaData == nil {
hyphaData = & HyphaData { }
}
return
}
2020-08-05 15:08:59 +00:00
// HyphaData represents a hypha's meta information: binary and text parts rooted paths and content types.
type HyphaData struct {
textPath string
binaryPath string
}
2020-10-22 16:42:48 +00:00
// uploadHelp is a helper function for UploadText and UploadBinary
2020-11-18 13:07:53 +00:00
func uploadHelp ( hop * history . HistoryOp , hyphaName , ext string , data [ ] byte , u * user . User ) * history . HistoryOp {
2020-10-22 16:42:48 +00:00
var (
2020-11-18 13:07:53 +00:00
hyphaData , isOld = GetHyphaData ( hyphaName )
fullPath = filepath . Join ( WikiDir , hyphaName + ext )
originalFullPath = & hyphaData . textPath
2020-10-22 16:42:48 +00:00
)
2020-11-18 13:07:53 +00:00
if hop . Type == history . TypeEditBinary {
originalFullPath = & hyphaData . binaryPath
}
2020-10-22 16:42:48 +00:00
if err := os . MkdirAll ( filepath . Dir ( fullPath ) , 0777 ) ; err != nil {
return hop . WithError ( err )
}
if err := ioutil . WriteFile ( fullPath , data , 0644 ) ; err != nil {
return hop . WithError ( err )
}
2020-11-18 13:07:53 +00:00
2020-10-25 13:50:14 +00:00
if isOld && * originalFullPath != fullPath && * originalFullPath != "" {
if err := history . Rename ( * originalFullPath , fullPath ) ; err != nil {
2020-10-22 16:42:48 +00:00
return hop . WithError ( err )
}
2020-10-25 13:50:14 +00:00
log . Println ( "Move" , * originalFullPath , "to" , fullPath )
2020-10-22 16:42:48 +00:00
}
2020-11-18 13:07:53 +00:00
// New hyphae must be added to the hypha storage
2020-10-22 16:42:48 +00:00
if ! isOld {
2020-11-18 13:07:53 +00:00
HyphaStorage [ hyphaName ] = hyphaData
2020-10-22 16:42:48 +00:00
}
2020-10-25 13:50:14 +00:00
* originalFullPath = fullPath
2020-10-22 16:42:48 +00:00
return hop . WithFiles ( fullPath ) .
2020-11-18 13:07:53 +00:00
WithUser ( u ) .
2020-10-22 16:42:48 +00:00
Apply ( )
}
2020-11-18 13:07:53 +00:00
// UploadText loads a new text part from `textData` for hypha `hyphaName`.
func UploadText ( hyphaName , textData string , u * user . User ) * history . HistoryOp {
return uploadHelp (
history .
Operation ( history . TypeEditText ) .
WithMsg ( fmt . Sprintf ( "Edit ‘ %s’ " , hyphaName ) ) ,
hyphaName , ".myco" , [ ] byte ( textData ) , u )
2020-10-22 16:42:48 +00:00
}
// UploadBinary loads a new binary part from `file` for hypha `hyphaName` with `hd`. The contents have the specified `mime` type. It must be marked if the hypha `isOld`.
2020-11-18 13:07:53 +00:00
func UploadBinary ( hyphaName , mime string , file multipart . File , u * user . User ) * history . HistoryOp {
2020-10-22 16:42:48 +00:00
var (
hop = history . Operation ( history . TypeEditBinary ) . WithMsg ( fmt . Sprintf ( "Upload binary part for ‘ %s’ with type ‘ %s’ " , hyphaName , mime ) )
data , err = ioutil . ReadAll ( file )
)
if err != nil {
2020-11-01 19:09:41 +00:00
return hop . WithError ( err ) . Apply ( )
2020-10-22 16:42:48 +00:00
}
2020-11-18 13:07:53 +00:00
return uploadHelp ( hop , hyphaName , MimeToExtension ( mime ) , data , u )
2020-10-22 16:42:48 +00:00
}
2020-09-29 15:04:22 +00:00
// DeleteHypha deletes hypha and makes a history record about that.
2020-11-18 13:07:53 +00:00
func ( hd * HyphaData ) DeleteHypha ( hyphaName string , u * user . User ) * history . HistoryOp {
2020-10-03 14:19:49 +00:00
hop := history . Operation ( history . TypeDeleteHypha ) .
2020-09-29 15:04:22 +00:00
WithFilesRemoved ( hd . textPath , hd . binaryPath ) .
WithMsg ( fmt . Sprintf ( "Delete ‘ %s’ " , hyphaName ) ) .
2020-11-18 13:07:53 +00:00
WithUser ( u ) .
2020-09-29 15:04:22 +00:00
Apply ( )
2020-10-03 14:19:49 +00:00
if len ( hop . Errs ) == 0 {
delete ( HyphaStorage , hyphaName )
}
return hop
2020-09-29 15:04:22 +00:00
}
2020-10-03 16:56:56 +00:00
func findHyphaeToRename ( hyphaName string , recursive bool ) [ ] string {
hyphae := [ ] string { hyphaName }
if recursive {
hyphae = append ( hyphae , util . FindSubhyphae ( hyphaName , IterateHyphaNamesWith ) ... )
}
return hyphae
}
2020-10-21 18:37:39 +00:00
func renamingPairs ( hyphaNames [ ] string , replaceName func ( string ) string ) ( map [ string ] string , error ) {
2020-10-03 16:56:56 +00:00
renameMap := make ( map [ string ] string )
for _ , hn := range hyphaNames {
if hd , ok := HyphaStorage [ hn ] ; ok {
2020-10-21 18:37:39 +00:00
if _ , nameUsed := HyphaStorage [ replaceName ( hn ) ] ; nameUsed {
return nil , errors . New ( "Hypha " + replaceName ( hn ) + " already exists" )
}
2020-10-03 16:56:56 +00:00
if hd . textPath != "" {
renameMap [ hd . textPath ] = replaceName ( hd . textPath )
}
if hd . binaryPath != "" {
renameMap [ hd . binaryPath ] = replaceName ( hd . binaryPath )
}
}
}
2020-10-21 18:37:39 +00:00
return renameMap , nil
2020-10-03 16:56:56 +00:00
}
// word Data is plural here
func relocateHyphaData ( hyphaNames [ ] string , replaceName func ( string ) string ) {
for _ , hyphaName := range hyphaNames {
if hd , ok := HyphaStorage [ hyphaName ] ; ok {
hd . textPath = replaceName ( hd . textPath )
hd . binaryPath = replaceName ( hd . binaryPath )
HyphaStorage [ replaceName ( hyphaName ) ] = hd
delete ( HyphaStorage , hyphaName )
}
}
}
// RenameHypha renames hypha from old name `hyphaName` to `newName` and makes a history record about that. If `recursive` is `true`, its subhyphae will be renamed the same way.
2020-11-18 13:07:53 +00:00
func RenameHypha ( hyphaName , newName string , recursive bool , u * user . User ) * history . HistoryOp {
2020-10-02 15:31:59 +00:00
var (
2020-10-03 16:56:56 +00:00
replaceName = func ( str string ) string {
return strings . Replace ( str , hyphaName , newName , 1 )
}
2020-10-21 18:37:39 +00:00
hyphaNames = findHyphaeToRename ( hyphaName , recursive )
renameMap , err = renamingPairs ( hyphaNames , replaceName )
renameMsg = "Rename ‘ %s’ to ‘ %s’ "
hop = history . Operation ( history . TypeRenameHypha )
2020-10-02 15:31:59 +00:00
)
2020-10-21 18:37:39 +00:00
if err != nil {
hop . Errs = append ( hop . Errs , err )
return hop
}
2020-10-03 16:56:56 +00:00
if recursive {
renameMsg += " recursively"
}
hop . WithFilesRenamed ( renameMap ) .
WithMsg ( fmt . Sprintf ( renameMsg , hyphaName , newName ) ) .
2020-11-18 13:07:53 +00:00
WithUser ( u ) .
2020-10-03 16:56:56 +00:00
Apply ( )
2020-10-02 15:31:59 +00:00
if len ( hop . Errs ) == 0 {
2020-10-03 16:56:56 +00:00
relocateHyphaData ( hyphaNames , replaceName )
2020-10-02 15:31:59 +00:00
}
return hop
}
2020-08-05 15:08:59 +00:00
// binaryHtmlBlock creates an html block for binary part of the hypha.
2020-10-22 17:12:12 +00:00
func binaryHtmlBlock ( hyphaName string , hd * HyphaData ) string {
switch filepath . Ext ( hd . binaryPath ) {
case ".jpg" , ".gif" , ".png" , ".webp" , ".svg" , ".ico" :
2020-08-05 15:08:59 +00:00
return fmt . Sprintf ( `
< div class = "binary-container binary-container_with-img" >
2020-11-26 18:41:26 +00:00
< a href = "/binary/%[1]s" > < img src = "/binary/%[1]s" / > < / a >
2020-08-05 15:08:59 +00:00
< / div > ` , hyphaName )
2020-10-22 17:12:12 +00:00
case ".ogg" , ".webm" , ".mp4" :
2020-08-05 15:08:59 +00:00
return fmt . Sprintf ( `
< div class = "binary-container binary-container_with-video" >
< video >
< source src = "/binary/%[1]s" / >
< p > Your browser does not support video . See video ' s < a href = "/binary/%[1]s" > direct url < / a > < / p >
< / video >
` , hyphaName )
2020-10-22 17:12:12 +00:00
case ".mp3" :
2020-08-05 15:08:59 +00:00
return fmt . Sprintf ( `
< div class = "binary-container binary-container_with-audio" >
< audio >
< source src = "/binary/%[1]s" / >
< p > Your browser does not support audio . See audio ' s < a href = "/binary/%[1]s" > direct url < / a > < / p >
< / audio >
` , hyphaName )
default :
return fmt . Sprintf ( `
< div class = "binary-container binary-container_with-nothing" >
< p > This hypha ' s media cannot be rendered . Access it < a href = "/binary/%s" > directly < / a > < / p >
< / div >
` , hyphaName )
}
}
// Index finds all hypha files in the full `path` and saves them to HyphaStorage. This function is recursive.
func Index ( path string ) {
nodes , err := ioutil . ReadDir ( path )
if err != nil {
log . Fatal ( err )
}
for _ , node := range nodes {
2020-10-22 16:42:48 +00:00
// If this hypha looks like it can be a hypha path, go deeper. Do not touch the .git and static folders for they have an admnistrative importance!
if node . IsDir ( ) && isCanonicalName ( node . Name ( ) ) && node . Name ( ) != ".git" && node . Name ( ) != "static" {
2020-08-05 15:08:59 +00:00
Index ( filepath . Join ( path , node . Name ( ) ) )
2020-11-03 16:39:06 +00:00
continue
2020-08-05 15:08:59 +00:00
}
2020-10-22 16:42:48 +00:00
var (
hyphaPartPath = filepath . Join ( path , node . Name ( ) )
hyphaName , isText , skip = DataFromFilename ( hyphaPartPath )
hyphaData * HyphaData
)
2020-08-05 15:08:59 +00:00
if ! skip {
2020-10-22 16:42:48 +00:00
// Reuse the entry for existing hyphae, create a new one for those that do not exist yet.
if hd , ok := HyphaStorage [ hyphaName ] ; ok {
hyphaData = hd
} else {
2020-08-05 15:08:59 +00:00
hyphaData = & HyphaData { }
HyphaStorage [ hyphaName ] = hyphaData
}
if isText {
2020-10-22 16:42:48 +00:00
hyphaData . textPath = hyphaPartPath
2020-08-05 15:08:59 +00:00
} else {
2020-10-22 16:42:48 +00:00
// Notify the user about binary part collisions. It's a design decision to just use any of them, it's the user's fault that they have screwed up the folder structure, but the engine should at least let them know, right?
if hyphaData . binaryPath != "" {
log . Println ( "There is a file collision for binary part of a hypha:" , hyphaData . binaryPath , "and" , hyphaPartPath , "-- going on with the latter" )
}
hyphaData . binaryPath = hyphaPartPath
2020-08-05 15:08:59 +00:00
}
}
2020-10-22 16:42:48 +00:00
2020-08-05 15:08:59 +00:00
}
}
// FetchTextPart tries to read text file in the `d`. If there is no file, empty string is returned.
func FetchTextPart ( d * HyphaData ) ( string , error ) {
if d . textPath == "" {
return "" , nil
}
_ , err := os . Stat ( d . textPath )
if os . IsNotExist ( err ) {
return "" , nil
} else if err != nil {
return "" , err
}
text , err := ioutil . ReadFile ( d . textPath )
if err != nil {
return "" , err
}
return string ( text ) , nil
}