mirror of
https://github.com/wisplite/raster.git
synced 2026-05-01 06:32:44 -05:00
Image loading and auth
This commit is contained in:
@@ -36,4 +36,42 @@ func RegisterMediaRoutes(rg *gin.RouterGroup) {
|
|||||||
}
|
}
|
||||||
c.JSON(http.StatusOK, gin.H{"media": media})
|
c.JSON(http.StatusOK, gin.H{"media": media})
|
||||||
})
|
})
|
||||||
|
media.GET("/getAllMediaInAlbum", func(c *gin.Context) {
|
||||||
|
accessToken := c.GetHeader("Authorization")
|
||||||
|
albumID := c.Query("albumId")
|
||||||
|
media, err := services.GetAllMediaInAlbum(albumID, accessToken)
|
||||||
|
if err != nil {
|
||||||
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.JSON(http.StatusOK, gin.H{"media": media})
|
||||||
|
})
|
||||||
|
media.GET("/:albumId/:mediaId", func(c *gin.Context) {
|
||||||
|
albumID := c.Param("albumId")
|
||||||
|
mediaID := c.Param("mediaId")
|
||||||
|
if albumID == "root" {
|
||||||
|
albumID = ""
|
||||||
|
}
|
||||||
|
accessToken := c.GetHeader("Authorization")
|
||||||
|
userID, err := services.ValidateAccessToken(accessToken)
|
||||||
|
if err != nil {
|
||||||
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
accessLevel, err := services.CheckUserAlbumAccess(userID, albumID)
|
||||||
|
if err != nil {
|
||||||
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if accessLevel < 0 {
|
||||||
|
c.JSON(http.StatusForbidden, gin.H{"error": "user does not have permission to view media in this album"})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
mediaData, err := services.GetMedia(albumID, mediaID)
|
||||||
|
if err != nil {
|
||||||
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
c.File(mediaData.Path)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,3 +38,32 @@ func UploadMedia(file *multipart.FileHeader, albumID string, accessToken string)
|
|||||||
}
|
}
|
||||||
return media, nil
|
return media, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func GetAllMediaInAlbum(albumID string, accessToken string) ([]models.Media, error) {
|
||||||
|
userID, err := ValidateAccessToken(accessToken)
|
||||||
|
if err != nil {
|
||||||
|
return []models.Media{}, err
|
||||||
|
}
|
||||||
|
accessLevel, err := CheckUserAlbumAccess(userID, albumID)
|
||||||
|
if err != nil {
|
||||||
|
return []models.Media{}, err
|
||||||
|
}
|
||||||
|
if accessLevel < 0 {
|
||||||
|
return []models.Media{}, fmt.Errorf("user does not have permission to view media in this album")
|
||||||
|
}
|
||||||
|
media := []models.Media{}
|
||||||
|
result := db.GetDB().Where("album_id = ?", albumID).Find(&media)
|
||||||
|
if result.Error != nil {
|
||||||
|
return []models.Media{}, result.Error
|
||||||
|
}
|
||||||
|
return media, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetMedia(albumID string, mediaID string) (models.Media, error) {
|
||||||
|
media := models.Media{}
|
||||||
|
result := db.GetDB().First(&media, "album_id = ? AND id = ?", albumID, mediaID)
|
||||||
|
if result.Error != nil {
|
||||||
|
return models.Media{}, result.Error
|
||||||
|
}
|
||||||
|
return media, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -0,0 +1,65 @@
|
|||||||
|
import { useState, useEffect } from 'react'
|
||||||
|
|
||||||
|
export default function AuthImage({ src, token, alt, className, ...props }) {
|
||||||
|
const [imageSrc, setImageSrc] = useState(null)
|
||||||
|
const [loading, setLoading] = useState(true)
|
||||||
|
const [error, setError] = useState(false)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
let objectUrl = null
|
||||||
|
let active = true
|
||||||
|
|
||||||
|
const fetchImage = async () => {
|
||||||
|
setLoading(true)
|
||||||
|
setError(false)
|
||||||
|
try {
|
||||||
|
const response = await fetch(src, {
|
||||||
|
headers: {
|
||||||
|
'Authorization': token
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error('Failed to load image')
|
||||||
|
}
|
||||||
|
|
||||||
|
const blob = await response.blob()
|
||||||
|
if (active) {
|
||||||
|
objectUrl = URL.createObjectURL(blob)
|
||||||
|
setImageSrc(objectUrl)
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
if (active) {
|
||||||
|
console.error("Error loading image:", err)
|
||||||
|
setError(true)
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (src && token) {
|
||||||
|
fetchImage()
|
||||||
|
} else {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
active = false
|
||||||
|
if (objectUrl) {
|
||||||
|
URL.revokeObjectURL(objectUrl)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [src, token])
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <div className={`bg-gray-800 animate-pulse ${className}`} />
|
||||||
|
}
|
||||||
|
|
||||||
|
if (error || !imageSrc) {
|
||||||
|
return <div className={`bg-gray-800 flex items-center justify-center text-gray-500 ${className}`}>Error</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
return <img src={imageSrc} alt={alt} className={className} {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
import { Upload } from 'lucide-react'
|
import { Upload } from 'lucide-react'
|
||||||
import MediaUploadModal from './MediaUploadModal'
|
import MediaUploadModal from './MediaUploadModal'
|
||||||
|
import AuthImage from '../../components/AuthImage'
|
||||||
import { useState, useEffect } from 'react'
|
import { useState, useEffect } from 'react'
|
||||||
import { getServerUrl } from '../../hooks/getConstants'
|
import { getServerUrl } from '../../hooks/getConstants'
|
||||||
import { useAccount } from '../../contexts/useAccount'
|
import { useAccount } from '../../contexts/useAccount'
|
||||||
@@ -14,12 +15,23 @@ export default function MediaList({ albumId, albumName }) {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
let ignore = false;
|
let ignore = false;
|
||||||
const getMedia = async () => {
|
const getMedia = async () => {
|
||||||
if (!albumId) return
|
|
||||||
|
|
||||||
// TODO: Implement media fetching from API
|
// TODO: Implement media fetching from API
|
||||||
// const response = await fetch(...)
|
// const response = await fetch(...)
|
||||||
|
const response = await fetch(`${getServerUrl()}/api/media/getAllMediaInAlbum?albumId=${albumId}`, {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'Authorization': getAccessToken(),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const data = await response.json()
|
||||||
|
if (ignore) return
|
||||||
|
if (data.error) {
|
||||||
|
showError(data.error)
|
||||||
|
} else {
|
||||||
|
console.log("data.media", data.media)
|
||||||
|
setMedia(data.media)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
getMedia()
|
getMedia()
|
||||||
return () => { ignore = true; }
|
return () => { ignore = true; }
|
||||||
}, [albumId])
|
}, [albumId])
|
||||||
@@ -31,12 +43,21 @@ export default function MediaList({ albumId, albumName }) {
|
|||||||
<Upload className="w-6 h-6 cursor-pointer" color="white" onClick={() => setOpen(true)} />
|
<Upload className="w-6 h-6 cursor-pointer" color="white" onClick={() => setOpen(true)} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Media Grid Placeholder */}
|
{/* Media Grid */}
|
||||||
<div className="flex flex-row items-center justify-start gap-2 w-full px-6 flex-wrap">
|
<div className="flex flex-row items-center justify-start gap-2 w-full px-6 flex-wrap">
|
||||||
{media.length === 0 && (
|
{media.length === 0 && (
|
||||||
<p className="text-gray-500 red-hat-text">No media in this album</p>
|
<p className="text-gray-500 red-hat-text">No media in this album</p>
|
||||||
)}
|
)}
|
||||||
{/* Render media items here */}
|
{media.map((media) => (
|
||||||
|
<div key={media.id} className="flex flex-col items-center justify-center w-24 h-24 bg-[#1A1A1A] rounded-md border border-[#2B2B2B]">
|
||||||
|
<AuthImage
|
||||||
|
src={`${getServerUrl()}/api/media/${albumId ? albumId : 'root'}/${media.ID}`}
|
||||||
|
token={getAccessToken()}
|
||||||
|
alt={media.Title}
|
||||||
|
className="w-full h-full object-cover"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<MediaUploadModal
|
<MediaUploadModal
|
||||||
|
|||||||
Reference in New Issue
Block a user