Start finder refactoring

This commit is contained in:
cyber-dream 2023-05-10 15:39:49 +03:00
parent f81e15f1b8
commit 9c0cc8d709
10 changed files with 453 additions and 134 deletions

View File

@ -1,4 +1,5 @@
class Finder{ class Finder{
static appId = "Finder"
appId = "Finder" appId = "Finder"
fileView = undefined fileView = undefined
path = "/" path = "/"
@ -10,27 +11,29 @@ class Finder{
// this.appElem = appElem // this.appElem = appElem
// WebDesktopEnvironment.RegisterApp(this) // WebDesktopEnvironment.RegisterApp(this)
} }
/** /**
* @param {string[]} args * @param {string[]} args
*/ */
async NewWindow(args){ async NewWindow(args){
let newFinder = new FinderWindow()
await newFinder.Init(args)
return
this.path = args[0] this.path = args[0]
if (args[1] == "desktop"){ if (args[1] == "desktop"){
this.NewDesktop(args) this.NewDesktop(args)
return return
} }
const params = new URLSearchParams({isMobile: WebDesktopEnvironment.isMobile}) // const params = new URLSearchParams({isMobile: WebDesktopEnvironment.isMobile})
fetch(`/app/${this.appId}/render?` + params) fetch(`/app/${Finder.appId}/render?` + params)
.then((response) => response.text()) .then((response) => response.text())
.then((html) => { .then((html) => {
let newWindow = WebDesktopEnvironment.CreateNewWindow(this.appId, 500, 350 ) let newWindow = WebDesktopEnvironment.CreateNewWindow(Finder.appId, 500, 350 )
newWindow.innerHTML = html newWindow.innerHTML = html
this.fileView = new FileView(newWindow.querySelector(".FileTileView"), this.fileView = new FileView(newWindow.querySelector(".FileTileView"),
(event) =>{ this.Click(event) }, (event) =>{ this.Click(event) },
(event) =>{ this.RightClick(event) }, (event) =>{ this.RightClick(event) },
(event) =>{ this.FileUploading(event) }, (event, draggedElem) =>{ this.DropEvent(event, draggedElem) },
) )
this.OpenDir(this.path) this.OpenDir(this.path)
@ -61,14 +64,23 @@ class Finder{
} }
/** /**
* @param {DataTransferItemList} filesList * @param {DragEvent} event
* @param {HTMLElement} draggedElem
*/ */
async DropEvent(event){
FileUploading(filesList){ // console.log(event.dataTransfer.getData("dropType"))
if (event.dataTransfer.getData("dropType") == "move"){
let params = new URLSearchParams({
sourcePath: event.dataTransfer.getData("filePath"),
targetPath: this.path + "/" + event.dataTransfer.getData("fileName"),
})
let response = await fetch('/fs/move/?' + params)
console.log(response.status)
} else {
let formData = new FormData() let formData = new FormData()
console.log(this.path) console.log(this.path)
for (let i = 0; i < filesList.length; i++) { for (let i = 0; i < event.length; i++) {
const element = filesList[i]; const element = event[i];
formData.append("file", element.getAsFile()) formData.append("file", element.getAsFile())
// console.log(formData) // console.log(formData)
} }
@ -85,12 +97,13 @@ class Finder{
WebDesktopEnvironment.Alert(error); WebDesktopEnvironment.Alert(error);
}) })
} }
}
/** /**
* @param {string[]} args * @param {string[]} args
*/ */
NewDesktop(args){ NewDesktop(args){
fetch(`${window.location.origin}/application/${this.appId}/renderDesktop?` + new URLSearchParams({ fetch(`${window.location.origin}/application/${Finder.appId}/renderDesktop?` + new URLSearchParams({
isMobile: WebDesktopEnvironment.isMobile, isMobile: WebDesktopEnvironment.isMobile,
path: args[0] path: args[0]
})) }))
@ -133,13 +146,122 @@ class Finder{
// OpenNewDir(){ // OpenNewDir(){
// WebDesktopEnvironment.Open("finder", [this.path]) // WebDesktopEnvironment.Open("finder", [this.path])
// } // }
/**
* @param {string} path
* @returns {boolean}
*/
static async RenderProperites(path){
const params = new URLSearchParams({
path: path
})
const response = await fetch(`/application/${Finder.appId}/renderProps?` + params)
if (response.status != 200){
WebDesktopEnvironment.Alert("Error in properties render") //TODO
return false
}
const html = response.text()
let newWindow = WebDesktopEnvironment.CreateNewWindow(Finder.appId, 350, 500 )
newWindow.innerHTML = html
newWindow.querySelector("#closeWindowButton").addEventListener('click', function (params) {
WebDesktopEnvironment.CloseWindow(newWindow)
})
}
/** /**
* @param {MouseEvent} event * @param {string} path
* @returns {boolean}
*/ */
RightClick(event){ static async DeleteFile(path){
this.CreateContextMenu(event.target, [event.clientY, event.clientX]) const params = new URLSearchParams({
path: path
})
const response = await fetch(`/fs/delete?` + params)
if (response.status != 200){
WebDesktopEnvironment.Alert("DELETE ERROR") //TODO
return false
} }
return true
}
/**
* @param {string} path
* @returns {boolean}
*/
static async CreateDirectory(path){
if (path == undefined){
WebDesktopEnvironment.Alert("Path is undefined")
return false
}
const params = new URLSearchParams({
path: `${path}/New Directory`
})
const response = await fetch(`/fs/createDir?` + params)
if (response.status != 200){
WebDesktopEnvironment.Alert("DIRCTORY CREATION ERROR") //TODO
return false
}
return true
}
}
class FinderWindow{
CurPath = ""
fileView = undefined
windowElem = undefined
async Init(args){
const params = new URLSearchParams({isMobile: WebDesktopEnvironment.isMobile})
const response = await fetch(`/app/${Finder.appId}/render?` + params)
if (response.status != 200){
WebDesktopEnvironment.Alert("Finder ERROR TODO") //TODO
return
}
const html = await response.text()
let newWindow = WebDesktopEnvironment.CreateNewWindow(Finder.appId, 500, 350 )
newWindow.innerHTML = html
this.fileView = new FileView(newWindow.querySelector(".FileTileView"),
(event) =>{ this.Click(event) },
(event) =>{ this.RightClick(event) },
(event, draggedElem) =>{ this.DropEvent(event, draggedElem)
})
newWindow.querySelector("#closeWindowButton").addEventListener('click', function (params) {
WebDesktopEnvironment.CloseWindow(newWindow)
})
newWindow.querySelector("#RootButton").addEventListener('click', () =>{
this.RenderDir('/')
})
newWindow.querySelector("#HomeButton").addEventListener('click', () =>{
this.RenderDir('/home/user')
})
let scrollBar = new WdeScrollBar(newWindow.querySelector(".ScrollbarPlace"), newWindow.querySelector(".FileTileView"))
this.windowElem = newWindow
this.RenderDir(args[0])
}
RenderDir(path){
this.CurPath = path
this.fileView.OpenFolder(path)
}
ReRenderDir(){
this.RenderDir(this.CurPath)
}
/** /**
* @param {MouseEvent} event * @param {MouseEvent} event
@ -151,16 +273,16 @@ class Finder{
switch (fileType) { switch (fileType) {
case "directory": case "directory":
if (fileExtension == "app"){ if (fileExtension == "app"){
WebDesktopEnvironment.Open(`${this.path}/${fileName}`, []) WebDesktopEnvironment.Open(`${this.CurPath}/${fileName}`, [])
} else{ } else{
WebDesktopEnvironment.Open(`/Applications/Finder.app`, [`${this.path}/${fileName}`]) WebDesktopEnvironment.Open(`/Applications/Finder.app`, [`${this.CurPath}/${fileName}`])
} }
break break
case "blog": case "blog":
WebDesktopEnvironment.Open("blog-viewer", [`${this.path}/${fileName}`]) WebDesktopEnvironment.Open("blog-viewer", [`${this.CurPath}/${fileName}`])
break break
case "personal-properties": case "personal-properties":
WebDesktopEnvironment.Open("personal-properties", [`${this.path}/${fileName}`]) WebDesktopEnvironment.Open("personal-properties", [`${this.CurPath}/${fileName}`])
break break
// case "app": // case "app":
// //TODO get real id // //TODO get real id
@ -168,7 +290,7 @@ class Finder{
// break; // break;
case "jpeg": case "jpeg":
case "png": case "png":
WebDesktopEnvironment.Open("img-viewer", [this.path + "/" + fileName]) WebDesktopEnvironment.Open("img-viewer", [this.CurPath + "/" + fileName])
break; break;
default: default:
// console.log("Unsupported file type") // console.log("Unsupported file type")
@ -177,7 +299,18 @@ class Finder{
} }
} }
CreateContextMenu(target, pos){ /**
* @param {MouseEvent} event
*/
RightClick(event){
this.CreateContextMenu(event.target, [event.clientY, event.clientX])
}
/**
* @param {HTMLElement} target
* @param {string[]} pos
*/
async CreateContextMenu(target, pos){
let context = "" let context = ""
const fileName = target.getAttribute("name") const fileName = target.getAttribute("name")
const fileType = target.getAttribute("fileType") const fileType = target.getAttribute("fileType")
@ -187,10 +320,13 @@ class Finder{
} else { } else {
context = fileType context = fileType
} }
const params = new URLSearchParams({context: context, path: `${this.path}/${fileName}`}) const params = new URLSearchParams({context: context, path: `${this.CurPath}/${fileName}`})
fetch(`/app/${this.appId}/contextMenu?` + params) const response = await fetch(`/app/${Finder.appId}/contextMenu?` + params)
.then((response) => response.text()) if (response.status != 200){
.then((html) => { WebDesktopEnvironment.Alert("ERROR in Context menu TODO"); //TODO
return
}
const html = await response.text()
let overlay = document.createElement("div") //TODO Move to WDE.CreateOverlay() let overlay = document.createElement("div") //TODO Move to WDE.CreateOverlay()
overlay.setAttribute('id', 'finder-context-menu-overlay') overlay.setAttribute('id', 'finder-context-menu-overlay')
overlay.style.position = 'absolute' overlay.style.position = 'absolute'
@ -204,70 +340,55 @@ class Finder{
menu.style.left = pos[1] + "px"; menu.style.left = pos[1] + "px";
menu.innerHTML = html menu.innerHTML = html
menu.children[0].firstElementChild.remove()
menu.children[0].lastElementChild.remove() //FIXME Can't ommit rendering of horLine in end of menu on backend menu.children[0].lastElementChild.remove() //FIXME Can't ommit rendering of horLine in end of menu on backend
overlay.appendChild(menu) overlay.appendChild(menu)
document.body.appendChild(overlay) document.body.appendChild(overlay)
overlay.addEventListener('click', (event) => { overlay.addEventListener('click', async (event) => {
if (event.target.classList.contains("Row")){ //TODO add uuid id to rows to more accurate checks?? if (event.target.classList.contains("Row")){ //TODO add uuid id to rows to more accurate checks??
let res = false
switch (event.target.children[0].getAttribute("action")) { switch (event.target.children[0].getAttribute("action")) {
case "createDir": case "createDir":
fetch(`/fs/createDir?` + new URLSearchParams({ res = await Finder.CreateDirectory(`${this.CurPath}`)
path: `${this.path}/New Directory` console.log(res)
})) if (res){
.then((response) => { this.ReRenderDir()
console.log(response.status)
if (response.status == 200){
this.OpenDir(this.path)
} }
})
.catch((error) => {
WebDesktopEnvironment.Alert(error);
})
break break
case "deleteFile": case "deleteFile":
console.log(fileName) res = await Finder.DeleteFile(`${this.CurPath}/${fileName}`)
// break console.log(res)
fetch(`/fs/delete?` + new URLSearchParams({ if (res){
path: `${this.path}/${fileName}` this.ReRenderDir()
}))
.then((response) => {
console.log(response.status)
if (response.status == 200){
this.OpenDir(this.path)
} }
})
.catch((error) => {
WebDesktopEnvironment.Alert(error);
})
break break
case "getInfo": case "getInfo":
fetch(`/application/${this.appId}/renderProps?` + new URLSearchParams({ res = await Finder.RenderProperites(`${this.CurPath}/${fileName}`)
path: `${this.path}/${fileName}` // fetch(`/application/${Finder.appId}/renderProps?` + new URLSearchParams({
})) // path: `${this.CurPath}/${fileName}`
.then((response) => { // }))
console.log(response) // .then((response) => {
if (response.status == 200){ // console.log(response)
console.log("Success") // if (response.status == 200){
return response.text(); // console.log("Success")
} // return response.text();
}) // }
.then((html) =>{ // })
let newWindow = WebDesktopEnvironment.CreateNewWindow(this.appId, 350, 500 ) // .then((html) =>{
newWindow.innerHTML = html // let newWindow = WebDesktopEnvironment.CreateNewWindow(Finder.appId, 350, 500 )
newWindow.querySelector("#closeWindowButton").addEventListener('click', function (params) { // newWindow.innerHTML = html
WebDesktopEnvironment.CloseWindow(newWindow) // newWindow.querySelector("#closeWindowButton").addEventListener('click', function (params) {
}) // WebDesktopEnvironment.CloseWindow(newWindow)
}) // })
.catch((error) => { // })
WebDesktopEnvironment.Alert(error); // .catch((error) => {
}) // WebDesktopEnvironment.Alert(error);
// })
break break
case "openAsDir": case "openAsDir":
WebDesktopEnvironment.Open(`/Applications/${this.appId}.app`,[`${this.path}/${fileName}`]) WebDesktopEnvironment.Open(`/Applications/${Finder.appId}.app`,[`${this.CurPath}/${fileName}`])
break break
default: default:
break; break;
@ -279,9 +400,5 @@ class Finder{
event.preventDefault(); event.preventDefault();
overlay.remove() overlay.remove()
}) })
})
.catch((error) => {
WebDesktopEnvironment.Alert(error);
})
} }
} }

View File

@ -7,6 +7,11 @@
/* FIXME Bug, on desktop mode top ~10 pixel are not active, like margin:10px */ /* FIXME Bug, on desktop mode top ~10 pixel are not active, like margin:10px */
} }
.FileTileView.DragDropBorder{
box-shadow: inset 0px 0px 0px 4px #9999CC;
/* background-color: blue; */
}
.FileTileView .FlexContainer{ .FileTileView .FlexContainer{
width: 100%; width: 100%;
height: auto; height: auto;

View File

@ -1,4 +1,5 @@
class FileView{ class FileView{
path = ""
parentElem = undefined parentElem = undefined
selected = [] selected = []
/** /**
@ -35,19 +36,58 @@ class FileView{
}) })
if (fileUploadCallback !== undefined) { if (fileUploadCallback !== undefined) {
let counter = 0
let draggedElem = undefined
fileViewElem.addEventListener('dragstart', (event) => {
// console.log(event.target)
// console.log(this.path)
// draggedElem = event.target
event.dataTransfer.setData("fileName", event.target.getAttribute("name"))
event.dataTransfer.setData("filePath", this.path + "/" + event.target.getAttribute("name"))
event.dataTransfer.setData("dropType", "move")
// event.dataTransfer.setData("fileName", )
// console.log(draggedElem)
})
fileViewElem.addEventListener('dragenter', function(event) { fileViewElem.addEventListener('dragenter', function(event) {
// console.log("start") event.preventDefault();
counter++
fileViewElem.classList.add("DragDropBorder")
// draggedElem = event.relatedTarget
})
fileViewElem.addEventListener('dragend', function(event) {
// console.log(fileViewElem)
event.preventDefault();
counter--
if (counter === 0){
fileViewElem.classList.remove("DragDropBorder")
}
// draggedElem = undefined
})
fileViewElem.addEventListener('dragleave', function(event) {
// console.log(fileViewElem)
event.preventDefault();
counter--
if (counter === 0){
fileViewElem.classList.remove("DragDropBorder")
}
// draggedElem = undefined
}) })
fileViewElem.addEventListener('dragover', function(event) { fileViewElem.addEventListener('dragover', function(event) {
event.preventDefault(); event.preventDefault();
// console.log("over")
}) })
fileViewElem.addEventListener("drop", (event) => { fileViewElem.addEventListener("drop", (event) => {
event.preventDefault(); event.preventDefault();
// console.log(event.dataTransfer.items[0]) // console.log(event.dataTransfer.getData("dropType"))
fileUploadCallback(event.dataTransfer.items) fileUploadCallback(event)
fileViewElem.classList.remove("DragDropBorder")
// draggedElem = undefined
}) })
} }
} }
@ -73,6 +113,7 @@ class FileView{
* @param {string} path * @param {string} path
*/ */
OpenFolder(path){ OpenFolder(path){
this.path = path
fetch(`${window.location.origin}/system/wde/widgets/file-tile-view?path=${path}`) //TODO Move to wde func. Or Not? fetch(`${window.location.origin}/system/wde/widgets/file-tile-view?path=${path}`) //TODO Move to wde func. Or Not?
.then((response) => response.text()) .then((response) => response.text())
.then((html) => { .then((html) => {

View File

@ -1,7 +1,7 @@
{{ define "wde-widgets/file-tile-view.tmpl" }} {{ define "wde-widgets/file-tile-view.tmpl" }}
<div class="FlexContainer NoClick"> <div class="FlexContainer NoClick">
{{ range $fileTile := .Files }} {{ range $fileTile := .Files }}
<div fileType="{{ $fileTile.Type }}" class="Tile Click" name="{{$fileTile.Name}}"> <div fileType="{{ $fileTile.Type }}" class="Tile Click" name="{{$fileTile.Name}}" draggable="true">
<!-- <div class="Icon NoClick"></div> --> <!-- <div class="Icon NoClick"></div> -->
<img class="Icon NoClick" src="{{ $fileTile.Icon }}"> <img class="Icon NoClick" src="{{ $fileTile.Icon }}">
<div class="Lable NoClick">{{ $fileTile.Name }}</div> <div class="Lable NoClick">{{ $fileTile.Name }}</div>

View File

@ -51,6 +51,7 @@ func (fs *WebFileSystem) CreateDirectory(dirPath string) (primitive.ObjectID, pr
} }
func (fs *WebFileSystem) AppendChildToDirectory(childId primitive.ObjectID, parentId primitive.ObjectID) error { func (fs *WebFileSystem) AppendChildToDirectory(childId primitive.ObjectID, parentId primitive.ObjectID) error {
// println(parentId)
parentDirData := DirectoryData{} parentDirData := DirectoryData{}
parentDir, err := fs.readFSDocs(parentId, &parentDirData) parentDir, err := fs.readFSDocs(parentId, &parentDirData)
if err != nil { if err != nil {
@ -67,6 +68,28 @@ func (fs *WebFileSystem) AppendChildToDirectory(childId primitive.ObjectID, pare
return nil return nil
} }
func (fs *WebFileSystem) RemoveChildToDirectory(childId primitive.ObjectID, parentId primitive.ObjectID) error {
parentDirData := DirectoryData{}
parentDir, err := fs.readFSDocs(parentId, &parentDirData)
if err != nil {
return err
}
newChildrenList := []primitive.ObjectID{}
for _, child := range parentDirData.Children {
if child != childId {
newChildrenList = append(newChildrenList, child)
}
}
parentDirData.Children = newChildrenList
err = fs.UpdateFileData(parentDir, parentDirData)
if err != nil {
return err
}
return nil
}
func (fs *WebFileSystem) ListDir(dirPath string) ([]*FileHeader, error) { func (fs *WebFileSystem) ListDir(dirPath string) ([]*FileHeader, error) {
dirId, err := fs.FindFile(dirPath) dirId, err := fs.FindFile(dirPath)
if err != nil { if err != nil {

View File

@ -44,6 +44,7 @@ func (fs *WebFileSystem) removeFromMongo(fileId primitive.ObjectID) error {
return errors.New("todo not implemented yet") return errors.New("todo not implemented yet")
} }
//Deprecated
func (fs *WebFileSystem) Validate() error { func (fs *WebFileSystem) Validate() error {
// filter := primitive.D{ // filter := primitive.D{
// { // {

View File

@ -1,5 +1,7 @@
package webfilesystem package webfilesystem
import "go.mongodb.org/mongo-driver/bson/primitive"
func (fs *WebFileSystem) Read(filePath string, fileData interface{}) (*FileHeader, error) { func (fs *WebFileSystem) Read(filePath string, fileData interface{}) (*FileHeader, error) {
fileId, err := fs.FindFile(filePath) fileId, err := fs.FindFile(filePath)
if err != nil { if err != nil {
@ -13,3 +15,12 @@ func (fs *WebFileSystem) Read(filePath string, fileData interface{}) (*FileHeade
return fileHeader, nil return fileHeader, nil
} }
func (fs *WebFileSystem) ReadByObjectID(fileID primitive.ObjectID, fileData interface{}) (*FileHeader, error) {
fileHeader, err := fs.readFSDocs(fileID, fileData)
if err != nil {
return nil, err
}
return fileHeader, nil
}

View File

@ -123,7 +123,27 @@ func (fs *WebFileSystem) Route(route *gin.RouterGroup) {
}) })
route.GET("validate", func(ctx *gin.Context) { route.GET("validate", func(ctx *gin.Context) {
err := fs.Validate() err := fs.validate()
if err != nil {
ctx.Status(http.StatusInternalServerError)
return
}
ctx.Status(http.StatusOK)
})
route.GET("move", func(ctx *gin.Context) {
sourcePath := ctx.Query("sourcePath")
if sourcePath == "" {
ctx.Status(http.StatusBadRequest) //TODO
return
}
targetPath := ctx.Query("targetPath")
if targetPath == "" {
ctx.Status(http.StatusBadRequest) //TODO
return
}
err := fs.Move(sourcePath, targetPath)
if err != nil { if err != nil {
ctx.Status(http.StatusInternalServerError) ctx.Status(http.StatusInternalServerError)
return return

58
webfilesystem/validate.go Normal file
View File

@ -0,0 +1,58 @@
package webfilesystem
func (fs *WebFileSystem) validate() error {
// rootHeader, rootData, err := fs.GetRootDir()
// if err != nil {
// return err
// }
// for _, child := range rootData.Children {
// fs.Chi
// }
// println()
// fs.Read
// filter := primitive.M{
// "type": "directory",
// }
// cur, err := fs.webfsFilesTable.Find(fs.ctx, filter)
// if err != nil {
// return err
// }
// directories := []FileHeader{}
// for cur.Next(fs.ctx) {
// fileHeader := FileHeader{}
// err := cur.Decode(&fileHeader)
// if err != nil {
// println(err.Error())
// continue
// }
// directories = append(directories, fileHeader)
// }
// allChilds := map[primitive.ObjectID{}]
// for _, dir := range directories {
// println(dir.Name)
// dirData := DirectoryData{}
// _, err := fs.ReadByObjectID(dir.MongoId, &dirData)
// if err != nil {
// println(err.Error())
// continue
// }
// //Check for broken links in children
// for _, dirChild := range dirData.Children {
// dirChildHeader, err := fs.ReadByObjectID(dirChild, nil)
// println(" -" + dirChildHeader.Name)
// if err != nil {
// println(err.Error())
// continue
// }
// }
// }
return nil
}
// func (fs *WebFileSystem) GetChildrenHeaders(directoryID primitive.ObjectID) ([]FileHeader, error) {
// fs.ReadByObjectID(directoryID, nil)
// }

View File

@ -143,6 +143,49 @@ func (fs *WebFileSystem) GetRootDir() (*FileHeader, *DirectoryData, error) {
return &rootDir, &rootDirData, nil return &rootDir, &rootDirData, nil
} }
func (fs *WebFileSystem) Move(sourcePath string, targetPath string) error {
//TODO Check about move dir to itself
//TODO if it a dir fix its parentId
//TODO Rename file if in path set new name for it
sourceFileHeader, err := fs.Read(sourcePath, nil)
if err != nil {
return err
}
sourceParentDirPath := fs.GetParentPath(sourcePath)
sourceParentDirData := DirectoryData{}
sourceParentDirHeader, err := fs.Read(sourceParentDirPath, &sourceParentDirData)
if err != nil {
return err
}
if sourceParentDirHeader.Type != "directory" {
return errors.New("source parent object is not a directory")
}
targetParentDirPath := fs.GetParentPath(targetPath)
targetParentDirData := DirectoryData{}
targetParentDirHeader, err := fs.Read(targetParentDirPath, &targetParentDirData)
if err != nil {
return err
}
if targetParentDirHeader.Type != "directory" {
return errors.New("target parent object is not a directory")
}
err = fs.RemoveChildToDirectory(sourceFileHeader.MongoId, sourceParentDirHeader.MongoId)
if err != nil {
return err
}
err = fs.AppendChildToDirectory(sourceFileHeader.MongoId, targetParentDirHeader.MongoId)
if err != nil {
return err
}
return nil
}
func (fs *WebFileSystem) Remove(filePath string) error { func (fs *WebFileSystem) Remove(filePath string) error {
parentPath := fs.GetParentPath(filePath) parentPath := fs.GetParentPath(filePath)
parentDirId, err := fs.FindFile(parentPath) parentDirId, err := fs.FindFile(parentPath)