1
1
package com .browserstack .client ;
2
2
3
- import java .io .IOException ;
4
- import java .io .InputStream ;
5
- import java .io .Reader ;
6
- import java .lang .reflect .Type ;
7
- import java .nio .charset .Charset ;
8
- import java .util .ArrayList ;
9
- import java .util .Arrays ;
10
- import java .util .List ;
11
- import java .util .Map ;
12
3
import com .browserstack .automate .Automate .BuildStatus ;
13
4
import com .browserstack .automate .exception .BuildNotFound ;
14
5
import com .browserstack .automate .exception .SessionNotFound ;
34
25
import com .google .api .client .http .javanet .NetHttpTransport ;
35
26
import com .google .api .client .util .ObjectParser ;
36
27
28
+ import javax .annotation .Nonnull ;
29
+ import java .io .IOException ;
30
+ import java .io .InputStream ;
31
+ import java .io .Reader ;
32
+ import java .lang .reflect .Type ;
33
+ import java .nio .charset .Charset ;
34
+ import java .util .ArrayList ;
35
+ import java .util .Arrays ;
36
+ import java .util .List ;
37
+ import java .util .Map ;
38
+
37
39
public abstract class BrowserStackClient implements BrowserStackClientInterface {
38
40
private static final String BASE_URL = "https://www.browserstack.com" ;
39
41
private static final String CACHE_KEY_PREFIX_BROWSERS = "browsers" ;
@@ -262,10 +264,11 @@ public enum Product {
262
264
*
263
265
* @param status Return only builds that match the specified build status.
264
266
* @param limit Limit results to the specified count.
267
+ * @param buildName build name to be searched with.
265
268
* @return List of {@link Build} objects.
266
269
* @throws BrowserStackException
267
270
*/
268
- public List <Build > getBuilds (final BuildStatus status , final int limit )
271
+ public List <Build > getBuilds (final BuildStatus status , final int limit , final String buildName )
269
272
throws BrowserStackException {
270
273
BrowserStackRequest httpRequest ;
271
274
try {
@@ -282,7 +285,11 @@ public List<Build> getBuilds(final BuildStatus status, final int limit)
282
285
httpRequest .queryString (Constants .Filter .FILTER , status .name ().toLowerCase ());
283
286
}
284
287
285
- List <BuildNode > buildNodes ;
288
+ if (buildName != null && !buildName .isEmpty ()) {
289
+ httpRequest .queryString (Constants .Filter .BUILD_NAME , buildName );
290
+ }
291
+
292
+ final List <BuildNode > buildNodes ;
286
293
try {
287
294
buildNodes = Arrays .asList (httpRequest .asObject (BuildNode [].class ));
288
295
} catch (BrowserStackException e ) {
@@ -299,6 +306,23 @@ public List<Build> getBuilds(final BuildStatus status, final int limit)
299
306
return builds ;
300
307
}
301
308
309
+ /**
310
+ * Gets the list of builds via build status and the count required
311
+ *
312
+ * <p>
313
+ * A build is an organizational structure for tests.
314
+ * </p>
315
+ *
316
+ * @param status Return only builds that match the specified build status.
317
+ * @param limit Limit results to the specified count.
318
+ * @return List of {@link Build} objects.
319
+ * @throws BrowserStackException
320
+ */
321
+ public List <Build > getBuilds (final BuildStatus status , final int limit )
322
+ throws BrowserStackException {
323
+ return getBuilds (status , limit , null );
324
+ }
325
+
302
326
/**
303
327
* Gets the list of builds.
304
328
*
@@ -368,6 +392,26 @@ public Build getBuild(final String buildId) throws BuildNotFound, BrowserStackEx
368
392
}
369
393
}
370
394
395
+ /**
396
+ * Gets the build identified using the build name.
397
+ *
398
+ * @param buildName Name of the build which will be used for searching
399
+ * @return {@link Build} object
400
+ * @throws BuildNotFound
401
+ * @throws BrowserStackException
402
+ */
403
+ public Build getBuildByName (@ Nonnull final String buildName ) throws BuildNotFound , BrowserStackException {
404
+ try {
405
+ final List <Build > build = getBuilds (null , 1 , buildName );
406
+ if (build .size () == 1 ) {
407
+ return build .get (0 );
408
+ }
409
+ throw new BuildNotFound ("Build not found by name: " + buildName );
410
+ } catch (BrowserStackException e ) {
411
+ throw e ;
412
+ }
413
+ }
414
+
371
415
/**
372
416
* Delete the build identified by the build identifier.
373
417
*
@@ -389,6 +433,7 @@ public boolean deleteBuild(final String buildId) throws BrowserStackException {
389
433
390
434
/**
391
435
* Retrieves the list of sessions existing under a specific build.
436
+ * If no limit is specified, all the sessions will be fetched from that build
392
437
*
393
438
* @param buildId ID that uniquely identifies a build.
394
439
* @param status Include only builds that match the specified build status.
@@ -400,39 +445,64 @@ public boolean deleteBuild(final String buildId) throws BrowserStackException {
400
445
public List <Session > getSessions (final String buildId , final BuildStatus status , final int limit )
401
446
throws BuildNotFound , BrowserStackException {
402
447
403
- BrowserStackRequest httpRequest = null ;
448
+ // validation of the limit field. Default will be set to 1000 if 0 is provided
449
+ final int totalLimit =
450
+ (limit <= 0 || limit > Constants .Filter .MAX_SESSIONS )
451
+ ? Constants .Filter .MAX_SESSIONS
452
+ : limit ;
453
+ int totalRequests = totalLimit /Constants .Filter .MAX_LIMIT ;
454
+
455
+ // An extra request to fetch the remainder sessions
456
+ if ((totalLimit % Constants .Filter .MAX_LIMIT ) > 0 ) {
457
+ totalRequests ++;
458
+ }
459
+
460
+ final List <Session > sessions = new ArrayList <Session >();
461
+
462
+ // currReq will act as offset to fetch all* sessions from the build
463
+ for (int currReq = 0 ; currReq < totalRequests ; currReq ++) {
464
+ final List <SessionNode > sessionNodes = getSessionNodes (buildId , status , totalLimit , currReq * Constants .Filter .MAX_LIMIT );
465
+
466
+ for (SessionNode sessionNode : sessionNodes ) {
467
+ if (sessionNode != null && sessionNode .getSession () != null ) {
468
+ sessions .add (sessionNode .getSession ().<Session >setClient (this ));
469
+ }
470
+ }
471
+
472
+ // break the loop since there are no more sessions left to fetch
473
+ if (sessionNodes .size () < Constants .Filter .MAX_LIMIT ) {
474
+ break ;
475
+ }
476
+ }
477
+
478
+ return sessions ;
479
+ }
480
+
481
+ private List <SessionNode > getSessionNodes (String buildId , BuildStatus status , int totalLimit , int offset ) throws BrowserStackException {
482
+ BrowserStackRequest httpRequest ;
404
483
try {
405
484
httpRequest =
406
- newRequest (Method .GET , "/builds/{buildId}/sessions.json" ).routeParam ("buildId" , buildId );
485
+ newRequest (Method .GET , "/builds/{buildId}/sessions.json" ).routeParam ("buildId" , buildId );
407
486
} catch (BrowserStackException e ) {
408
487
throw e ;
409
488
}
410
489
411
- if (limit > 0 ) {
412
- httpRequest .queryString (Constants .Filter .LIMIT , limit );
413
- }
490
+ httpRequest .queryString (Constants .Filter .LIMIT , totalLimit );
491
+ httpRequest .queryString (Constants .Filter .OFFSET , offset );
414
492
415
493
if (status != null ) {
416
494
httpRequest .queryString (Constants .Filter .FILTER , status );
417
495
}
418
496
419
- List <SessionNode > sessionNodes ;
497
+ final List <SessionNode > sessionNodes ;
420
498
try {
421
499
sessionNodes = Arrays .asList (httpRequest .asObject (SessionNode [].class ));
422
500
} catch (BrowserStackObjectNotFound e ) {
423
501
throw new BuildNotFound ("Build not found: " + buildId );
424
502
} catch (BrowserStackException e ) {
425
503
throw e ;
426
504
}
427
-
428
- List <Session > sessions = new ArrayList <Session >();
429
- for (SessionNode sessionNode : sessionNodes ) {
430
- if (sessionNode != null && sessionNode .getSession () != null ) {
431
- sessions .add (sessionNode .getSession ().<Session >setClient (this ));
432
- }
433
- }
434
-
435
- return sessions ;
505
+ return sessionNodes ;
436
506
}
437
507
438
508
/**
0 commit comments