@@ -25,7 +25,7 @@ import {ShipService} from './ship.service';
25
25
import { ReadShipDto , UpdateShipDto } from './ship.dto' ;
26
26
import { User } from '../user/user.schema' ;
27
27
import { Auth , AuthUser } from '../auth/auth.decorator' ;
28
- import { NotFound , ObjectIdPipe } from '@mean-stream/nestx' ;
28
+ import { notFound , NotFound , ObjectIdPipe } from '@mean-stream/nestx' ;
29
29
import { Types } from 'mongoose' ;
30
30
import { Ship , ShipDocument } from './ship.schema' ;
31
31
import { EmpireDocument } from '../empire/empire.schema' ;
@@ -59,13 +59,12 @@ export class ShipController {
59
59
async getFleetShips (
60
60
@AuthUser ( ) user : User ,
61
61
@Param ( 'game' , ObjectIdPipe ) game : Types . ObjectId ,
62
- @Param ( 'fleet' , ObjectIdPipe ) fleetId : Types . ObjectId ,
62
+ @Param ( 'fleet' , ObjectIdPipe ) fleet : Types . ObjectId ,
63
63
@Query ( 'type' ) type ?: ShipTypeName ,
64
64
) : Promise < ReadShipDto [ ] | Ship [ ] > {
65
- const fleet = await this . getFleet ( fleetId ) ;
66
- const ships = await this . shipService . findAll ( { fleet : fleet . _id , type} ) ;
67
- const empire = await this . empireService . findOne ( { game, user : user . _id } ) ;
68
- return this . checkUserAccess ( fleet , empire ) ? ships : ships . map ( ship => this . toReadShipDto ( ship . toObject ( ) ) ) ;
65
+ const isOwner = await this . hasUserAccess ( game , fleet , user ) ;
66
+ const ships = await this . shipService . findAll ( { game, fleet, type} ) ;
67
+ return isOwner ? ships : ships . map ( ship => this . toReadShipDto ( ship . toObject ( ) ) ) ;
69
68
}
70
69
71
70
@Get ( ':id' )
@@ -74,14 +73,13 @@ export class ShipController {
74
73
@NotFound ( 'Ship not found.' )
75
74
async getFleetShip (
76
75
@Param ( 'game' , ObjectIdPipe ) game : Types . ObjectId ,
77
- @Param ( 'fleet' , ObjectIdPipe ) fleetId : Types . ObjectId ,
76
+ @Param ( 'fleet' , ObjectIdPipe ) fleet : Types . ObjectId ,
78
77
@Param ( 'id' , ObjectIdPipe ) id : Types . ObjectId ,
79
78
@AuthUser ( ) user : User ,
80
- ) : Promise < ReadShipDto | Ship > {
81
- const fleet = await this . getFleet ( fleetId ) ;
82
- const ship = await this . getShip ( id , fleet . _id ) ;
83
- const empire = await this . empireService . findOne ( { game, user : user . _id } ) ;
84
- return this . checkUserAccess ( fleet , empire ) ? ship : this . toReadShipDto ( ship . toObject ( ) ) ;
79
+ ) : Promise < ReadShipDto | Ship | null > {
80
+ const isOwner = await this . hasUserAccess ( game , fleet , user ) ;
81
+ const ship = await this . shipService . findOne ( { game, fleet, _id : id } ) ;
82
+ return ! ship || isOwner ? ship : this . toReadShipDto ( ship . toObject ( ) ) ;
85
83
}
86
84
87
85
@Patch ( ':id' )
@@ -92,19 +90,17 @@ export class ShipController {
92
90
@ApiConflictResponse ( { description : 'Both fleets need to be in the same location to transfer ships.' } )
93
91
async updateFleetShip (
94
92
@Param ( 'game' , ObjectIdPipe ) game : Types . ObjectId ,
95
- @Param ( 'fleet' , ObjectIdPipe ) fleetId : Types . ObjectId ,
93
+ @Param ( 'fleet' , ObjectIdPipe ) fleet : Types . ObjectId ,
96
94
@Param ( 'id' , ObjectIdPipe ) id : Types . ObjectId ,
97
95
@Body ( ) updateShipDto : UpdateShipDto ,
98
96
@AuthUser ( ) user : User ,
99
97
) : Promise < Ship | null > {
100
- const fleet = await this . getFleet ( fleetId ) ;
101
- const ship = await this . getShip ( id , fleet . _id ) ;
102
- await this . getUserEmpireAccess ( game , user , ship ) ;
98
+ await this . checkUserAccess ( game , fleet , user ) ;
103
99
104
100
// Change fleet if in same system
105
- if ( updateShipDto . fleet && ! updateShipDto . fleet . equals ( ship . fleet ) ) {
101
+ if ( updateShipDto . fleet && ! updateShipDto . fleet . equals ( fleet ) ) {
106
102
const newFleet = await this . fleetService . find ( updateShipDto . fleet , { game} ) ;
107
- const currentFleet = await this . fleetService . find ( ship . fleet , { game} ) ;
103
+ const currentFleet = await this . fleetService . find ( fleet , { game} ) ;
108
104
109
105
if ( ! newFleet || ! currentFleet ) {
110
106
throw new NotFoundException ( 'Fleet not found.' ) ;
@@ -114,7 +110,7 @@ export class ShipController {
114
110
throw new ConflictException ( 'Both fleets need to be in the same location to transfer ships.' ) ;
115
111
}
116
112
}
117
- return this . shipService . update ( id , updateShipDto ) ;
113
+ return this . shipService . updateOne ( { game , fleet , _id : id } , updateShipDto ) ;
118
114
}
119
115
120
116
@Delete ( ':id' )
@@ -124,44 +120,28 @@ export class ShipController {
124
120
@ApiForbiddenResponse ( { description : 'You do not own this ship.' } )
125
121
async deleteFleetShip (
126
122
@Param ( 'game' , ObjectIdPipe ) game : Types . ObjectId ,
127
- @Param ( 'fleet' , ObjectIdPipe ) fleetId : Types . ObjectId ,
123
+ @Param ( 'fleet' , ObjectIdPipe ) fleet : Types . ObjectId ,
128
124
@Param ( 'id' , ObjectIdPipe ) id : Types . ObjectId ,
129
125
@AuthUser ( ) user : User ,
130
126
) : Promise < Ship | null > {
131
- const fleet = await this . getFleet ( fleetId ) ;
132
- await this . getUserEmpireAccess ( game , user , await this . getShip ( id , fleet . _id ) ) ;
133
- return this . shipService . delete ( id ) ;
127
+ await this . checkUserAccess ( game , fleet , user ) ;
128
+ return this . shipService . deleteOne ( { game, fleet, _id : id } ) ;
134
129
}
135
130
136
- private async getFleet ( fleetId : Types . ObjectId ) : Promise < FleetDocument > {
137
- const fleet = await this . fleetService . find ( fleetId ) ;
138
- if ( ! fleet ) {
139
- throw new NotFoundException ( 'Fleet not found.' ) ;
140
- }
141
- return fleet ;
142
- }
143
-
144
- private async getUserEmpireAccess ( game : Types . ObjectId , user : User , ship : ShipDocument ) : Promise < EmpireDocument > {
145
- const userEmpire = await this . empireService . findOne ( { game, user : user . _id } ) ;
146
- if ( ! userEmpire || ! ship . empire ?. equals ( userEmpire . _id ) ) {
147
- throw new ForbiddenException ( 'You do not own this ship.' ) ;
148
- }
149
- return userEmpire ;
150
- }
151
-
152
- private async getShip ( id : Types . ObjectId , fleetId : Types . ObjectId ) : Promise < ShipDocument > {
153
- const ship = await this . shipService . find ( id , { fleet : fleetId } ) ;
154
- if ( ! ship ) {
155
- throw new NotFoundException ( 'Ship not found.' ) ;
131
+ private async hasUserAccess ( game : Types . ObjectId , fleet : Types . ObjectId , user : User ) : Promise < boolean > {
132
+ const fleetDoc = await this . fleetService . findOne ( { game, _id : fleet } , { projection : 'empire' } ) ?? notFound ( `Fleet ${ fleet } not found` ) ;
133
+ if ( ! fleetDoc . empire ) {
134
+ return false ;
156
135
}
157
- return ship ;
136
+ const empire = await this . empireService . find ( fleetDoc . empire , { projection : 'user' } ) ?? notFound ( `Empire ${ fleetDoc . empire } not found` ) ;
137
+ return user . _id . equals ( empire . user ) ;
158
138
}
159
139
160
- private checkUserAccess ( fleet : FleetDocument , empire : EmpireDocument | null ) : boolean {
161
- if ( ! empire || ! fleet . empire ) {
162
- return false ;
140
+ private async checkUserAccess ( game : Types . ObjectId , fleet : Types . ObjectId , user : User ) : Promise < void > {
141
+ const isOwner = await this . hasUserAccess ( game , fleet , user ) ;
142
+ if ( ! isOwner ) {
143
+ throw new ForbiddenException ( 'You do not own this fleet.' ) ;
163
144
}
164
- return fleet . empire . equals ( empire . _id ) ;
165
145
}
166
146
167
147
private toReadShipDto ( ship : Ship ) : ReadShipDto {
0 commit comments