Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

43 events gallery of photos #203

Merged
merged 41 commits into from
Nov 20, 2024
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
6885bee
Adding gallery to event model and simple service methods
AvilaAndre Sep 16, 2023
6eca9e9
Uploading image
AvilaAndre Sep 26, 2023
d6a1f3d
removed code added by accident
AvilaAndre Oct 17, 2023
0e67a81
Changed galery image folder
AvilaAndre Oct 17, 2023
28c58f1
Created addPhoto test
AvilaAndre Oct 17, 2023
3f061dc
fixed photoUrl type and added errors when event/photo doesn't exist
AvilaAndre Oct 18, 2023
a8b0c9e
test photo add if event does not exist
AvilaAndre Oct 23, 2023
e2262dd
test photo add wrong image format test
AvilaAndre Oct 23, 2023
8a2f4b9
Remove photo from gallery test
AvilaAndre Oct 23, 2023
79ee8f1
gallery photo remove from unexistent event test
AvilaAndre Oct 23, 2023
c8d19ef
gallery remove unexistent photo test
AvilaAndre Oct 23, 2023
4b0634f
generalized gallery to activity
AvilaAndre Jan 30, 2024
50acea5
Move gallery service methods from Event to Activity
AvilaAndre Feb 4, 2024
91f0ea6
Merge branch 'develop' into 43-events-gallery-of-photos
AvilaAndre Feb 6, 2024
91ace92
added gallery to develop tests
AvilaAndre Feb 6, 2024
a3516e7
added gallery documentation to activity payload
AvilaAndre Feb 6, 2024
cd6f37f
Update src/main/kotlin/pt/up/fe/ni/website/backend/model/Activity.kt
AvilaAndre Mar 28, 2024
971b2c0
Added missing import
AvilaAndre Mar 28, 2024
6b3e978
Formatted imports
AvilaAndre Mar 28, 2024
c6f4c3c
Removed unnnecessary code
AvilaAndre Mar 28, 2024
31d5a4b
Replaced photo with image
AvilaAndre Mar 28, 2024
8639688
added image folder to gallery
AvilaAndre Mar 28, 2024
1863c6f
Updated gallery documentation
AvilaAndre Mar 28, 2024
250bb18
Fixing tests failing after changes
AvilaAndre Mar 29, 2024
dd4818b
replace endpoints to be restful and add delete method to testing
AvilaAndre Jul 20, 2024
1b664c1
generalized image to file
AvilaAndre Jul 20, 2024
02ced4e
added file deletion to FileUploader class and delete files on removal
AvilaAndre Jul 21, 2024
26df8a7
feat: serve static files on the /static/ path
AvilaAndre Oct 18, 2024
7d42ee7
tweak: change static port from 3000 to 8080 for development
AvilaAndre Oct 18, 2024
95f4a50
Merge branch 'develop' into 43-events-gallery-of-photos
AvilaAndre Oct 18, 2024
0c68988
fix: add gallery list to Auth test
AvilaAndre Oct 18, 2024
02f45a9
lint: line too long
AvilaAndre Oct 18, 2024
693afa8
Update src/main/kotlin/pt/up/fe/ni/website/backend/service/activity/A…
AvilaAndre Nov 6, 2024
c796bac
Update src/test/kotlin/pt/up/fe/ni/website/backend/utils/mockmvc/Mock…
AvilaAndre Nov 6, 2024
0b5c228
fix: Added explanation to line and replaced variable
AvilaAndre Nov 6, 2024
2735fdc
lint: missing space
AvilaAndre Nov 6, 2024
12d6616
docs: started documentation while #144 is not done
AvilaAndre Nov 6, 2024
2322b14
test: added gallery tests to Project
AvilaAndre Nov 6, 2024
aeef473
refactor: make slug non optional
AvilaAndre Nov 7, 2024
2f26ff6
Merge branch 'develop' into 43-events-gallery-of-photos
AvilaAndre Nov 7, 2024
88e915d
Merge branch 'develop' into 43-events-gallery-of-photos
MRita443 Nov 20, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -38,3 +38,6 @@ out/

### VS Code ###
.vscode/

### Static Content
static/
Original file line number Diff line number Diff line change
Expand Up @@ -12,12 +12,12 @@ import org.springframework.security.access.AccessDeniedException
import org.springframework.security.core.AuthenticationException
import org.springframework.web.bind.MethodArgumentNotValidException
import org.springframework.web.bind.annotation.ExceptionHandler
import org.springframework.web.bind.annotation.RequestMapping
import org.springframework.web.bind.annotation.ResponseStatus
import org.springframework.web.bind.annotation.RestController
import org.springframework.web.bind.annotation.RestControllerAdvice
import org.springframework.web.multipart.MaxUploadSizeExceededException
import org.springframework.web.multipart.support.MissingServletRequestPartException
import org.springframework.web.servlet.NoHandlerFoundException
import pt.up.fe.ni.website.backend.config.Logging

data class SimpleError(
Expand All @@ -32,7 +32,7 @@ data class CustomError(val errors: List<SimpleError>)
@RestControllerAdvice
class ErrorController(private val objectMapper: ObjectMapper) : ErrorController, Logging {

@RequestMapping("/**")
@ExceptionHandler(NoHandlerFoundException::class)
@ResponseStatus(HttpStatus.NOT_FOUND)
fun endpointNotFound(): CustomError = wrapSimpleError("invalid endpoint")

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,15 +75,15 @@ class EventController(private val service: EventService) {
@PathVariable idAccount: Long
) = service.removeTeamMemberById(idEvent, idAccount)

@PutMapping("/{idEvent}/gallery/addImage", consumes = ["multipart/form-data"])
@PutMapping("/{idEvent}/gallery", consumes = ["multipart/form-data"])
fun addGalleryImage(
@PathVariable idEvent: Long,
@RequestParam
@ValidImage
image: MultipartFile
) = service.addGalleryImage(idEvent, image)
) = service.addGalleryImage(idEvent, image, EventService.IMAGE_FOLDER)

@PutMapping("/{idEvent}/gallery/removeImage")
@DeleteMapping("/{idEvent}/gallery")
fun removeGalleryImage(
@PathVariable idEvent: Long,
@RequestPart imageUrl: String
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ object ErrorMessages {

fun roleNotFound(id: Long): String = "role not found with id $id"

fun imageNotFound(imageName: String): String = "image not found with name $imageName"
fun fileNotFound(fileName: String): String = "file not found with name $fileName"

fun userAlreadyHasRole(roleId: Long, userId: Long): String = "user $userId already has role $roleId"

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,6 @@ import org.springframework.stereotype.Service
import org.springframework.web.multipart.MultipartFile
import pt.up.fe.ni.website.backend.dto.entity.ActivityDto
import pt.up.fe.ni.website.backend.model.Activity
import pt.up.fe.ni.website.backend.model.Event
import pt.up.fe.ni.website.backend.model.Project
import pt.up.fe.ni.website.backend.repository.ActivityRepository
import pt.up.fe.ni.website.backend.service.AccountService
import pt.up.fe.ni.website.backend.service.ErrorMessages
Expand All @@ -18,10 +16,10 @@ abstract class AbstractActivityService<T : Activity>(
protected val accountService: AccountService,
protected val fileUploader: FileUploader
) {
fun getAll() = repository.findAll().toList()

fun getActivityById(id: Long): T =
repository.findByIdOrNull(id)
?: throw NoSuchElementException(ErrorMessages.activityNotFound(id))
fun getActivityById(id: Long): T = repository.findByIdOrNull(id)
?: throw NoSuchElementException(ErrorMessages.activityNotFound(id))

fun <U : ActivityDto<T>> createActivity(dto: U, imageFolder: String): T {
repository.findBySlug(dto.slug)?.let {
Expand Down Expand Up @@ -85,17 +83,9 @@ abstract class AbstractActivityService<T : Activity>(
return repository.save(activity)
}

fun addGalleryImage(activityId: Long, image: MultipartFile): Activity {
fun addGalleryImage(activityId: Long, image: MultipartFile, imageFolder: String): Activity {
val activity = getActivityById(activityId)

var imageFolder = "activities"

if (activity is Event) {
imageFolder = EventService.IMAGE_FOLDER
} else if (activity is Project) {
imageFolder = ProjectService.IMAGE_FOLDER
}

val fileName = fileUploader.buildFileName(image, activity.title)
val imageName = fileUploader.uploadImage(imageFolder + "/gallery", fileName, image.bytes)
AvilaAndre marked this conversation as resolved.
Show resolved Hide resolved

Expand All @@ -107,10 +97,11 @@ abstract class AbstractActivityService<T : Activity>(
fun removeGalleryImage(activityId: Long, imageName: String): Activity {
AvilaAndre marked this conversation as resolved.
Show resolved Hide resolved
val activity = getActivityById(activityId)

val imageRemoved = activity.gallery.remove(imageName)

if (!imageRemoved) {
throw NoSuchElementException(ErrorMessages.imageNotFound(imageName))
if (activity.gallery.contains(imageName)) {
fileUploader.deleteImage(imageName)
activity.gallery.remove(imageName)
} else {
throw NoSuchElementException(ErrorMessages.fileNotFound(imageName))
}

return repository.save(activity)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,4 +20,11 @@ class CloudinaryFileUploader(private val basePath: String, private val cloudinar

return result["url"]?.toString() ?: ""
}

override fun deleteImage(fileName: String) {
cloudinary.uploader().destroy(
fileName,
MRita443 marked this conversation as resolved.
Show resolved Hide resolved
null
)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import pt.up.fe.ni.website.backend.utils.extensions.filenameExtension

abstract class FileUploader {
abstract fun uploadImage(folder: String, fileName: String, image: ByteArray): String
abstract fun deleteImage(fileName: String)

fun buildFileName(photoFile: MultipartFile, prefix: String = ""): String {
val limitedPrefix = prefix.take(100) // File name length has a limit of 256 characters
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,4 +11,11 @@ class StaticFileUploader(private val storePath: String, private val servePath: S

return "$servePath/$folder/$fileName"
}

override fun deleteImage(fileName: String) {
val storedFileName = fileName.replace(servePath, storePath)
val file = File(storedFileName)

if (file.exists()) file.delete()
}
}
9 changes: 7 additions & 2 deletions src/main/resources/application.properties
Original file line number Diff line number Diff line change
Expand Up @@ -32,9 +32,14 @@ upload.provider=static
upload.cloudinary-base-path=website
upload.cloudinary-url=GET_YOURS_AT_CLOUDINARY_DASHBOARD
# Folder in which files will be stored
upload.static-path=classpath:static
upload.static-path=file:static
# URL that will serve static content
upload.static-serve=http://localhost:3000/static
upload.static-serve=http://localhost:8080/static

# Files are served in the following path
spring.mvc.static-path-pattern=/static/**
MRita443 marked this conversation as resolved.
Show resolved Hide resolved
# Add NoHandlerFoundException as an exception
spring.mvc.throw-exception-if-no-handler-found=true

# Cors Origin
cors.allow-origin = http://localhost:3000
Original file line number Diff line number Diff line change
Expand Up @@ -244,8 +244,8 @@ class AuthControllerTest @Autowired constructor(
inner class CheckPermissions {
private val testPermissions = listOf(Permission.CREATE_ACCOUNT, Permission.CREATE_ACTIVITY)
private val testActivity = Project(
"Test Activity", "Test Description", mutableListOf(), mutableListOf(), "test slug", "test image", false,
emptyList(), null, "test target audience"
"Test Activity", "Test Description", mutableListOf(), mutableListOf(), "test slug", "test image",
mutableListOf(), false, emptyList(), null, "test target audience"
)
private val testRole = Role("MEMBER", Permissions(testPermissions), false)
private val testPerActivityRole = PerActivityRole(Permissions(listOf(Permission.EDIT_ACTIVITY)))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -813,7 +813,7 @@ internal class EventControllerTest @Autowired constructor(
}

@NestedTest
@DisplayName("PUT /events/{idEvent}/gallery/addImage")
@DisplayName("PUT /events/{idEvent}/gallery")
inner class AddGalleryImage {

private val uuid: UUID = UUID.randomUUID()
Expand All @@ -839,7 +839,7 @@ internal class EventControllerTest @Autowired constructor(
fun `should add an image`() {
val expectedImagePath = "${uploadConfigProperties.staticServe}/events/gallery/${testEvent.title}-$uuid.jpeg"

mockMvc.multipartBuilder("/events/${testEvent.id}/gallery/addImage")
mockMvc.multipartBuilder("/events/${testEvent.id}/gallery")
.asPutMethod()
.addFile("image", contentType = MediaType.IMAGE_JPEG_VALUE)
.perform()
Expand All @@ -865,7 +865,7 @@ internal class EventControllerTest @Autowired constructor(
fun `should fail if event does not exist`() {
val unexistentID = 5
AvilaAndre marked this conversation as resolved.
Show resolved Hide resolved

mockMvc.multipartBuilder("/events/$unexistentID/gallery/addImage")
mockMvc.multipartBuilder("/events/$unexistentID/gallery")
.asPutMethod()
.addFile("image", contentType = MediaType.IMAGE_JPEG_VALUE)
.perform()
Expand All @@ -879,7 +879,7 @@ internal class EventControllerTest @Autowired constructor(

@Test
fun `should fail if image in wrong format`() {
mockMvc.multipartBuilder("/events/${testEvent.id}/gallery/addImage")
mockMvc.multipartBuilder("/events/${testEvent.id}/gallery")
.asPutMethod()
.addFile("image", filename = "image.gif", contentType = MediaType.IMAGE_JPEG_VALUE)
.perform()
Expand All @@ -892,8 +892,9 @@ internal class EventControllerTest @Autowired constructor(
}
}

@Nested
@NestedTest
@DisplayName("PUT /events/{idEvent}/gallery/removeImage")
@DisplayName("DELETE /events/{idEvent}/gallery")
inner class RemoveGalleryImage {

private val uuid: UUID = UUID.randomUUID()
Expand Down Expand Up @@ -921,8 +922,8 @@ internal class EventControllerTest @Autowired constructor(

@Test
fun `should remove an image`() {
mockMvc.multipartBuilder("/events/${testEvent.id}/gallery/removeImage")
.asPutMethod()
mockMvc.multipartBuilder("/events/${testEvent.id}/gallery")
.asDeleteMethod()
.addPart("imageUrl", mockImageUrl)
.perform()
.andExpectAll(
Expand All @@ -946,8 +947,8 @@ internal class EventControllerTest @Autowired constructor(
fun `should fail if event does not exist`() {
val unexistentID = 5
AvilaAndre marked this conversation as resolved.
Show resolved Hide resolved

mockMvc.multipartBuilder("/events/$unexistentID/gallery/removeImage")
.asPutMethod()
mockMvc.multipartBuilder("/events/$unexistentID/gallery")
.asDeleteMethod()
.addPart("imageUrl", mockImageUrl)
.perform()
.andExpectAll(
Expand All @@ -962,15 +963,15 @@ internal class EventControllerTest @Autowired constructor(
fun `should fail if image does not exist`() {
val wrongImageUrl = "${uploadConfigProperties.staticServe}/gallery/Another${testEvent.title}-$uuid.jpeg"

mockMvc.multipartBuilder("/events/${testEvent.id}/gallery/removeImage")
.asPutMethod()
mockMvc.multipartBuilder("/events/${testEvent.id}/gallery")
.asDeleteMethod()
.addPart("imageUrl", wrongImageUrl)
.perform()
.andExpectAll(
status().isNotFound,
content().contentType(MediaType.APPLICATION_JSON),
jsonPath("$.errors.length()").value(1),
jsonPath("$.errors[0].message").value("image not found with name $wrongImageUrl")
jsonPath("$.errors[0].message").value("file not found with name $wrongImageUrl")
)
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,16 @@ class MockMvcMultipartBuilder(
return this
}

fun asDeleteMethod(): MockMvcMultipartBuilder {
multipart.with(
{
it.method = "DELETE"
it
}
)
return this
}

AvilaAndre marked this conversation as resolved.
Show resolved Hide resolved
fun perform(): ResultActions {
return mockMvc.perform(multipart)
}
Expand Down
Loading