@@ -20,7 +20,10 @@ const debug = createDebug("discojs:client");
20
20
* Main, abstract, class representing a Disco client in a network, which handles
21
21
* communication with other nodes, be it peers or a server.
22
22
*/
23
- export abstract class Client extends EventEmitter < { 'status' : RoundStatus } > {
23
+ export abstract class Client extends EventEmitter < {
24
+ 'status' : RoundStatus ,
25
+ 'participants' : number
26
+ } > {
24
27
// Own ID provided by the network's server.
25
28
protected _ownId ?: NodeID
26
29
// The network's server.
@@ -40,7 +43,10 @@ export abstract class Client extends EventEmitter<{'status': RoundStatus}>{
40
43
* we were doing before waiting (training locally or updating our model).
41
44
* We use this attribute to store the status to rollback to when we stop waiting
42
45
*/
43
- private previousStatus : RoundStatus | undefined ;
46
+ #previousStatus: RoundStatus | undefined ;
47
+
48
+ // Current number of participants including this client in the training session
49
+ #nbOfParticipants: number = 1 ;
44
50
45
51
constructor (
46
52
public readonly url : URL , // The network server's URL to connect to
@@ -82,7 +88,7 @@ export abstract class Client extends EventEmitter<{'status': RoundStatus}>{
82
88
* the waiting status and once enough participants join, it can display the previous status again
83
89
*/
84
90
protected saveAndEmit ( status : RoundStatus ) {
85
- this . previousStatus = status
91
+ this . # previousStatus = status
86
92
this . emit ( "status" , status )
87
93
}
88
94
@@ -111,12 +117,13 @@ export abstract class Client extends EventEmitter<{'status': RoundStatus}>{
111
117
protected setupServerCallbacks ( setMessageInversionFlag : ( ) => void ) {
112
118
// Setup an event callback if the server signals that we should
113
119
// wait for more participants
114
- this . server . on ( type . WaitingForMoreParticipants , ( ) => {
120
+ this . server . on ( type . WaitingForMoreParticipants , ( event ) => {
115
121
if ( this . promiseForMoreParticipants !== undefined )
116
122
throw new Error ( "Server sent multiple WaitingForMoreParticipants messages" )
117
123
debug ( `[${ shortenId ( this . ownId ) } ] received WaitingForMoreParticipants message from server` )
118
124
// Display the waiting status right away
119
125
this . emit ( "status" , "not enough participants" )
126
+ this . nbOfParticipants = event . nbOfParticipants // emits the `participants` event
120
127
// Upon receiving a WaitingForMoreParticipants message,
121
128
// the client will await for this promise to resolve before sending its
122
129
// local weight update
@@ -129,10 +136,10 @@ export abstract class Client extends EventEmitter<{'status': RoundStatus}>{
129
136
// and directly follows with an EnoughParticipants message when the 2nd participant joins
130
137
// However, the EnoughParticipants can arrive before the NewNodeInfo (which can be much bigger)
131
138
// so we check whether we received the EnoughParticipants before being assigned a node ID
132
- this . server . once ( type . EnoughParticipants , ( ) => {
139
+ this . server . once ( type . EnoughParticipants , ( event ) => {
133
140
if ( this . _ownId === undefined ) {
134
- debug ( `Received EnoughParticipants message from server before the NewFederatedNodeInfo message` )
135
141
setMessageInversionFlag ( )
142
+ this . nbOfParticipants = event . nbOfParticipants
136
143
}
137
144
} )
138
145
}
@@ -146,10 +153,11 @@ export abstract class Client extends EventEmitter<{'status': RoundStatus}>{
146
153
protected async createPromiseForMoreParticipants ( ) : Promise < void > {
147
154
return new Promise < void > ( ( resolve ) => {
148
155
// "once" is important because we can't resolve the same promise multiple times
149
- this . server . once ( type . EnoughParticipants , ( ) => {
156
+ this . server . once ( type . EnoughParticipants , ( event ) => {
150
157
debug ( `[${ shortenId ( this . ownId ) } ] received EnoughParticipants message from server` )
151
158
// Emit the last status emitted before waiting if defined
152
- if ( this . previousStatus !== undefined ) this . emit ( "status" , this . previousStatus )
159
+ if ( this . #previousStatus !== undefined ) this . emit ( "status" , this . #previousStatus)
160
+ this . nbOfParticipants = event . nbOfParticipants
153
161
resolve ( )
154
162
} )
155
163
} )
@@ -190,7 +198,18 @@ export abstract class Client extends EventEmitter<{'status': RoundStatus}>{
190
198
* If federated, it should the number of participants excluding the server
191
199
* If local it should be 1
192
200
*/
193
- abstract getNbOfParticipants ( ) : number ;
201
+ public get nbOfParticipants ( ) : number {
202
+ return this . #nbOfParticipants
203
+ }
204
+
205
+ /**
206
+ * Setter for the number of participants
207
+ * It emits the number of participants to the client
208
+ */
209
+ public set nbOfParticipants ( nbOfParticipants : number ) {
210
+ this . #nbOfParticipants = nbOfParticipants
211
+ this . emit ( "participants" , nbOfParticipants )
212
+ }
194
213
195
214
get ownId ( ) : NodeID {
196
215
if ( this . _ownId === undefined ) {
0 commit comments