- Added support for Faster Whisper transcription via Jigasi and Skynet. - Updated .env.example to include new environment variables for transcription settings. - Enhanced Jitsi Docker Compose configuration to include Skynet and Jigasi services. - Introduced new API endpoints for managing organizational folders in the drive service. - Updated Nextcloud initialization script to enable external file mounting. - Improved error handling and response structures in the drive API. - Added new properties for organization settings related to transcription and agenda management.
383 lines
13 KiB
Go
383 lines
13 KiB
Go
package drive
|
|
|
|
import (
|
|
"io"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
|
|
"github.com/ultisuite/ulti-backend/internal/api/apiresponse"
|
|
"github.com/ultisuite/ulti-backend/internal/api/apivalidate"
|
|
"github.com/ultisuite/ulti-backend/internal/api/middleware"
|
|
"github.com/ultisuite/ulti-backend/internal/api/query"
|
|
"github.com/ultisuite/ulti-backend/internal/driveroot"
|
|
"github.com/ultisuite/ulti-backend/internal/nextcloud"
|
|
)
|
|
|
|
func (h *Handler) registerOrgAndMountRoutes(r chi.Router, read, write func(http.Handler) http.Handler) {
|
|
r.With(read).Get("/org-folders", h.ListOrgFolders)
|
|
r.With(read).Get("/org-folders/{folderID}/files/*", h.ListOrgFolderFiles)
|
|
r.With(read).Get("/org-folders/{folderID}/files/info/*", h.GetOrgFolderFileInfo)
|
|
r.With(read).Get("/org-folders/{folderID}/download/*", h.DownloadOrgFolderFile)
|
|
r.With(read).Get("/org-folders/{folderID}/preview/*", h.PreviewOrgFolderFile)
|
|
r.With(write).Post("/org-folders/{folderID}/files/*", h.UploadOrgFolderFile)
|
|
r.With(write).Post("/org-folders/{folderID}/folders/*", h.CreateOrgFolderDir)
|
|
r.With(write).Delete("/org-folders/{folderID}/files/*", h.DeleteOrgFolderFile)
|
|
|
|
r.With(read).Get("/mounts", h.ListMounts)
|
|
r.With(write).Post("/mounts", h.CreateMount)
|
|
r.With(write).Delete("/mounts/{mountID}", h.DeleteMount)
|
|
r.With(read).Get("/mounts/{mountID}/oauth-url", h.GetMountOAuthURL)
|
|
r.With(write).Post("/mounts/{mountID}/oauth/complete", h.CompleteMountOAuth)
|
|
r.With(read).Get("/mounts/{mountID}/files/*", h.ListMountFiles)
|
|
r.With(read).Get("/mounts/{mountID}/files/info/*", h.GetMountFileInfo)
|
|
r.With(read).Get("/mounts/{mountID}/download/*", h.DownloadMountFile)
|
|
r.With(read).Get("/mounts/{mountID}/preview/*", h.PreviewMountFile)
|
|
r.With(write).Post("/mounts/{mountID}/files/*", h.UploadMountFile)
|
|
r.With(write).Post("/mounts/{mountID}/folders/*", h.CreateMountDir)
|
|
r.With(write).Delete("/mounts/{mountID}/files/*", h.DeleteMountFile)
|
|
}
|
|
|
|
func (h *Handler) ListOrgFolders(w http.ResponseWriter, r *http.Request) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
folders, err := h.svc.ListOrgFoldersForUser(r.Context(), ncUser)
|
|
if err != nil {
|
|
h.logger.Error("list org folders", "error", err)
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
apiresponse.WriteJSON(w, http.StatusOK, map[string]any{"folders": folders})
|
|
}
|
|
|
|
func (h *Handler) ListOrgFolderFiles(w http.ResponseWriter, r *http.Request) {
|
|
h.listRootFiles(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) ListMountFiles(w http.ResponseWriter, r *http.Request) {
|
|
h.listRootFiles(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) listRootFiles(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
params, err := query.ParseListRequest(r)
|
|
if err != nil {
|
|
apivalidate.WriteQueryError(w, r, err)
|
|
return
|
|
}
|
|
path := nextcloud.NormalizeClientPath(chi.URLParam(r, "*"))
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: path}
|
|
result, err := h.svc.ListFilesAtRoot(r.Context(), ncUser, ref, params)
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
h.svc.EnrichSources(r.Context(), claims.Sub, result.Files)
|
|
apiresponse.WriteJSON(w, http.StatusOK, result)
|
|
}
|
|
|
|
func (h *Handler) GetOrgFolderFileInfo(w http.ResponseWriter, r *http.Request) {
|
|
h.getRootFileInfo(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) GetMountFileInfo(w http.ResponseWriter, r *http.Request) {
|
|
h.getRootFileInfo(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) getRootFileInfo(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
path := nextcloud.NormalizeClientPath(chi.URLParam(r, "*"))
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: path}
|
|
file, err := h.svc.StatFileAtRoot(r.Context(), ncUser, ref)
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
h.svc.EnrichSources(r.Context(), claims.Sub, []nextcloud.FileInfo{file})
|
|
apiresponse.WriteJSON(w, http.StatusOK, file)
|
|
}
|
|
|
|
func (h *Handler) DownloadOrgFolderFile(w http.ResponseWriter, r *http.Request) {
|
|
h.downloadRootFile(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) DownloadMountFile(w http.ResponseWriter, r *http.Request) {
|
|
h.downloadRootFile(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) downloadRootFile(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
path := chi.URLParam(r, "*")
|
|
if verr := validatePath(path); verr != nil {
|
|
apivalidate.WriteValidationError(w, r, verr)
|
|
return
|
|
}
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: nextcloud.NormalizeClientPath(path)}
|
|
body, contentType, err := h.svc.DownloadAtRoot(r.Context(), ncUser, ref)
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
defer body.Close()
|
|
w.Header().Set("Content-Type", contentType)
|
|
io.Copy(w, body)
|
|
}
|
|
|
|
func (h *Handler) PreviewOrgFolderFile(w http.ResponseWriter, r *http.Request) {
|
|
h.previewRootFile(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) PreviewMountFile(w http.ResponseWriter, r *http.Request) {
|
|
h.previewRootFile(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) previewRootFile(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
path := chi.URLParam(r, "*")
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: nextcloud.NormalizeClientPath(path)}
|
|
file, err := h.svc.StatFileAtRoot(r.Context(), ncUser, ref)
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
width, _ := strconv.Atoi(r.URL.Query().Get("w"))
|
|
height, _ := strconv.Atoi(r.URL.Query().Get("h"))
|
|
_ = width
|
|
_ = height
|
|
var body io.ReadCloser
|
|
var contentType string
|
|
if kind == driveroot.KindPersonal {
|
|
body, contentType, err = h.svc.Preview(r.Context(), ncUser, file.Path, width, height)
|
|
} else {
|
|
body, contentType, err = h.svc.DownloadAtRoot(r.Context(), ncUser, ref)
|
|
}
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
defer body.Close()
|
|
w.Header().Set("Content-Type", contentType)
|
|
w.Header().Set("Cache-Control", "private, max-age=300")
|
|
io.Copy(w, body)
|
|
}
|
|
|
|
func (h *Handler) UploadOrgFolderFile(w http.ResponseWriter, r *http.Request) {
|
|
h.uploadRootFile(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) UploadMountFile(w http.ResponseWriter, r *http.Request) {
|
|
h.uploadRootFile(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) uploadRootFile(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
path := chi.URLParam(r, "*")
|
|
if verr := validatePath(path); verr != nil {
|
|
apivalidate.WriteValidationError(w, r, verr)
|
|
return
|
|
}
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: nextcloud.NormalizeClientPath(path)}
|
|
if err := h.svc.UploadAtRoot(r.Context(), ncUser, ref, r.Body, r.Header.Get("Content-Type"), r.ContentLength); err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
apiresponse.WriteJSON(w, http.StatusCreated, map[string]string{"status": "uploaded", "path": path})
|
|
}
|
|
|
|
func (h *Handler) CreateOrgFolderDir(w http.ResponseWriter, r *http.Request) {
|
|
h.createRootDir(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) CreateMountDir(w http.ResponseWriter, r *http.Request) {
|
|
h.createRootDir(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) createRootDir(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
path := chi.URLParam(r, "*")
|
|
if verr := validatePath(path); verr != nil {
|
|
apivalidate.WriteValidationError(w, r, verr)
|
|
return
|
|
}
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: nextcloud.NormalizeClientPath(path)}
|
|
if err := h.svc.CreateFolderAtRoot(r.Context(), ncUser, ref); err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusCreated)
|
|
}
|
|
|
|
func (h *Handler) DeleteOrgFolderFile(w http.ResponseWriter, r *http.Request) {
|
|
h.deleteRootFile(w, r, driveroot.KindOrg, chi.URLParam(r, "folderID"))
|
|
}
|
|
|
|
func (h *Handler) DeleteMountFile(w http.ResponseWriter, r *http.Request) {
|
|
h.deleteRootFile(w, r, driveroot.KindMount, chi.URLParam(r, "mountID"))
|
|
}
|
|
|
|
func (h *Handler) deleteRootFile(w http.ResponseWriter, r *http.Request, kind driveroot.Kind, rootID string) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
path := chi.URLParam(r, "*")
|
|
ref := driveroot.Ref{Kind: kind, RootID: rootID, Path: nextcloud.NormalizeClientPath(path)}
|
|
if err := h.svc.DeleteAtRoot(r.Context(), ncUser, ref); err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNoContent)
|
|
}
|
|
|
|
func (h *Handler) ListMounts(w http.ResponseWriter, r *http.Request) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
platformUserID, err := h.svc.platformUserID(r.Context(), claims.Sub)
|
|
if err != nil {
|
|
apivalidate.WriteInternal(w, r)
|
|
return
|
|
}
|
|
orgSlugs := parseOrgSlugs(r.URL.Query().Get("org_slugs"))
|
|
mounts, err := h.svc.ListMountsForUser(r.Context(), platformUserID, ncUser, orgSlugs)
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
apiresponse.WriteJSON(w, http.StatusOK, map[string]any{"mounts": mounts})
|
|
}
|
|
|
|
func (h *Handler) CreateMount(w http.ResponseWriter, r *http.Request) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
platformUserID, err := h.svc.platformUserID(r.Context(), claims.Sub)
|
|
if err != nil {
|
|
apivalidate.WriteInternal(w, r)
|
|
return
|
|
}
|
|
var req createMountRequest
|
|
if err := apivalidate.DecodeJSON(w, r, maxJSONRequestBody, &req); err != nil {
|
|
return
|
|
}
|
|
mount, err := h.svc.CreateMount(r.Context(), platformUserID, ncUser, CreateMountParams{
|
|
Scope: req.Scope,
|
|
OrgSlug: req.OrgSlug,
|
|
DisplayName: req.DisplayName,
|
|
BackendType: req.BackendType,
|
|
WebDAV: req.WebDAV,
|
|
OAuthBackend: req.OAuthBackend,
|
|
OAuthAuth: req.OAuthAuth,
|
|
})
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
apiresponse.WriteJSON(w, http.StatusCreated, mount)
|
|
}
|
|
|
|
func (h *Handler) DeleteMount(w http.ResponseWriter, r *http.Request) {
|
|
mountID := chi.URLParam(r, "mountID")
|
|
if err := h.svc.DeleteMount(r.Context(), mountID); err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
w.WriteHeader(http.StatusNoContent)
|
|
}
|
|
|
|
func (h *Handler) GetMountOAuthURL(w http.ResponseWriter, r *http.Request) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
platformUserID, err := h.svc.platformUserID(r.Context(), claims.Sub)
|
|
if err != nil {
|
|
apivalidate.WriteInternal(w, r)
|
|
return
|
|
}
|
|
url, err := h.svc.GetMountOAuthURL(r.Context(), chi.URLParam(r, "mountID"), platformUserID, ncUser, strings.TrimSpace(r.URL.Query().Get("redirect_uri")))
|
|
if err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
apiresponse.WriteJSON(w, http.StatusOK, map[string]string{"oauth_url": url})
|
|
}
|
|
|
|
type completeMountOAuthRequest struct {
|
|
Code string `json:"code"`
|
|
RedirectURI string `json:"redirect_uri"`
|
|
}
|
|
|
|
func (h *Handler) CompleteMountOAuth(w http.ResponseWriter, r *http.Request) {
|
|
claims := middleware.ClaimsFromContext(r.Context())
|
|
ncUser, ok := h.nextcloudUser(w, r, claims)
|
|
if !ok {
|
|
return
|
|
}
|
|
platformUserID, err := h.svc.platformUserID(r.Context(), claims.Sub)
|
|
if err != nil {
|
|
apivalidate.WriteInternal(w, r)
|
|
return
|
|
}
|
|
var req completeMountOAuthRequest
|
|
if err := apivalidate.DecodeJSON(w, r, maxJSONRequestBody, &req); err != nil {
|
|
return
|
|
}
|
|
if err := h.svc.CompleteMountOAuth(r.Context(), chi.URLParam(r, "mountID"), platformUserID, ncUser, strings.TrimSpace(req.RedirectURI), req.Code); err != nil {
|
|
writeDriveError(w, r, err)
|
|
return
|
|
}
|
|
apiresponse.WriteJSON(w, http.StatusOK, map[string]string{"status": "active"})
|
|
}
|
|
|
|
func parseOrgSlugs(raw string) []string {
|
|
raw = strings.TrimSpace(raw)
|
|
if raw == "" {
|
|
return nil
|
|
}
|
|
parts := strings.Split(raw, ",")
|
|
out := make([]string, 0, len(parts))
|
|
for _, p := range parts {
|
|
p = strings.TrimSpace(strings.ToLower(p))
|
|
if p != "" {
|
|
out = append(out, p)
|
|
}
|
|
}
|
|
return out
|
|
}
|