fix(workflow): stop leaking peer share tokens from participant session API (#6241)

This commit is contained in:
ConnorYoh
2026-04-28 17:36:20 +01:00
committed by GitHub
parent b966e771a4
commit 4e4918b91e
4 changed files with 194 additions and 11 deletions

View File

@@ -101,7 +101,9 @@ public class WorkflowParticipantController {
}
WorkflowSession session = participant.getWorkflowSession();
return ResponseEntity.ok(WorkflowMapper.toResponse(session));
// Strip peer share tokens — a single participant token must not enumerate peer bearer
// tokens (GHSA-qgg6-mxw4-xg62).
return ResponseEntity.ok(WorkflowMapper.toResponse(session, null, false));
}
@Operation(
@@ -122,7 +124,7 @@ public class WorkflowParticipantController {
HttpStatus.FORBIDDEN,
"Invalid or expired participant token"));
return ResponseEntity.ok(WorkflowMapper.toParticipantResponse(participant));
return ResponseEntity.ok(WorkflowMapper.toParticipantResponse(participant, false));
}
@Operation(
@@ -181,7 +183,7 @@ public class WorkflowParticipantController {
participant.getEmail(),
participant.getWorkflowSession().getSessionId());
return ResponseEntity.ok(WorkflowMapper.toParticipantResponse(participant));
return ResponseEntity.ok(WorkflowMapper.toParticipantResponse(participant, false));
} catch (ResponseStatusException e) {
throw e;
@@ -235,7 +237,7 @@ public class WorkflowParticipantController {
participant.getEmail(),
participant.getWorkflowSession().getSessionId());
return ResponseEntity.ok(WorkflowMapper.toParticipantResponse(participant));
return ResponseEntity.ok(WorkflowMapper.toParticipantResponse(participant, false));
}
@Operation(

View File

@@ -21,7 +21,7 @@ public class WorkflowMapper {
/** Converts a WorkflowSession entity to a response DTO. */
public static WorkflowSessionResponse toResponse(WorkflowSession session) {
return toResponse(session, null);
return toResponse(session, null, true);
}
/**
@@ -34,6 +34,21 @@ public class WorkflowMapper {
*/
public static WorkflowSessionResponse toResponse(
WorkflowSession session, ObjectMapper objectMapper) {
return toResponse(session, objectMapper, true);
}
/**
* Converts a WorkflowSession entity to a response DTO.
*
* @param session The workflow session entity
* @param objectMapper ObjectMapper for JSON processing (null to skip wet signature extraction)
* @param includeShareTokens Whether to include each participant's share token in the response.
* Owner-facing endpoints set this to true so the owner can distribute share links;
* participant-facing endpoints set this to false so a single participant's token cannot be
* used to enumerate peer bearer tokens (GHSA-qgg6-mxw4-xg62).
*/
public static WorkflowSessionResponse toResponse(
WorkflowSession session, ObjectMapper objectMapper, boolean includeShareTokens) {
if (session == null) {
return null;
}
@@ -64,12 +79,12 @@ public class WorkflowMapper {
if (objectMapper != null) {
response.setParticipants(
session.getParticipants().stream()
.map(p -> toParticipantResponse(p, objectMapper))
.map(p -> toParticipantResponse(p, objectMapper, includeShareTokens))
.collect(Collectors.toList()));
} else {
response.setParticipants(
session.getParticipants().stream()
.map(WorkflowMapper::toParticipantResponse)
.map(p -> toParticipantResponse(p, includeShareTokens))
.collect(Collectors.toList()));
}
@@ -88,8 +103,21 @@ public class WorkflowMapper {
return response;
}
/** Converts a WorkflowParticipant entity to a response DTO. */
/** Converts a WorkflowParticipant entity to a response DTO, including the share token. */
public static ParticipantResponse toParticipantResponse(WorkflowParticipant participant) {
return toParticipantResponse(participant, true);
}
/**
* Converts a WorkflowParticipant entity to a response DTO.
*
* @param participant The participant entity
* @param includeShareToken Whether to include the participant's share token in the response.
* Owner-facing endpoints set this to true; participant-facing endpoints set this to false
* so the response cannot be used to enumerate peer bearer tokens (GHSA-qgg6-mxw4-xg62).
*/
public static ParticipantResponse toParticipantResponse(
WorkflowParticipant participant, boolean includeShareToken) {
if (participant == null) {
return null;
}
@@ -102,7 +130,9 @@ public class WorkflowMapper {
response.setEmail(participant.getEmail());
response.setName(participant.getName());
response.setStatus(participant.getStatus());
response.setShareToken(participant.getShareToken());
if (includeShareToken) {
response.setShareToken(participant.getShareToken());
}
response.setAccessRole(participant.getAccessRole());
response.setExpiresAt(participant.getExpiresAt());
response.setLastUpdated(participant.getLastUpdated());
@@ -122,7 +152,19 @@ public class WorkflowMapper {
*/
public static ParticipantResponse toParticipantResponse(
WorkflowParticipant participant, ObjectMapper objectMapper) {
ParticipantResponse response = toParticipantResponse(participant);
return toParticipantResponse(participant, objectMapper, true);
}
/**
* Converts a WorkflowParticipant entity to a response DTO with wet signatures extracted.
*
* @param participant The participant entity
* @param objectMapper ObjectMapper for JSON processing
* @param includeShareToken Whether to include the participant's share token in the response.
*/
public static ParticipantResponse toParticipantResponse(
WorkflowParticipant participant, ObjectMapper objectMapper, boolean includeShareToken) {
ParticipantResponse response = toParticipantResponse(participant, includeShareToken);
if (response != null) {
response.setWetSignatures(extractWetSignatures(participant, objectMapper));
}

View File

@@ -0,0 +1,136 @@
package stirling.software.proprietary.workflow.util;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertNull;
import org.junit.jupiter.api.Test;
import stirling.software.proprietary.security.model.User;
import stirling.software.proprietary.storage.model.ShareAccessRole;
import stirling.software.proprietary.storage.model.StoredFile;
import stirling.software.proprietary.workflow.dto.ParticipantResponse;
import stirling.software.proprietary.workflow.dto.WorkflowSessionResponse;
import stirling.software.proprietary.workflow.model.ParticipantStatus;
import stirling.software.proprietary.workflow.model.WorkflowParticipant;
import stirling.software.proprietary.workflow.model.WorkflowSession;
import stirling.software.proprietary.workflow.model.WorkflowType;
/**
* Regression test for GHSA-qgg6-mxw4-xg62 — verifies that the {@code includeShareToken(s)} flag
* controls whether {@link WorkflowMapper} discloses participant bearer tokens in responses.
* Owner-facing endpoints must still receive tokens (so they can distribute share links);
* participant-facing endpoints must not, so a single participant token cannot be used to enumerate
* peer bearer tokens.
*/
class WorkflowMapperShareTokenTest {
private static final String TOKEN_A = "token-aaaa-1111";
private static final String TOKEN_B = "token-bbbb-2222";
private WorkflowSession buildSessionWithTwoParticipants() {
User owner = new User();
owner.setId(1L);
owner.setUsername("owner@example.com");
StoredFile original = new StoredFile();
original.setId(42L);
WorkflowSession session = new WorkflowSession();
session.setSessionId("session-xyz");
session.setOwner(owner);
session.setOriginalFile(original);
session.setWorkflowType(WorkflowType.SIGNING);
session.setDocumentName("contract.pdf");
WorkflowParticipant a = new WorkflowParticipant();
a.setId(10L);
a.setEmail("alice@example.com");
a.setName("Alice");
a.setStatus(ParticipantStatus.PENDING);
a.setShareToken(TOKEN_A);
a.setAccessRole(ShareAccessRole.EDITOR);
session.addParticipant(a);
WorkflowParticipant b = new WorkflowParticipant();
b.setId(11L);
b.setEmail("bob@example.com");
b.setName("Bob");
b.setStatus(ParticipantStatus.PENDING);
b.setShareToken(TOKEN_B);
b.setAccessRole(ShareAccessRole.EDITOR);
session.addParticipant(b);
return session;
}
@Test
void toResponse_legacyOverload_includesShareTokensForOwnerCompatibility() {
WorkflowSession session = buildSessionWithTwoParticipants();
WorkflowSessionResponse response = WorkflowMapper.toResponse(session);
assertNotNull(response);
assertEquals(2, response.getParticipants().size());
assertEquals(TOKEN_A, response.getParticipants().get(0).getShareToken());
assertEquals(TOKEN_B, response.getParticipants().get(1).getShareToken());
}
@Test
void toResponse_withIncludeShareTokensFalse_stripsAllPeerTokens() {
WorkflowSession session = buildSessionWithTwoParticipants();
WorkflowSessionResponse response = WorkflowMapper.toResponse(session, null, false);
assertNotNull(response);
assertEquals(2, response.getParticipants().size());
for (ParticipantResponse p : response.getParticipants()) {
assertNull(
p.getShareToken(),
"Participant share token must not be exposed in participant-facing responses");
}
}
@Test
void toResponse_withIncludeShareTokensFalse_preservesOtherFields() {
WorkflowSession session = buildSessionWithTwoParticipants();
WorkflowSessionResponse response = WorkflowMapper.toResponse(session, null, false);
ParticipantResponse alice = response.getParticipants().get(0);
assertEquals(10L, alice.getId());
assertEquals("alice@example.com", alice.getEmail());
assertEquals("Alice", alice.getName());
assertEquals(ParticipantStatus.PENDING, alice.getStatus());
assertEquals(ShareAccessRole.EDITOR, alice.getAccessRole());
}
@Test
void toParticipantResponse_legacyOverload_includesShareToken() {
WorkflowParticipant p = new WorkflowParticipant();
p.setId(1L);
p.setEmail("a@example.com");
p.setStatus(ParticipantStatus.PENDING);
p.setShareToken(TOKEN_A);
ParticipantResponse response = WorkflowMapper.toParticipantResponse(p);
assertEquals(TOKEN_A, response.getShareToken());
}
@Test
void toParticipantResponse_withIncludeShareTokenFalse_stripsToken() {
WorkflowParticipant p = new WorkflowParticipant();
p.setId(1L);
p.setEmail("a@example.com");
p.setStatus(ParticipantStatus.PENDING);
p.setShareToken(TOKEN_A);
ParticipantResponse response = WorkflowMapper.toParticipantResponse(p, false);
assertNull(response.getShareToken());
assertEquals(1L, response.getId());
assertEquals("a@example.com", response.getEmail());
assertEquals(ParticipantStatus.PENDING, response.getStatus());
}
}

View File

@@ -6,7 +6,10 @@ export interface ParticipantResponse {
email: string;
name: string;
status: "PENDING" | "NOTIFIED" | "VIEWED" | "SIGNED" | "DECLINED";
shareToken: string;
// Null for participant-facing endpoints (`/api/v1/workflow/participant/...`); the owner-facing
// `/api/v1/security/cert-sign/sessions/...` endpoints still populate it for share-link
// distribution. Never used to look up other participants — see GHSA-qgg6-mxw4-xg62.
shareToken: string | null;
accessRole: "EDITOR" | "COMMENTER" | "VIEWER";
expiresAt?: string;
lastUpdated: string;