API.pm 78 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972
  1. # ABSTRACT: Provides an interface to TestRail's REST api via HTTP
  2. # PODNAME: TestRail::API
  3. package TestRail::API;
  4. =head1 SYNOPSIS
  5. use TestRail::API;
  6. my ($username,$password,$host) = ('foo','bar','http://testrail.baz.foo');
  7. my $tr = TestRail::API->new($host, $username, $password);
  8. =head1 DESCRIPTION
  9. C<TestRail::API> provides methods to access an existing TestRail account using API v2. You can then do things like look up tests, set statuses and create runs from lists of cases.
  10. It is by no means exhaustively implementing every TestRail API function.
  11. =head1 IMPORTANT
  12. All the methods aside from the constructor should not die, but return a false value upon failure (see exceptions below).
  13. When the server is not responsive, expect a -500 response, and retry accordingly by setting the num_tries parameter in the constructor.
  14. Also, all *ByName methods are vulnerable to duplicate naming issues. Try not to use the same name for:
  15. * projects
  16. * testsuites within the same project
  17. * sections within the same testsuite that are peers
  18. * test cases
  19. * test plans and runs outside of plans which are not completed
  20. * configurations
  21. To do so will result in the first of said item found being returned rather than an array of possibilities to choose from.
  22. There are two exceptions to this, in the case of 401 and 403 responses, as these failing generally mean your program has no chance of success anyways.
  23. =cut
  24. use 5.010;
  25. use strict;
  26. use warnings;
  27. use Carp qw{cluck confess};
  28. use Scalar::Util qw{reftype looks_like_number};
  29. use Clone 'clone';
  30. use Try::Tiny;
  31. use Types::Standard qw( slurpy ClassName Object Str Int Bool HashRef ArrayRef Maybe Optional);
  32. use Type::Params qw( compile );
  33. use JSON::MaybeXS 1.001000 ();
  34. use HTTP::Request;
  35. use LWP::UserAgent;
  36. use HTTP::CookieJar::LWP;
  37. use Data::Validate::URI qw{is_uri};
  38. use List::Util 1.33;
  39. use Encode ();
  40. =head1 CONSTRUCTOR
  41. =head2 B<new (api_url, user, password, encoding, debug, do_post_redirect)>
  42. Creates new C<TestRail::API> object.
  43. =over 4
  44. =item STRING C<API URL> - base url for your TestRail api server.
  45. =item STRING C<USER> - Your TestRail User.
  46. =item STRING C<PASSWORD> - Your TestRail password, or a valid API key (TestRail 4.2 and above).
  47. =item STRING C<ENCODING> - The character encoding used by the caller. Defaults to 'UTF-8', see L<Encode::Supported> and for supported encodings.
  48. =item BOOLEAN C<DEBUG> (optional) - Print the JSON responses from TestRail with your requests. Default false.
  49. =item BOOLEAN C<DO_POST_REDIRECT> (optional) - Follow redirects on POST requests (most add/edit/delete calls are POSTs). Default false.
  50. =item INTEGER C<MAX_TRIES> (optional) - Try requests up to X number of times if they fail with anything other than 401/403. Useful with flaky external authenticators, or timeout issues. Default 1.
  51. =item HASHREF C<USER_FETCH_OPTS> (optional) - Options relating to getUsers call done during new:
  52. =over 4
  53. =item BOOLEAN C<skip_userfetch> - Skip fetching all TR users during construction. Default false.
  54. This will save you some time on servers with quite a few users, especially if you don't
  55. particularly have a need to know about things related to TR users themselves.
  56. If you do need this info, you don't really save any time, however, as it will fetch them
  57. in the relevant subroutines that need this information.
  58. Also, on newer versions of TestRail, user fetching is not possible unless you either:
  59. * Are an administrator on the server
  60. * Provide the project_id (https://www.gurock.com/testrail/docs/api/reference/users)
  61. =item STRING C<project_id> - String or number corresponding to a project ID to use when fetching users.
  62. =back
  63. =back
  64. Returns C<TestRail::API> object if login is successful.
  65. my $tr = TestRail::API->new('http://tr.test/testrail', 'moo','M000000!');
  66. Dies on all communication errors with the TestRail server.
  67. Does not do above checks if debug is passed.
  68. =cut
  69. sub new {
  70. state $check = compile(ClassName, Str, Str, Str, Optional[Maybe[Str]], Optional[Maybe[Bool]], Optional[Maybe[Bool]], Optional[Maybe[Int]],Optional[Maybe[HashRef]]);
  71. my ($class,$apiurl,$user,$pass,$encoding,$debug, $do_post_redirect,$max_tries,$userfetch_opts) = $check->(@_);
  72. die("Invalid URI passed to constructor") if !is_uri($apiurl);
  73. $debug //= 0;
  74. my $self = {
  75. user => $user,
  76. pass => $pass,
  77. apiurl => $apiurl,
  78. debug => $debug,
  79. encoding => $encoding || 'UTF-8',
  80. testtree => [],
  81. flattree => [],
  82. user_cache => [],
  83. configurations => {},
  84. tr_fields => undef,
  85. tr_project_id => $userfetch_opts->{'project_id'},
  86. default_request => undef,
  87. global_limit => 250, #Discovered by experimentation
  88. browser => LWP::UserAgent->new(
  89. keep_alive => 10,
  90. cookie_jar => HTTP::CookieJar::LWP->new(),
  91. ),
  92. do_post_redirect => $do_post_redirect,
  93. max_tries => $max_tries // 1,
  94. retry_delay => 5,
  95. };
  96. #Allow POST redirects
  97. if ($self->{do_post_redirect}) {
  98. push @{ $self->{'browser'}->requests_redirectable }, 'POST';
  99. }
  100. #Check chara encoding
  101. $self->{'encoding-nonaliased'} = Encode::resolve_alias($self->{'encoding'});
  102. die("Invalid encoding alias '".$self->{'encoding'}."' passed, see Encoding::Supported for a list of allowed encodings")
  103. unless $self->{'encoding-nonaliased'};
  104. die("Invalid encoding '".$self->{'encoding-nonaliased'}."' passed, see Encoding::Supported for a list of allowed encodings")
  105. unless grep {$_ eq $self->{'encoding-nonaliased'}} (Encode->encodings(":all"));
  106. #Create default request to pass on to LWP::UserAgent
  107. $self->{'default_request'} = HTTP::Request->new();
  108. $self->{'default_request'}->authorization_basic($user,$pass);
  109. bless( $self, $class );
  110. return $self if $self->debug; #For easy class testing without mocks
  111. # Manually do the get_users call to check HTTP status...
  112. # Allow users to skip the check if you have a zillion users etc,
  113. # as apparently that is fairly taxing on TR itself.
  114. if( !$userfetch_opts->{skip_usercache} ) {
  115. my $res = $self->getUsers($userfetch_opts->{project_id});
  116. confess "Error: network unreachable" if !defined($res);
  117. if ( (reftype($res) || 'undef') ne 'ARRAY') {
  118. confess "Unexpected return from _doRequest: $res" if !looks_like_number($res);
  119. confess "Could not communicate with TestRail Server! Check that your URI is correct, and your TestRail installation is functioning correctly." if $res == -500;
  120. confess "Could not list testRail users! Check that your TestRail installation has it's API enabled, and your credentials are correct" if $res == -403;
  121. confess "Bad user credentials!" if $res == -401;
  122. confess "HTTP error $res encountered while communicating with TestRail server. Resolve issue and try again." if $res < 0;
  123. confess "Unknown error occurred: $res";
  124. }
  125. confess "No users detected on TestRail Install! Check that your API is functioning correctly." if !scalar(@$res);
  126. }
  127. return $self;
  128. }
  129. =head1 GETTERS
  130. =head2 B<apiurl>
  131. =head2 B<debug>
  132. Accessors for these parameters you pass into the constructor, in case you forget.
  133. =cut
  134. sub apiurl {
  135. state $check = compile(Object);
  136. my ($self) = $check->(@_);
  137. return $self->{'apiurl'}
  138. }
  139. sub debug {
  140. state $check = compile(Object);
  141. my ($self) = $check->(@_);
  142. return $self->{'debug'};
  143. }
  144. =head2 B<retry_delay>
  145. There is no getter/setter for this parameter, but it is worth mentioning.
  146. This is the number of seconds to wait between failed request retries when max_retries > 1.
  147. #Do something other than the default of 5s, like spam the server mercilessly
  148. $tr->{retry_delay} = 0;
  149. ...
  150. =cut
  151. #Convenient JSON-HTTP fetcher
  152. sub _doRequest {
  153. state $check = compile(Object, Str, Optional[Maybe[Str]], Optional[Maybe[HashRef]]);
  154. my ($self,$path,$method,$data) = $check->(@_);
  155. $self->{num_tries}++;
  156. my $req = clone $self->{'default_request'};
  157. $method //= 'GET';
  158. $req->method($method);
  159. $req->url($self->apiurl.'/'.$path);
  160. warn "$method ".$self->apiurl."/$path" if $self->debug;
  161. my $coder = JSON::MaybeXS->new;
  162. #Data sent is JSON, and encoded per user preference
  163. my $content = $data ? Encode::encode( $self->{'encoding-nonaliased'}, $coder->encode($data) ) : '';
  164. $req->content($content);
  165. $req->header( "Content-Type" => "application/json; charset=".$self->{'encoding'} );
  166. my $response = eval { $self->{'browser'}->request($req) };
  167. #Uncomment to generate mocks
  168. #use Data::Dumper;
  169. #open(my $fh, '>>', 'mock.out');
  170. #print $fh "{\n\n";
  171. #print $fh Dumper($path,'200','OK',$response->headers,$response->content);
  172. #print $fh '$mockObject->map_response(qr/\Q$VAR1\E/,HTTP::Response->new($VAR2, $VAR3, $VAR4, $VAR5));';
  173. #print $fh "\n\n}\n\n";
  174. #close $fh;
  175. if ($@) {
  176. #LWP threw an ex, probably a timeout
  177. if ($self->{num_tries} >= $self->{max_tries}) {
  178. $self->{num_tries} = 0;
  179. confess "Failed to satisfy request after $self->{num_tries} tries!";
  180. }
  181. cluck "WARNING: TestRail API request failed due to timeout, or other LWP fatal condition, re-trying request...\n";
  182. sleep $self->{retry_delay} if $self->{retry_delay};
  183. goto &_doRequest;
  184. }
  185. return $response if !defined($response); #worst case
  186. if ($response->code == 403) {
  187. confess "ERROR 403: Access Denied: ".$response->content;
  188. }
  189. if ($response->code == 401) {
  190. confess "ERROR 401: Authentication failed: ".$response->content;
  191. }
  192. if ($response->code != 200) {
  193. #LWP threw an ex, probably a timeout
  194. if ($self->{num_tries} >= $self->{max_tries}) {
  195. $self->{num_tries} = 0;
  196. cluck "ERROR: Arguments Bad? (got code ".$response->code."): ".$response->content;
  197. return -int($response->code);
  198. }
  199. cluck "WARNING: TestRail API request failed (got code ".$response->code."), re-trying request...\n";
  200. sleep $self->{retry_delay} if $self->{retry_delay};
  201. goto &_doRequest;
  202. }
  203. $self->{num_tries} = 0;
  204. try {
  205. return $coder->decode($response->content);
  206. } catch {
  207. if ($response->code == 200 && !$response->content) {
  208. return 1; #This function probably just returns no data
  209. } else {
  210. cluck "ERROR: Malformed JSON returned by API.";
  211. cluck $@;
  212. if (!$self->debug) { #Otherwise we've already printed this, but we need to know if we encounter this
  213. cluck "RAW CONTENT:";
  214. cluck $response->content
  215. }
  216. return 0;
  217. }
  218. }
  219. }
  220. =head1 USER METHODS
  221. =head2 B<getUsers ()>
  222. Get all the user definitions for the provided Test Rail install.
  223. Returns ARRAYREF of user definition HASHREFs.
  224. =cut
  225. sub getUsers {
  226. state $check = compile(Object,Optional[Maybe[Str]]);
  227. my ($self,$project_id) = $check->(@_);
  228. # Return shallow clone of user_cache if set.
  229. return [ @{ $self->{'user_cache'} } ] if ref $self->{'user_cache'} eq 'ARRAY' && scalar(@{$self->{'user_cache'}});
  230. my $maybe_project = $project_id ? "/$project_id" : '';
  231. my $res = $self->_doRequest("index.php?/api/v2/get_users$maybe_project");
  232. return -500 if !$res || (reftype($res) || 'undef') ne 'ARRAY';
  233. $self->{'user_cache'} = $res;
  234. return clone($res);
  235. }
  236. =head2 B<getUserByID(id)>
  237. =cut
  238. =head2 B<getUserByName(name)>
  239. =cut
  240. =head2 B<getUserByEmail(email)>
  241. Get user definition hash by ID, Name or Email.
  242. Returns user definition HASHREF.
  243. For efficiency's sake, these methods cache the result of getUsers until you explicitly run it again.
  244. =cut
  245. sub getUserByID {
  246. state $check = compile(Object, Int);
  247. my ($self,$user) = $check->(@_);
  248. my $users = $self->getUsers();
  249. return $users if ref $users ne 'ARRAY';
  250. foreach my $usr (@$users) {
  251. return $usr if $usr->{'id'} == $user;
  252. }
  253. return 0;
  254. }
  255. sub getUserByName {
  256. state $check = compile(Object, Str);
  257. my ($self,$user) = $check->(@_);
  258. my $users = $self->getUsers();
  259. return $users if ref $users ne 'ARRAY';
  260. foreach my $usr (@$users) {
  261. return $usr if $usr->{'name'} eq $user;
  262. }
  263. return 0;
  264. }
  265. sub getUserByEmail {
  266. state $check = compile(Object, Str);
  267. my ($self,$email) = $check->(@_);
  268. my $users = $self->getUsers();
  269. return $users if ref $users ne 'ARRAY';
  270. foreach my $usr (@$users) {
  271. return $usr if $usr->{'email'} eq $email;
  272. }
  273. return 0;
  274. }
  275. =head2 userNamesToIds(names)
  276. Convenience method to translate a list of user names to TestRail user IDs.
  277. =over 4
  278. =item ARRAY C<NAMES> - Array of user names to translate to IDs.
  279. =back
  280. Returns ARRAY of user IDs.
  281. Throws an exception in the case of one (or more) of the names not corresponding to a valid username.
  282. =cut
  283. sub userNamesToIds {
  284. state $check = compile(Object, slurpy ArrayRef[Str]);
  285. my ($self,$names) = $check->(@_);
  286. confess("At least one user name must be provided") if !scalar(@$names);
  287. my @ret = grep {defined $_} map {my $user = $_; my @list = grep {$user->{'name'} eq $_} @$names; scalar(@list) ? $user->{'id'} : undef} @{$self->getUsers()};
  288. confess("One or more user names provided does not exist in TestRail.") unless scalar(@$names) == scalar(@ret);
  289. return @ret;
  290. };
  291. =head1 PROJECT METHODS
  292. =head2 B<createProject (name, [description,send_announcement])>
  293. Creates new Project (Database of testsuites/tests).
  294. Optionally specify an announcement to go out to the users.
  295. Requires TestRail admin login.
  296. =over 4
  297. =item STRING C<NAME> - Desired name of project.
  298. =item STRING C<DESCRIPTION> (optional) - Description of project. Default value is 'res ipsa loquiter'.
  299. =item BOOLEAN C<SEND ANNOUNCEMENT> (optional) - Whether to confront users with an announcement about your awesome project on next login. Default false.
  300. =back
  301. Returns project definition HASHREF on success, false otherwise.
  302. $tl->createProject('Widgetronic 4000', 'Tests for the whiz-bang new product', true);
  303. =cut
  304. sub createProject {
  305. state $check = compile(Object, Str, Optional[Maybe[Str]], Optional[Maybe[Bool]]);
  306. my ($self,$name,$desc,$announce) = $check->(@_);
  307. $desc //= 'res ipsa loquiter';
  308. $announce //= 0;
  309. my $input = {
  310. name => $name,
  311. announcement => $desc,
  312. show_announcement => $announce
  313. };
  314. return $self->_doRequest('index.php?/api/v2/add_project','POST',$input);
  315. }
  316. =head2 B<deleteProject (id)>
  317. Deletes specified project by ID.
  318. Requires TestRail admin login.
  319. =over 4
  320. =item STRING C<NAME> - Desired name of project.
  321. =back
  322. Returns BOOLEAN.
  323. $success = $tl->deleteProject(1);
  324. =cut
  325. sub deleteProject {
  326. state $check = compile(Object, Int);
  327. my ($self,$proj) = $check->(@_);
  328. return $self->_doRequest('index.php?/api/v2/delete_project/'.$proj,'POST');
  329. }
  330. =head2 B<getProjects (filters)>
  331. Get all available projects
  332. =over 4
  333. =item HASHREF C<FILTERS> (optional) - HASHREF describing parameters to filter cases by.
  334. =back
  335. Returns array of project definition HASHREFs, false otherwise.
  336. $projects = $tl->getProjects;
  337. See:
  338. L<https://www.gurock.com/testrail/docs/api/reference/projects#getprojects>
  339. for details as to the allowable filter keys.
  340. =cut
  341. sub getProjects {
  342. state $check = compile(Object,Optional[Maybe[HashRef]]);
  343. my ($self,$filters) = $check->(@_);
  344. my $result = $self->_doRequest('index.php?/api/v2/get_projects' . _convert_filters_to_string($filters) );
  345. #Save state for future use, if needed
  346. return -500 if !$result || (reftype($result) || 'undef') ne 'ARRAY';
  347. $self->{'testtree'} = $result;
  348. #Note that it's a project for future reference by recursive tree search
  349. return -500 if !$result || (reftype($result) || 'undef') ne 'ARRAY';
  350. foreach my $pj (@{$result}) {
  351. $pj->{'type'} = 'project';
  352. }
  353. return $result;
  354. }
  355. =head2 B<getProjectByName ($project)>
  356. Gets some project definition hash by it's name
  357. =over 4
  358. =item STRING C<PROJECT> - desired project
  359. =back
  360. Returns desired project definition HASHREF, false otherwise.
  361. $project = $tl->getProjectByName('FunProject');
  362. =cut
  363. sub getProjectByName {
  364. state $check = compile(Object, Str);
  365. my ($self,$project) = $check->(@_);
  366. #See if we already have the project list...
  367. my $projects = $self->{'testtree'};
  368. return -500 if !$projects || (reftype($projects) || 'undef') ne 'ARRAY';
  369. $projects = $self->getProjects() unless scalar(@$projects);
  370. #Search project list for project
  371. return -500 if !$projects || (reftype($projects) || 'undef') ne 'ARRAY';
  372. for my $candidate (@$projects) {
  373. return $candidate if ($candidate->{'name'} eq $project);
  374. }
  375. return 0;
  376. }
  377. =head2 B<getProjectByID ($project)>
  378. Gets some project definition hash by it's ID
  379. =over 4
  380. =item INTEGER C<PROJECT> - desired project
  381. =back
  382. Returns desired project definition HASHREF, false otherwise.
  383. $projects = $tl->getProjectByID(222);
  384. =cut
  385. sub getProjectByID {
  386. state $check = compile(Object, Int);
  387. my ($self,$project) = $check->(@_);
  388. #See if we already have the project list...
  389. my $projects = $self->{'testtree'};
  390. $projects = $self->getProjects() unless scalar(@$projects);
  391. #Search project list for project
  392. return -500 if !$projects || (reftype($projects) || 'undef') ne 'ARRAY';
  393. for my $candidate (@$projects) {
  394. return $candidate if ($candidate->{'id'} eq $project);
  395. }
  396. return 0;
  397. }
  398. =head1 TESTSUITE METHODS
  399. =head2 B<createTestSuite (project_id, name, [description])>
  400. Creates new TestSuite (folder of tests) in the database of test specifications under given project id having given name and details.
  401. =over 4
  402. =item INTEGER C<PROJECT ID> - ID of project this test suite should be under.
  403. =item STRING C<NAME> - Desired name of test suite.
  404. =item STRING C<DESCRIPTION> (optional) - Description of test suite. Default value is 'res ipsa loquiter'.
  405. =back
  406. Returns TS definition HASHREF on success, false otherwise.
  407. $tl->createTestSuite(1, 'broken tests', 'Tests that should be reviewed');
  408. =cut
  409. sub createTestSuite {
  410. state $check = compile(Object, Int, Str, Optional[Maybe[Str]]);
  411. my ($self,$project_id,$name,$details) = $check->(@_);
  412. $details //= 'res ipsa loquiter';
  413. my $input = {
  414. name => $name,
  415. description => $details
  416. };
  417. return $self->_doRequest('index.php?/api/v2/add_suite/'.$project_id,'POST',$input);
  418. }
  419. =head2 B<deleteTestSuite (suite_id)>
  420. Deletes specified testsuite.
  421. =over 4
  422. =item INTEGER C<SUITE ID> - ID of testsuite to delete.
  423. =back
  424. Returns BOOLEAN.
  425. $tl->deleteTestSuite(1);
  426. =cut
  427. sub deleteTestSuite {
  428. state $check = compile(Object, Int);
  429. my ($self,$suite_id) = $check->(@_);
  430. return $self->_doRequest('index.php?/api/v2/delete_suite/'.$suite_id,'POST');
  431. }
  432. =head2 B<getTestSuites (project_id)>
  433. Gets the testsuites for a project
  434. =over 4
  435. =item STRING C<PROJECT ID> - desired project's ID
  436. =back
  437. Returns ARRAYREF of testsuite definition HASHREFs, 0 on error.
  438. $suites = $tl->getTestSuites(123);
  439. =cut
  440. sub getTestSuites {
  441. state $check = compile(Object, Int);
  442. my ($self,$proj) = $check->(@_);
  443. return $self->_doRequest('index.php?/api/v2/get_suites/'.$proj);
  444. }
  445. =head2 B<getTestSuiteByName (project_id,testsuite_name)>
  446. Gets the testsuite that matches the given name inside of given project.
  447. =over 4
  448. =item STRING C<PROJECT ID> - ID of project holding this testsuite
  449. =item STRING C<TESTSUITE NAME> - desired parent testsuite name
  450. =back
  451. Returns desired testsuite definition HASHREF, false otherwise.
  452. $suites = $tl->getTestSuitesByName(321, 'hugSuite');
  453. =cut
  454. sub getTestSuiteByName {
  455. state $check = compile(Object, Int, Str);
  456. my ($self,$project_id,$testsuite_name) = $check->(@_);
  457. #TODO cache
  458. my $suites = $self->getTestSuites($project_id);
  459. return -500 if !$suites || (reftype($suites) || 'undef') ne 'ARRAY'; #No suites for project, or no project
  460. foreach my $suite (@$suites) {
  461. return $suite if $suite->{'name'} eq $testsuite_name;
  462. }
  463. return 0; #Couldn't find it
  464. }
  465. =head2 B<getTestSuiteByID (testsuite_id)>
  466. Gets the testsuite with the given ID.
  467. =over 4
  468. =item STRING C<TESTSUITE_ID> - TestSuite ID.
  469. =back
  470. Returns desired testsuite definition HASHREF, false otherwise.
  471. $tests = $tl->getTestSuiteByID(123);
  472. =cut
  473. sub getTestSuiteByID {
  474. state $check = compile(Object, Int);
  475. my ($self,$testsuite_id) = $check->(@_);
  476. return $self->_doRequest('index.php?/api/v2/get_suite/'.$testsuite_id);
  477. }
  478. =head1 SECTION METHODS
  479. =head2 B<createSection(project_id,suite_id,name,[parent_id])>
  480. Creates a section.
  481. =over 4
  482. =item INTEGER C<PROJECT ID> - Parent Project ID.
  483. =item INTEGER C<SUITE ID> - Parent TestSuite ID.
  484. =item STRING C<NAME> - desired section name.
  485. =item INTEGER C<PARENT ID> (optional) - parent section id
  486. =back
  487. Returns new section definition HASHREF, false otherwise.
  488. $section = $tr->createSection(1,1,'nugs',1);
  489. =cut
  490. sub createSection {
  491. state $check = compile(Object, Int, Int, Str, Optional[Maybe[Int]]);
  492. my ($self,$project_id,$suite_id,$name,$parent_id) = $check->(@_);
  493. my $input = {
  494. name => $name,
  495. suite_id => $suite_id
  496. };
  497. $input->{'parent_id'} = $parent_id if $parent_id;
  498. return $self->_doRequest('index.php?/api/v2/add_section/'.$project_id,'POST',$input);
  499. }
  500. =head2 B<deleteSection (section_id)>
  501. Deletes specified section.
  502. =over 4
  503. =item INTEGER C<SECTION ID> - ID of section to delete.
  504. =back
  505. Returns BOOLEAN.
  506. $tr->deleteSection(1);
  507. =cut
  508. sub deleteSection {
  509. state $check = compile(Object, Int);
  510. my ($self,$section_id) = $check->(@_);
  511. return $self->_doRequest('index.php?/api/v2/delete_section/'.$section_id,'POST');
  512. }
  513. =head2 B<getSections (project_id,suite_id)>
  514. Gets sections for a given project and suite.
  515. =over 4
  516. =item INTEGER C<PROJECT ID> - ID of parent project.
  517. =item INTEGER C<SUITE ID> - ID of suite to get sections for.
  518. =back
  519. Returns ARRAYREF of section definition HASHREFs.
  520. $tr->getSections(1,2);
  521. =cut
  522. sub getSections {
  523. state $check = compile(Object, Int, Int);
  524. my ($self,$project_id,$suite_id) = $check->(@_);
  525. #Cache sections to reduce requests in tight loops
  526. return $self->{'sections'}->{$suite_id} if $self->{'sections'}->{$suite_id};
  527. $self->{'sections'}->{$suite_id} = $self->_doRequest("index.php?/api/v2/get_sections/$project_id&suite_id=$suite_id");
  528. return $self->{'sections'}->{$suite_id};
  529. }
  530. =head2 B<getSectionByID (section_id)>
  531. Gets desired section.
  532. =over 4
  533. =item INTEGER C<PROJECT ID> - ID of parent project.
  534. =item INTEGER C<SUITE ID> - ID of suite to get sections for.
  535. =back
  536. Returns section definition HASHREF.
  537. $tr->getSectionByID(344);
  538. =cut
  539. sub getSectionByID {
  540. state $check = compile(Object, Int);
  541. my ($self,$section_id) = $check->(@_);
  542. return $self->_doRequest("index.php?/api/v2/get_section/$section_id");
  543. }
  544. =head2 B<getSectionByName (project_id,suite_id,name)>
  545. Gets desired section.
  546. =over 4
  547. =item INTEGER C<PROJECT ID> - ID of parent project.
  548. =item INTEGER C<SUITE ID> - ID of suite to get section for.
  549. =item STRING C<NAME> - name of section to get
  550. =back
  551. Returns section definition HASHREF.
  552. $tr->getSectionByName(1,2,'nugs');
  553. =cut
  554. sub getSectionByName {
  555. state $check = compile(Object, Int, Int, Str);
  556. my ($self,$project_id,$suite_id,$section_name) = $check->(@_);
  557. my $sections = $self->getSections($project_id,$suite_id);
  558. return -500 if !$sections || (reftype($sections) || 'undef') ne 'ARRAY';
  559. foreach my $sec (@$sections) {
  560. return $sec if $sec->{'name'} eq $section_name;
  561. }
  562. return 0;
  563. }
  564. =head2 B<getChildSections ($project_id, section)>
  565. Gets desired section's child sections.
  566. =over 4
  567. =item INTEGER C<PROJECT_ID> - parent project ID of section.
  568. =item HASHREF C<SECTION> - section definition HASHREF.
  569. =back
  570. Returns ARRAYREF of section definition HASHREF. ARRAYREF is empty if there are none.
  571. Recursively searches for children, so the children of child sections will be returned as well.
  572. $tr->getChildSections($section);
  573. =cut
  574. sub getChildSections {
  575. state $check = compile(Object, Int, HashRef);
  576. my ($self, $project_id, $section) = $check->(@_);
  577. my $sections_orig = $self->getSections($project_id,$section->{suite_id});
  578. return [] if !$sections_orig || (reftype($sections_orig) || 'undef') ne 'ARRAY';
  579. my @sections = grep { $_->{'parent_id'} ? $_->{'parent_id'} == $section->{'id'} : 0 } @$sections_orig;
  580. foreach my $sec (@sections) {
  581. push(@sections, grep { $_->{'parent_id'} ? $_->{'parent_id'} == $sec->{'id'} : 0 } @$sections_orig);
  582. }
  583. return \@sections;
  584. }
  585. =head2 sectionNamesToIds(project_id,suite_id,names)
  586. Convenience method to translate a list of section names to TestRail section IDs.
  587. =over 4
  588. =item INTEGER C<PROJECT ID> - ID of parent project.
  589. =item INTEGER C<SUITE ID> - ID of parent suite.
  590. =item ARRAY C<NAMES> - Array of section names to translate to IDs.
  591. =back
  592. Returns ARRAY of section IDs.
  593. Throws an exception in the case of one (or more) of the names not corresponding to a valid section name.
  594. =cut
  595. sub sectionNamesToIds {
  596. my ($self,$project_id,$suite_id,@names) = @_;
  597. my $sections = $self->getSections($project_id,$suite_id) or confess("Could not find sections in provided project/suite.");
  598. return _X_in_my_Y($self,$sections,'id',@names);
  599. }
  600. =head1 CASE METHODS
  601. =head2 B<getCaseTypes ()>
  602. Gets possible case types.
  603. Returns ARRAYREF of case type definition HASHREFs.
  604. $tr->getCaseTypes();
  605. =cut
  606. sub getCaseTypes {
  607. state $check = compile(Object);
  608. my ($self) = $check->(@_);
  609. return clone($self->{'type_cache'}) if defined($self->{'type_cache'});
  610. my $types = $self->_doRequest("index.php?/api/v2/get_case_types");
  611. return -500 if !$types || (reftype($types) || 'undef') ne 'ARRAY';
  612. $self->{'type_cache'} = $types;
  613. return clone $types;
  614. }
  615. =head2 B<getCaseTypeByName (name)>
  616. Gets case type by name.
  617. =over 4
  618. =item STRING C<NAME> - Name of desired case type
  619. =back
  620. Returns case type definition HASHREF.
  621. Dies if named case type does not exist.
  622. $tr->getCaseTypeByName();
  623. =cut
  624. sub getCaseTypeByName {
  625. state $check = compile(Object, Str);
  626. my ($self,$name) = $check->(@_);
  627. my $types = $self->getCaseTypes();
  628. return -500 if !$types || (reftype($types) || 'undef') ne 'ARRAY';
  629. foreach my $type (@$types) {
  630. return $type if $type->{'name'} eq $name;
  631. }
  632. confess("No such case type '$name'!");
  633. }
  634. =head2 typeNamesToIds(names)
  635. Convenience method to translate a list of case type names to TestRail case type IDs.
  636. =over 4
  637. =item ARRAY C<NAMES> - Array of status names to translate to IDs.
  638. =back
  639. Returns ARRAY of type IDs in the same order as the type names passed.
  640. Throws an exception in the case of one (or more) of the names not corresponding to a valid case type.
  641. =cut
  642. sub typeNamesToIds {
  643. my ($self,@names) = @_;
  644. return _X_in_my_Y($self,$self->getCaseTypes(),'id',@names);
  645. };
  646. =head2 B<createCase(section_id,title,type_id,options,extra_options)>
  647. Creates a test case.
  648. =over 4
  649. =item INTEGER C<SECTION ID> - Parent Section ID.
  650. =item STRING C<TITLE> - Case title.
  651. =item INTEGER C<TYPE_ID> (optional) - desired test type's ID. Defaults to whatever your TR install considers the default type.
  652. =item HASHREF C<OPTIONS> (optional) - Custom fields in the case are the keys, set to the values provided. See TestRail API documentation for more info.
  653. =item HASHREF C<EXTRA OPTIONS> (optional) - contains priority_id, estimate, milestone_id and refs as possible keys. See TestRail API documentation for more info.
  654. =back
  655. Returns new case definition HASHREF, false otherwise.
  656. $custom_opts = {
  657. preconds => "Test harness installed",
  658. steps => "Do the needful",
  659. expected => "cubicle environment transforms into Dali painting"
  660. };
  661. $other_opts = {
  662. priority_id => 4,
  663. milestone_id => 666,
  664. estimate => '2m 45s',
  665. refs => ['TRACE-22','ON-166'] #ARRAYREF of bug IDs.
  666. }
  667. $case = $tr->createCase(1,'Do some stuff',3,$custom_opts,$other_opts);
  668. =cut
  669. sub createCase {
  670. state $check = compile(Object, Int, Str, Optional[Maybe[Int]], Optional[Maybe[HashRef]], Optional[Maybe[HashRef]]);
  671. my ($self,$section_id,$title,$type_id,$opts,$extras) = $check->(@_);
  672. my $stuff = {
  673. title => $title,
  674. type_id => $type_id
  675. };
  676. #Handle sort of optional but baked in options
  677. if (defined($extras) && reftype($extras) eq 'HASH') {
  678. $stuff->{'priority_id'} = $extras->{'priority_id'} if defined($extras->{'priority_id'});
  679. $stuff->{'estimate'} = $extras->{'estimate'} if defined($extras->{'estimate'});
  680. $stuff->{'milestone_id'} = $extras->{'milestone_id'} if defined($extras->{'milestone_id'});
  681. $stuff->{'refs'} = join(',',@{$extras->{'refs'}}) if defined($extras->{'refs'});
  682. }
  683. #Handle custom fields
  684. if (defined($opts) && reftype($opts) eq 'HASH') {
  685. foreach my $key (keys(%$opts)) {
  686. $stuff->{"custom_$key"} = $opts->{$key};
  687. }
  688. }
  689. return $self->_doRequest("index.php?/api/v2/add_case/$section_id",'POST',$stuff);
  690. }
  691. =head2 B<updateCase(case_id,options)>
  692. Updates a test case.
  693. =over 4
  694. =item INTEGER C<CASE ID> - Case ID.
  695. =item HASHREF C<OPTIONS> - Various things about a case to set. Everything except section_id in the output of getCaseBy* methods is a valid input here.
  696. =back
  697. Returns new case definition HASHREF, false otherwise.
  698. =cut
  699. sub updateCase {
  700. state $check = compile(Object, Int, Optional[Maybe[HashRef]]);
  701. my ($self,$case_id,$options) = $check->(@_);
  702. return $self->_doRequest("index.php?/api/v2/update_case/$case_id",'POST',$options);
  703. }
  704. =head2 B<deleteCase (case_id)>
  705. Deletes specified test case.
  706. =over 4
  707. =item INTEGER C<CASE ID> - ID of case to delete.
  708. =back
  709. Returns BOOLEAN.
  710. $tr->deleteCase(1324);
  711. =cut
  712. sub deleteCase {
  713. state $check = compile(Object, Int);
  714. my ($self,$case_id) = $check->(@_);
  715. return $self->_doRequest("index.php?/api/v2/delete_case/$case_id",'POST');
  716. }
  717. =head2 B<getCases (project_id,suite_id,filters)>
  718. Gets cases for provided section.
  719. =over 4
  720. =item INTEGER C<PROJECT ID> - ID of parent project.
  721. =item INTEGER C<SUITE ID> - ID of parent suite.
  722. =item HASHREF C<FILTERS> (optional) - HASHREF describing parameters to filter cases by.
  723. =back
  724. See:
  725. L<http://docs.gurock.com/testrail-api2/reference-cases#get_cases>
  726. for details as to the allowable filter keys.
  727. If the section ID is omitted, all cases for the suite will be returned.
  728. Returns ARRAYREF of test case definition HASHREFs.
  729. $tr->getCases(1,2, {'section_id' => 3} );
  730. =cut
  731. sub getCases {
  732. state $check = compile(Object, Int, Int, Optional[Maybe[HashRef]]);
  733. my ($self,$project_id,$suite_id,$filters) = $check->(@_);
  734. my $url = "index.php?/api/v2/get_cases/$project_id&suite_id=$suite_id";
  735. $url .= _convert_filters_to_string($filters);
  736. return $self->_doRequest($url);
  737. }
  738. =head2 B<getCaseByName (project_id,suite_id,name,filters)>
  739. Gets case by name.
  740. =over 4
  741. =item INTEGER C<PROJECT ID> - ID of parent project.
  742. =item INTEGER C<SUITE ID> - ID of parent suite.
  743. =item STRING C<NAME> - Name of desired test case.
  744. =item HASHREF C<FILTERS> - Filter dictionary acceptable to getCases.
  745. =back
  746. Returns test case definition HASHREF.
  747. $tr->getCaseByName(1,2,'nugs', {'section_id' => 3});
  748. =cut
  749. sub getCaseByName {
  750. state $check = compile(Object, Int, Int, Str, Optional[Maybe[HashRef]]);
  751. my ($self,$project_id,$suite_id,$name,$filters) = $check->(@_);
  752. my $cases = $self->getCases($project_id,$suite_id,$filters);
  753. return -500 if !$cases || (reftype($cases) || 'undef') ne 'ARRAY';
  754. foreach my $case (@$cases) {
  755. return $case if $case->{'title'} eq $name;
  756. }
  757. return 0;
  758. }
  759. =head2 B<getCaseByID (case_id)>
  760. Gets case by ID.
  761. =over 4
  762. =item INTEGER C<CASE ID> - ID of case.
  763. =back
  764. Returns test case definition HASHREF.
  765. $tr->getCaseByID(1345);
  766. =cut
  767. sub getCaseByID {
  768. state $check = compile(Object, Int);
  769. my ($self,$case_id) = $check->(@_);
  770. return $self->_doRequest("index.php?/api/v2/get_case/$case_id");
  771. }
  772. =head2 getCaseFields
  773. Returns ARRAYREF of available test case custom fields.
  774. $tr->getCaseFields();
  775. Output is cached in the case_fields parameter. Cache is invalidated when addCaseField is called.
  776. =cut
  777. sub getCaseFields {
  778. state $check = compile(Object);
  779. my ($self) = $check->(@_);
  780. return $self->{case_fields} if $self->{case_fields};
  781. $self->{case_fields} = $self->_doRequest("index.php?/api/v2/get_case_fields");
  782. return $self->{case_fields};
  783. }
  784. =head2 addCaseField(%options)
  785. Returns HASHREF describing the case field you just added.
  786. $tr->addCaseField(%options)
  787. =cut
  788. sub addCaseField {
  789. state $check = compile(Object,slurpy HashRef);
  790. my ($self,$options) = $check->(@_);
  791. $self->{case_fields} = undef;
  792. return $self->_doRequest("index.php?/api/v2/add_case_field", 'POST', $options);
  793. }
  794. =head1 PRIORITY METHODS
  795. =head2 B<getPriorities ()>
  796. Gets possible priorities.
  797. Returns ARRAYREF of priority definition HASHREFs.
  798. $tr->getPriorities();
  799. =cut
  800. sub getPriorities {
  801. state $check = compile(Object);
  802. my ($self) = $check->(@_);
  803. return clone($self->{'priority_cache'}) if defined($self->{'priority_cache'});
  804. my $priorities = $self->_doRequest("index.php?/api/v2/get_priorities");
  805. return -500 if !$priorities || (reftype($priorities) || 'undef') ne 'ARRAY';
  806. $self->{'priority_cache'} = $priorities;
  807. return clone $priorities;
  808. }
  809. =head2 B<getPriorityByName (name)>
  810. Gets priority by name.
  811. =over 4
  812. =item STRING C<NAME> - Name of desired priority
  813. =back
  814. Returns priority definition HASHREF.
  815. Dies if named priority does not exist.
  816. $tr->getPriorityByName();
  817. =cut
  818. sub getPriorityByName {
  819. state $check = compile(Object, Str);
  820. my ($self,$name) = $check->(@_);
  821. my $priorities = $self->getPriorities();
  822. return -500 if !$priorities || (reftype($priorities) || 'undef') ne 'ARRAY';
  823. foreach my $priority (@$priorities) {
  824. return $priority if $priority->{'name'} eq $name;
  825. }
  826. confess("No such priority '$name'!");
  827. }
  828. =head2 priorityNamesToIds(names)
  829. Convenience method to translate a list of priority names to TestRail priority IDs.
  830. =over 4
  831. =item ARRAY C<NAMES> - Array of priority names to translate to IDs.
  832. =back
  833. Returns ARRAY of priority IDs in the same order as the priority names passed.
  834. Throws an exception in the case of one (or more) of the names not corresponding to a valid priority.
  835. =cut
  836. sub priorityNamesToIds {
  837. my ($self,@names) = @_;
  838. return _X_in_my_Y($self,$self->getPriorities(),'id',@names);
  839. };
  840. =head1 RUN METHODS
  841. =head2 B<createRun (project_id,suite_id,name,description,milestone_id,assigned_to_id,case_ids)>
  842. Create a run.
  843. =over 4
  844. =item INTEGER C<PROJECT ID> - ID of parent project.
  845. =item INTEGER C<SUITE ID> - ID of suite to base run on
  846. =item STRING C<NAME> - Name of run
  847. =item STRING C<DESCRIPTION> (optional) - Description of run
  848. =item INTEGER C<MILESTONE ID> (optional) - ID of milestone
  849. =item INTEGER C<ASSIGNED TO ID> (optional) - User to assign the run to
  850. =item ARRAYREF C<CASE IDS> (optional) - Array of case IDs in case you don't want to use the whole testsuite when making the build.
  851. =back
  852. Returns run definition HASHREF.
  853. $tr->createRun(1,1345,'RUN AWAY','SO FAR AWAY',22,3,[3,4,5,6]);
  854. =cut
  855. #If you pass an array of case ids, it implies include_all is false
  856. sub createRun {
  857. state $check = compile(Object, Int, Int, Str, Optional[Maybe[Str]], Optional[Maybe[Int]], Optional[Maybe[Int]], Optional[Maybe[ArrayRef[Int]]]);
  858. my ($self,$project_id,$suite_id,$name,$desc,$milestone_id,$assignedto_id,$case_ids) = $check->(@_);
  859. my $stuff = {
  860. suite_id => $suite_id,
  861. name => $name,
  862. description => $desc,
  863. milestone_id => $milestone_id,
  864. assignedto_id => $assignedto_id,
  865. include_all => defined($case_ids) ? 0 : 1,
  866. case_ids => $case_ids
  867. };
  868. return $self->_doRequest("index.php?/api/v2/add_run/$project_id",'POST',$stuff);
  869. }
  870. =head2 B<deleteRun (run_id)>
  871. Deletes specified run.
  872. =over 4
  873. =item INTEGER C<RUN ID> - ID of run to delete.
  874. =back
  875. Returns BOOLEAN.
  876. $tr->deleteRun(1324);
  877. =cut
  878. sub deleteRun {
  879. state $check = compile(Object, Int);
  880. my ($self,$run_id) = $check->(@_);
  881. return $self->_doRequest("index.php?/api/v2/delete_run/$run_id",'POST');
  882. }
  883. =head2 B<getRuns (project_id,filters)>
  884. Get all runs for specified project.
  885. To do this, it must make (no. of runs/250) HTTP requests.
  886. This is due to the maximum result set limit enforced by testrail.
  887. =over 4
  888. =item INTEGER C<PROJECT_ID> - ID of parent project
  889. =item HASHREF C<FILTERS> - (optional) dictionary of filters, with keys corresponding to the documented filters for get_runs (other than limit/offset).
  890. =back
  891. Returns ARRAYREF of run definition HASHREFs.
  892. $allRuns = $tr->getRuns(6969);
  893. Possible filters:
  894. =over 4
  895. =item created_after (UNIX timestamp)
  896. =item created_before (UNIX timestamp)
  897. =item created_by (csv of ints) IDs of users plans were created by
  898. =item is_completed (bool)
  899. =item milestone_id (csv of ints) IDs of milestone assigned to plans
  900. =item refs_filter (string) A single Reference ID (e.g. TR-a, 4291, etc.)
  901. =item suite_id (csv of ints) A comma-separated list of test suite IDs to filter by.
  902. =back
  903. =cut
  904. sub getRuns {
  905. state $check = compile(Object, Int, Optional[Maybe[HashRef]]);
  906. my ($self,$project_id,$filters) = $check->(@_);
  907. my $initial_runs = $self->getRunsPaginated($project_id,$self->{'global_limit'},0,$filters);
  908. return $initial_runs unless (reftype($initial_runs) || 'undef') eq 'ARRAY';
  909. my $runs = [];
  910. push(@$runs,@$initial_runs);
  911. my $offset = 1;
  912. while (scalar(@$initial_runs) == $self->{'global_limit'}) {
  913. $initial_runs = $self->getRunsPaginated($project_id,$self->{'global_limit'},($self->{'global_limit'} * $offset),$filters);
  914. push(@$runs,@$initial_runs);
  915. $offset++;
  916. }
  917. return $runs;
  918. }
  919. =head2 B<getRunsPaginated (project_id,limit,offset,filters)>
  920. Get some runs for specified project.
  921. =over 4
  922. =item INTEGER C<PROJECT_ID> - ID of parent project
  923. =item INTEGER C<LIMIT> - Number of runs to return.
  924. =item INTEGER C<OFFSET> - Page of runs to return.
  925. =item HASHREF C<FILTERS> - (optional) other filters to apply to the requests other than limit/offset. See getRuns for more information.
  926. =back
  927. Returns ARRAYREF of run definition HASHREFs.
  928. $someRuns = $tr->getRunsPaginated(6969,22,4);
  929. =cut
  930. sub getRunsPaginated {
  931. state $check = compile(Object, Int, Optional[Maybe[Int]], Optional[Maybe[Int]], Optional[Maybe[HashRef]]);
  932. my ($self,$project_id,$limit,$offset,$filters) = $check->(@_);
  933. confess("Limit greater than ".$self->{'global_limit'}) if $limit > $self->{'global_limit'};
  934. my $apiurl = "index.php?/api/v2/get_runs/$project_id";
  935. $apiurl .= "&offset=$offset" if defined($offset);
  936. $apiurl .= "&limit=$limit" if $limit; #You have problems if you want 0 results
  937. $apiurl .= _convert_filters_to_string($filters);
  938. return $self->_doRequest($apiurl);
  939. }
  940. =head2 B<getRunByName (project_id,name)>
  941. Gets run by name.
  942. =over 4
  943. =item INTEGER C<PROJECT ID> - ID of parent project.
  944. =item STRING <NAME> - Name of desired run.
  945. =back
  946. Returns run definition HASHREF.
  947. $tr->getRunByName(1,'R2');
  948. =cut
  949. sub getRunByName {
  950. state $check = compile(Object, Int, Str);
  951. my ($self,$project_id,$name) = $check->(@_);
  952. my $runs = $self->getRuns($project_id);
  953. return -500 if !$runs || (reftype($runs) || 'undef') ne 'ARRAY';
  954. foreach my $run (@$runs) {
  955. return $run if $run->{'name'} eq $name;
  956. }
  957. return 0;
  958. }
  959. =head2 B<getRunByID (run_id)>
  960. Gets run by ID.
  961. =over 4
  962. =item INTEGER C<RUN ID> - ID of desired run.
  963. =back
  964. Returns run definition HASHREF.
  965. $tr->getRunByID(7779311);
  966. =cut
  967. sub getRunByID {
  968. state $check = compile(Object, Int);
  969. my ($self,$run_id) = $check->(@_);
  970. return $self->_doRequest("index.php?/api/v2/get_run/$run_id");
  971. }
  972. =head2 B<closeRun (run_id)>
  973. Close the specified run.
  974. =over 4
  975. =item INTEGER C<RUN ID> - ID of desired run.
  976. =back
  977. Returns run definition HASHREF on success, false on failure.
  978. $tr->closeRun(90210);
  979. =cut
  980. sub closeRun {
  981. state $check = compile(Object, Int);
  982. my ($self,$run_id) = $check->(@_);
  983. return $self->_doRequest("index.php?/api/v2/close_run/$run_id",'POST');
  984. }
  985. =head2 B<getRunSummary(runs)>
  986. Returns array of hashrefs describing the # of tests in the run(s) with the available statuses.
  987. Translates custom_statuses into their system names for you.
  988. =over 4
  989. =item ARRAY C<RUNS> - runs obtained from getRun* or getChildRun* methods.
  990. =back
  991. Returns ARRAY of run HASHREFs with the added key 'run_status' holding a hashref where status_name => count.
  992. $tr->getRunSummary($run,$run2);
  993. =cut
  994. sub getRunSummary {
  995. state $check = compile(Object, slurpy ArrayRef[HashRef]);
  996. my ($self,$runs) = $check->(@_);
  997. confess("At least one run must be passed!") unless scalar(@$runs);
  998. #Translate custom statuses
  999. my $statuses = $self->getPossibleTestStatuses();
  1000. my %shash;
  1001. #XXX so, they do these tricks with the status names, see...so map the counts to their relevant status ids.
  1002. @shash{map { ( $_->{'id'} < 6 ) ? $_->{'name'}."_count" : "custom_status".($_->{'id'} - 5)."_count" } @$statuses } = map { $_->{'id'} } @$statuses;
  1003. my @sname;
  1004. #Create listing of keys/values
  1005. @$runs = map {
  1006. my $run = $_;
  1007. @{$run->{statuses}}{grep {$_ =~ m/_count$/} keys(%$run)} = grep {$_ =~ m/_count$/} keys(%$run);
  1008. foreach my $status (keys(%{$run->{'statuses'}})) {
  1009. next if !exists($shash{$status});
  1010. @sname = grep {exists($shash{$status}) && $_->{'id'} == $shash{$status}} @$statuses;
  1011. $run->{'statuses_clean'}->{$sname[0]->{'label'}} = $run->{$status};
  1012. }
  1013. $run;
  1014. } @$runs;
  1015. return map { {'id' => $_->{'id'}, 'name' => $_->{'name'}, 'run_status' => $_->{'statuses_clean'}, 'config_ids' => $_->{'config_ids'} } } @$runs;
  1016. }
  1017. =head2 B<getRunResults(run_id)>
  1018. Returns array of hashrefs describing the results of the run.
  1019. Warning: This only returns the most recent results of a run.
  1020. If you want to know about the tortured journey a test may have taken to get to it's final status,
  1021. you will need to use getTestResults.
  1022. =over 4
  1023. =item INTEGER C<RUN_ID> - Relevant Run's ID.
  1024. =back
  1025. =cut
  1026. sub getRunResults {
  1027. state $check = compile(Object, Int, Optional[Maybe[HashRef]]);
  1028. my ($self,$run_id, $filters) = $check->(@_);
  1029. my $initial_results = $self->getRunResultsPaginated($run_id,$self->{'global_limit'},undef,$filters);
  1030. return $initial_results unless (reftype($initial_results) || 'undef') eq 'ARRAY';
  1031. my $results = [];
  1032. push(@$results,@$initial_results);
  1033. my $offset = 1;
  1034. while (scalar(@$initial_results) == $self->{'global_limit'}) {
  1035. $initial_results = $self->getRunResultsPaginated($run_id,$self->{'global_limit'},($self->{'global_limit'} * $offset),$filters);
  1036. push(@$results,@$initial_results);
  1037. $offset++;
  1038. }
  1039. return $results;
  1040. }
  1041. =head2 B<getRunResultsPaginated(run_id,limit,offset,filters)>
  1042. =cut
  1043. sub getRunResultsPaginated {
  1044. state $check = compile(Object, Int, Optional[Maybe[Int]], Optional[Maybe[Int]], Optional[Maybe[HashRef]]);
  1045. my ($self,$run_id,$limit,$offset,$filters) = $check->(@_);
  1046. confess("Limit greater than ".$self->{'global_limit'}) if $limit > $self->{'global_limit'};
  1047. my $apiurl = "index.php?/api/v2/get_results_for_run/$run_id";
  1048. $apiurl .= "&offset=$offset" if defined($offset);
  1049. $apiurl .= "&limit=$limit" if $limit; #You have problems if you want 0 results
  1050. $apiurl .= _convert_filters_to_string($filters);
  1051. return $self->_doRequest($apiurl);
  1052. }
  1053. =head1 RUN AS CHILD OF PLAN METHODS
  1054. =head2 B<getChildRuns(plan)>
  1055. Extract the child runs from a plan. Convenient, as the structure of this hash is deep, and correct error handling can be tedious.
  1056. =over 4
  1057. =item HASHREF C<PLAN> - Test Plan definition HASHREF returned by any of the PLAN methods below.
  1058. =back
  1059. Returns ARRAYREF of run definition HASHREFs. Returns 0 upon failure to extract the data.
  1060. =cut
  1061. sub getChildRuns {
  1062. state $check = compile(Object, HashRef);
  1063. my ($self,$plan) = $check->(@_);
  1064. return 0 unless defined($plan->{'entries'}) && (reftype($plan->{'entries'}) || 'undef') eq 'ARRAY';
  1065. my $entries = $plan->{'entries'};
  1066. my $plans = [];
  1067. foreach my $entry (@$entries) {
  1068. push(@$plans,@{$entry->{'runs'}}) if defined($entry->{'runs'}) && ((reftype($entry->{'runs'}) || 'undef') eq 'ARRAY')
  1069. }
  1070. return $plans;
  1071. }
  1072. =head2 B<getChildRunByName(plan,name,configurations,testsuite_id)>
  1073. =over 4
  1074. =item HASHREF C<PLAN> - Test Plan definition HASHREF returned by any of the PLAN methods below.
  1075. =item STRING C<NAME> - Name of run to search for within plan.
  1076. =item ARRAYREF C<CONFIGURATIONS> (optional) - Names of configurations to filter runs by.
  1077. =item INTEGER C<TESTSUITE_ID> (optional) - Filter by the provided Testsuite ID. Helpful for when child runs have duplicate names, but are from differing testsuites.
  1078. =back
  1079. Returns run definition HASHREF, or false if no such run is found.
  1080. Convenience method using getChildRuns.
  1081. Will throw a fatal error if one or more of the configurations passed does not exist in the project.
  1082. =cut
  1083. sub getChildRunByName {
  1084. state $check = compile(Object, HashRef, Str, Optional[Maybe[ArrayRef[Str]]], Optional[Maybe[Int]]);
  1085. my ($self,$plan,$name,$configurations,$testsuite_id) = $check->(@_);
  1086. my $runs = $self->getChildRuns($plan);
  1087. @$runs = grep {$_->{suite_id} == $testsuite_id} @$runs if $testsuite_id;
  1088. return 0 if !$runs;
  1089. my @pconfigs = ();
  1090. #Figure out desired config IDs
  1091. if (defined $configurations) {
  1092. my $avail_configs = $self->getConfigurations($plan->{'project_id'});
  1093. my ($cname);
  1094. @pconfigs = map {$_->{'id'}} grep { $cname = $_->{'name'}; grep {$_ eq $cname} @$configurations } @$avail_configs; #Get a list of IDs from the names passed
  1095. }
  1096. confess("One or more configurations passed does not exist in your project!") if defined($configurations) && (scalar(@pconfigs) != scalar(@$configurations));
  1097. my $found;
  1098. foreach my $run (@$runs) {
  1099. next if $run->{name} ne $name;
  1100. next if scalar(@pconfigs) != scalar(@{$run->{'config_ids'}});
  1101. #Compare run config IDs against desired, invalidate run if all conditions not satisfied
  1102. $found = 0;
  1103. foreach my $cid (@{$run->{'config_ids'}}) {
  1104. $found++ if grep {$_ == $cid} @pconfigs;
  1105. }
  1106. return $run if $found == scalar(@{$run->{'config_ids'}});
  1107. }
  1108. return 0;
  1109. }
  1110. =head1 PLAN METHODS
  1111. =head2 B<createPlan (project_id,name,description,milestone_id,entries)>
  1112. Create a test plan.
  1113. =over 4
  1114. =item INTEGER C<PROJECT ID> - ID of parent project.
  1115. =item STRING C<NAME> - Name of plan
  1116. =item STRING C<DESCRIPTION> (optional) - Description of plan
  1117. =item INTEGER C<MILESTONE_ID> (optional) - ID of milestone
  1118. =item ARRAYREF C<ENTRIES> (optional) - New Runs to initially populate the plan with -- See TestRail API documentation for more advanced inputs here.
  1119. =back
  1120. Returns test plan definition HASHREF, or false on failure.
  1121. $entries = [{
  1122. suite_id => 345,
  1123. include_all => 1,
  1124. assignedto_id => 1
  1125. }];
  1126. $tr->createPlan(1,'Gosplan','Robo-Signed Soviet 5-year plan',22,$entries);
  1127. =cut
  1128. sub createPlan {
  1129. state $check = compile(Object, Int, Str, Optional[Maybe[Str]], Optional[Maybe[Int]], Optional[Maybe[ArrayRef[HashRef]]]);
  1130. my ($self,$project_id,$name,$desc,$milestone_id,$entries) = $check->(@_);
  1131. my $stuff = {
  1132. name => $name,
  1133. description => $desc,
  1134. milestone_id => $milestone_id,
  1135. entries => $entries
  1136. };
  1137. return $self->_doRequest("index.php?/api/v2/add_plan/$project_id",'POST',$stuff);
  1138. }
  1139. =head2 B<deletePlan (plan_id)>
  1140. Deletes specified plan.
  1141. =over 4
  1142. =item INTEGER C<PLAN ID> - ID of plan to delete.
  1143. =back
  1144. Returns BOOLEAN.
  1145. $tr->deletePlan(8675309);
  1146. =cut
  1147. sub deletePlan {
  1148. state $check = compile(Object, Int);
  1149. my ($self,$plan_id) = $check->(@_);
  1150. return $self->_doRequest("index.php?/api/v2/delete_plan/$plan_id",'POST');
  1151. }
  1152. =head2 B<getPlans (project_id,filters)>
  1153. Gets all test plans in specified project.
  1154. Like getRuns, must make multiple HTTP requests when the number of results exceeds 250.
  1155. =over 4
  1156. =item INTEGER C<PROJECT ID> - ID of parent project.
  1157. =item HASHREF C<FILTERS> - (optional) dictionary of filters, with keys corresponding to the documented filters for get_plans (other than limit/offset).
  1158. =back
  1159. Returns ARRAYREF of all plan definition HASHREFs in a project.
  1160. $tr->getPlans(8);
  1161. Does not contain any information about child test runs.
  1162. Use getPlanByID or getPlanByName if you want that, in particular if you are interested in using getChildRunByName.
  1163. Possible filters:
  1164. =over 4
  1165. =item created_after (UNIX timestamp)
  1166. =item created_before (UNIX timestamp)
  1167. =item created_by (csv of ints) IDs of users plans were created by
  1168. =item is_completed (bool)
  1169. =item milestone_id (csv of ints) IDs of milestone assigned to plans
  1170. =back
  1171. =cut
  1172. sub getPlans {
  1173. state $check = compile(Object, Int, Optional[Maybe[HashRef]]);
  1174. my ($self,$project_id,$filters) = $check->(@_);
  1175. my $initial_plans = $self->getPlansPaginated($project_id,$self->{'global_limit'},0,$filters);
  1176. return $initial_plans unless (reftype($initial_plans) || 'undef') eq 'ARRAY';
  1177. my $plans = [];
  1178. push(@$plans,@$initial_plans);
  1179. my $offset = 1;
  1180. while (scalar(@$initial_plans) == $self->{'global_limit'}) {
  1181. $initial_plans = $self->getPlansPaginated($project_id,$self->{'global_limit'},($self->{'global_limit'} * $offset),$filters);
  1182. push(@$plans,@$initial_plans);
  1183. $offset++;
  1184. }
  1185. return $plans;
  1186. }
  1187. =head2 B<getPlansPaginated (project_id,limit,offset,filters)>
  1188. Get some plans for specified project.
  1189. =over 4
  1190. =item INTEGER C<PROJECT_ID> - ID of parent project
  1191. =item INTEGER C<LIMIT> - Number of plans to return.
  1192. =item INTEGER C<OFFSET> - Page of plans to return.
  1193. =item HASHREF C<FILTERS> - (optional) other filters to apply to the requests (other than limit/offset). See getPlans for more information.
  1194. =back
  1195. Returns ARRAYREF of plan definition HASHREFs.
  1196. $someRuns = $tr->getPlansPaginated(6969,222,44);
  1197. =cut
  1198. sub getPlansPaginated {
  1199. state $check = compile(Object, Int, Optional[Maybe[Int]], Optional[Maybe[Int]], Optional[Maybe[HashRef]]);
  1200. my ($self,$project_id,$limit,$offset,$filters) = $check->(@_);
  1201. confess("Limit greater than ".$self->{'global_limit'}) if $limit > $self->{'global_limit'};
  1202. my $apiurl = "index.php?/api/v2/get_plans/$project_id";
  1203. $apiurl .= "&offset=$offset" if defined($offset);
  1204. $apiurl .= "&limit=$limit" if $limit; #You have problems if you want 0 results
  1205. $apiurl .= _convert_filters_to_string($filters);
  1206. return $self->_doRequest($apiurl);
  1207. }
  1208. =head2 B<getPlanByName (project_id,name)>
  1209. Gets specified plan by name.
  1210. =over 4
  1211. =item INTEGER C<PROJECT ID> - ID of parent project.
  1212. =item STRING C<NAME> - Name of test plan.
  1213. =back
  1214. Returns plan definition HASHREF.
  1215. $tr->getPlanByName(8,'GosPlan');
  1216. =cut
  1217. sub getPlanByName {
  1218. state $check = compile(Object, Int, Str);
  1219. my ($self,$project_id,$name) = $check->(@_);
  1220. my $plans = $self->getPlans($project_id);
  1221. return -500 if !$plans || (reftype($plans) || 'undef') ne 'ARRAY';
  1222. foreach my $plan (@$plans) {
  1223. if ($plan->{'name'} eq $name) {
  1224. return $self->getPlanByID($plan->{'id'});
  1225. }
  1226. }
  1227. return 0;
  1228. }
  1229. =head2 B<getPlanByID (plan_id)>
  1230. Gets specified plan by ID.
  1231. =over 4
  1232. =item INTEGER C<PLAN ID> - ID of plan.
  1233. =back
  1234. Returns plan definition HASHREF.
  1235. $tr->getPlanByID(2);
  1236. =cut
  1237. sub getPlanByID {
  1238. state $check = compile(Object, Int);
  1239. my ($self,$plan_id) = $check->(@_);
  1240. return $self->_doRequest("index.php?/api/v2/get_plan/$plan_id");
  1241. }
  1242. =head2 B<getPlanSummary(plan_ID)>
  1243. Returns hashref describing the various pass, fail, etc. percentages for tests in the plan.
  1244. The 'totals' key has total cases in each status ('status' => count)
  1245. The 'percentages' key has the same, but as a percentage of the total.
  1246. =over 4
  1247. =item SCALAR C<plan_ID> - ID of your test plan.
  1248. =back
  1249. $tr->getPlanSummary($plan_id);
  1250. =cut
  1251. sub getPlanSummary {
  1252. state $check = compile(Object, Int);
  1253. my ($self,$plan_id) = $check->(@_);
  1254. my $runs = $self->getPlanByID( $plan_id );
  1255. $runs = $self->getChildRuns( $runs );
  1256. @$runs = $self->getRunSummary(@{$runs});
  1257. my $total_sum = 0;
  1258. my $ret = { plan => $plan_id };
  1259. #Compile totals
  1260. foreach my $summary ( @$runs ) {
  1261. my @elems = keys( %{ $summary->{'run_status'} } );
  1262. foreach my $key (@elems) {
  1263. $ret->{'totals'}->{$key} = 0 if !defined $ret->{'totals'}->{$key};
  1264. $ret->{'totals'}->{$key} += $summary->{'run_status'}->{$key};
  1265. $total_sum += $summary->{'run_status'}->{$key};
  1266. }
  1267. }
  1268. #Compile percentages
  1269. foreach my $key (keys(%{$ret->{'totals'}})) {
  1270. next if grep {$_ eq $key} qw{plan configs percentages};
  1271. $ret->{"percentages"}->{$key} = sprintf( "%.2f%%", ( $ret->{'totals'}->{$key} / $total_sum ) * 100 );
  1272. }
  1273. return $ret;
  1274. }
  1275. =head2 B<createRunInPlan (plan_id,suite_id,name,assigned_to_id,config_ids,case_ids)>
  1276. Create a run in a plan.
  1277. =over 4
  1278. =item INTEGER C<PLAN ID> - ID of parent project.
  1279. =item INTEGER C<SUITE ID> - ID of suite to base run on
  1280. =item STRING C<NAME> - Name of run
  1281. =item INTEGER C<ASSIGNED TO ID> (optional) - User to assign the run to
  1282. =item ARRAYREF C<CONFIG IDS> (optional) - Array of Configuration IDs (see getConfigurations) to apply to the created run
  1283. =item ARRAYREF C<CASE IDS> (optional) - Array of case IDs in case you don't want to use the whole testsuite when making the build.
  1284. =back
  1285. Returns run definition HASHREF.
  1286. $tr->createRun(1,1345,'PlannedRun',3,[1,4,77],[3,4,5,6]);
  1287. =cut
  1288. #If you pass an array of case ids, it implies include_all is false
  1289. sub createRunInPlan {
  1290. state $check = compile(Object, Int, Int, Str, Optional[Maybe[Int]], Optional[Maybe[ArrayRef[Int]]], Optional[Maybe[ArrayRef[Int]]]);
  1291. my ($self,$plan_id,$suite_id,$name,$assignedto_id,$config_ids,$case_ids) = $check->(@_);
  1292. my $runs = [
  1293. {
  1294. config_ids => $config_ids,
  1295. include_all => defined($case_ids) ? 0 : 1,
  1296. case_ids => $case_ids
  1297. }
  1298. ];
  1299. my $stuff = {
  1300. suite_id => $suite_id,
  1301. name => $name,
  1302. assignedto_id => $assignedto_id,
  1303. include_all => defined($case_ids) ? 0 : 1,
  1304. case_ids => $case_ids,
  1305. config_ids => $config_ids,
  1306. runs => $runs
  1307. };
  1308. return $self->_doRequest("index.php?/api/v2/add_plan_entry/$plan_id",'POST',$stuff);
  1309. }
  1310. =head2 B<closePlan (plan_id)>
  1311. Close the specified plan.
  1312. =over 4
  1313. =item INTEGER C<PLAN ID> - ID of desired plan.
  1314. =back
  1315. Returns plan definition HASHREF on success, false on failure.
  1316. $tr->closePlan(75020);
  1317. =cut
  1318. sub closePlan {
  1319. state $check = compile(Object, Int);
  1320. my ($self,$plan_id) = $check->(@_);
  1321. return $self->_doRequest("index.php?/api/v2/close_plan/$plan_id",'POST');
  1322. }
  1323. =head1 MILESTONE METHODS
  1324. =head2 B<createMilestone (project_id,name,description,due_on)>
  1325. Create a milestone.
  1326. =over 4
  1327. =item INTEGER C<PROJECT ID> - ID of parent project.
  1328. =item STRING C<NAME> - Name of milestone
  1329. =item STRING C<DESCRIPTION> (optional) - Description of milestone
  1330. =item INTEGER C<DUE_ON> - Date at which milestone should be completed. Unix Timestamp.
  1331. =back
  1332. Returns milestone definition HASHREF, or false on failure.
  1333. $tr->createMilestone(1,'Patriotic victory of world perlism','Accomplish by Robo-Signed Soviet 5-year plan',time()+157788000);
  1334. =cut
  1335. sub createMilestone {
  1336. state $check = compile(Object, Int, Str, Optional[Maybe[Str]], Optional[Maybe[Int]]);
  1337. my ($self,$project_id,$name,$desc,$due_on) = $check->(@_);
  1338. my $stuff = {
  1339. name => $name,
  1340. description => $desc,
  1341. due_on => $due_on # unix timestamp
  1342. };
  1343. return $self->_doRequest("index.php?/api/v2/add_milestone/$project_id",'POST',$stuff);
  1344. }
  1345. =head2 B<deleteMilestone (milestone_id)>
  1346. Deletes specified milestone.
  1347. =over 4
  1348. =item INTEGER C<MILESTONE ID> - ID of milestone to delete.
  1349. =back
  1350. Returns BOOLEAN.
  1351. $tr->deleteMilestone(86);
  1352. =cut
  1353. sub deleteMilestone {
  1354. state $check = compile(Object, Int);
  1355. my ($self,$milestone_id) = $check->(@_);
  1356. return $self->_doRequest("index.php?/api/v2/delete_milestone/$milestone_id",'POST');
  1357. }
  1358. =head2 B<getMilestones (project_id,filters)>
  1359. Get milestones for some project.
  1360. =over 4
  1361. =item INTEGER C<PROJECT ID> - ID of parent project.
  1362. =item HASHREF C<FILTERS> (optional) - HASHREF describing parameters to filter milestones by.
  1363. =back
  1364. See:
  1365. L<https://www.gurock.com/testrail/docs/api/reference/milestones#getmilestones>
  1366. for details as to the allowable filter keys.
  1367. Returns ARRAYREF of milestone definition HASHREFs.
  1368. $tr->getMilestones(8);
  1369. =cut
  1370. sub getMilestones {
  1371. state $check = compile(Object, Int, Optional[Maybe[HashRef]]);
  1372. my ($self,$project_id, $filters) = $check->(@_);
  1373. return $self->_doRequest("index.php?/api/v2/get_milestones/$project_id" . _convert_filters_to_string($filters));
  1374. }
  1375. =head2 B<getMilestoneByName (project_id,name)>
  1376. Gets specified milestone by name.
  1377. =over 4
  1378. =item INTEGER C<PROJECT ID> - ID of parent project.
  1379. =item STRING C<NAME> - Name of milestone.
  1380. =back
  1381. Returns milestone definition HASHREF.
  1382. $tr->getMilestoneByName(8,'whee');
  1383. =cut
  1384. sub getMilestoneByName {
  1385. state $check = compile(Object, Int, Str);
  1386. my ($self,$project_id,$name) = $check->(@_);
  1387. my $milestones = $self->getMilestones($project_id);
  1388. return -500 if !$milestones || (reftype($milestones) || 'undef') ne 'ARRAY';
  1389. foreach my $milestone (@$milestones) {
  1390. return $milestone if $milestone->{'name'} eq $name;
  1391. }
  1392. return 0;
  1393. }
  1394. =head2 B<getMilestoneByID (milestone_id)>
  1395. Gets specified milestone by ID.
  1396. =over 4
  1397. =item INTEGER C<MILESTONE ID> - ID of milestone.
  1398. =back
  1399. Returns milestone definition HASHREF.
  1400. $tr->getMilestoneByID(2);
  1401. =cut
  1402. sub getMilestoneByID {
  1403. state $check = compile(Object, Int);
  1404. my ($self,$milestone_id) = $check->(@_);
  1405. return $self->_doRequest("index.php?/api/v2/get_milestone/$milestone_id");
  1406. }
  1407. =head1 TEST METHODS
  1408. =head2 B<getTests (run_id,status_ids,assignedto_ids)>
  1409. Get tests for some run. Optionally filter by provided status_ids and assigned_to ids.
  1410. =over 4
  1411. =item INTEGER C<RUN ID> - ID of parent run.
  1412. =item ARRAYREF C<STATUS IDS> (optional) - IDs of relevant test statuses to filter by. Get with getPossibleTestStatuses.
  1413. =item ARRAYREF C<ASSIGNEDTO IDS> (optional) - IDs of users assigned to test to filter by. Get with getUsers.
  1414. =back
  1415. Returns ARRAYREF of test definition HASHREFs.
  1416. $tr->getTests(8,[1,2,3],[2]);
  1417. =cut
  1418. sub getTests {
  1419. state $check = compile(Object, Int, Optional[Maybe[ArrayRef[Int]]], Optional[Maybe[ArrayRef[Int]]]);
  1420. my ($self,$run_id,$status_ids,$assignedto_ids) = $check->(@_);
  1421. my $query_string = '';
  1422. $query_string = '&status_id='.join(',',@$status_ids) if defined($status_ids) && scalar(@$status_ids);
  1423. my $results = $self->_doRequest("index.php?/api/v2/get_tests/$run_id$query_string");
  1424. @$results = grep {my $aid = $_->{'assignedto_id'}; grep {defined($aid) && $aid == $_} @$assignedto_ids} @$results if defined($assignedto_ids) && scalar(@$assignedto_ids);
  1425. #Cache stuff for getTestByName
  1426. $self->{tests_cache} //= {};
  1427. $self->{tests_cache}->{$run_id} = $results;
  1428. return clone($results);
  1429. }
  1430. =head2 B<getTestByName (run_id,name)>
  1431. Gets specified test by name.
  1432. This is done by getting the list of all tests in the run and then picking out the relevant test.
  1433. As such, for efficiency the list of tests is cached.
  1434. The cache may be refreshed, or restricted by running getTests (with optional restrictions, such as assignedto_ids, etc).
  1435. =over 4
  1436. =item INTEGER C<RUN ID> - ID of parent run.
  1437. =item STRING C<NAME> - Name of milestone.
  1438. =back
  1439. Returns test definition HASHREF.
  1440. $tr->getTestByName(36,'wheeTest');
  1441. =cut
  1442. sub getTestByName {
  1443. state $check = compile(Object, Int, Str);
  1444. my ($self,$run_id,$name) = $check->(@_);
  1445. $self->{tests_cache} //= {};
  1446. my $tests = $self->{tests_cache}->{$run_id};
  1447. $tests = $self->getTests($run_id) if !$tests;
  1448. return -500 if !$tests || (reftype($tests) || 'undef') ne 'ARRAY';
  1449. foreach my $test (@$tests) {
  1450. return $test if $test->{'title'} eq $name;
  1451. }
  1452. return 0;
  1453. }
  1454. =head2 B<getTestByID (test_id)>
  1455. Gets specified test by ID.
  1456. =over 4
  1457. =item INTEGER C<TEST ID> - ID of test.
  1458. =back
  1459. Returns test definition HASHREF.
  1460. $tr->getTestByID(222222);
  1461. =cut
  1462. sub getTestByID {
  1463. state $check = compile(Object, Int);
  1464. my ($self,$test_id) = $check->(@_);
  1465. return $self->_doRequest("index.php?/api/v2/get_test/$test_id");
  1466. }
  1467. =head2 B<getTestResultFields()>
  1468. Gets custom fields that can be set for tests.
  1469. Returns ARRAYREF of result definition HASHREFs.
  1470. =cut
  1471. sub getTestResultFields {
  1472. state $check = compile(Object);
  1473. my ($self) = $check->(@_);
  1474. return $self->{'tr_fields'} if defined($self->{'tr_fields'}); #cache
  1475. $self->{'tr_fields'} = $self->_doRequest('index.php?/api/v2/get_result_fields');
  1476. return $self->{'tr_fields'};
  1477. }
  1478. =head2 B<getTestResultFieldByName(SYSTEM_NAME,PROJECT_ID)>
  1479. Gets a test result field by it's system name. Optionally filter by project ID.
  1480. =over 4
  1481. =item B<SYSTEM NAME> - STRING: system name of a result field.
  1482. =item B<PROJECT ID> - INTEGER (optional): Filter by whether or not the field is enabled for said project
  1483. =back
  1484. Returns a value less than 0 if unsuccessful.
  1485. =cut
  1486. sub getTestResultFieldByName {
  1487. state $check = compile(Object, Str, Optional[Maybe[Int]]);
  1488. my ($self,$system_name,$project_id) = $check->(@_);
  1489. my @candidates = grep { $_->{'name'} eq $system_name} @{$self->getTestResultFields()};
  1490. return 0 if !scalar(@candidates); #No such name
  1491. return -1 if ref($candidates[0]) ne 'HASH';
  1492. return -2 if ref($candidates[0]->{'configs'}) ne 'ARRAY' && !scalar(@{$candidates[0]->{'configs'}}); #bogofilter
  1493. #Give it to the user
  1494. my $ret = $candidates[0]; #copy/save for later
  1495. return $ret if !defined($project_id);
  1496. #Filter by project ID
  1497. foreach my $config (@{$candidates[0]->{'configs'}}) {
  1498. return $ret if ( grep { $_ == $project_id} @{ $config->{'context'}->{'project_ids'} } )
  1499. }
  1500. return -3;
  1501. }
  1502. =head2 B<getPossibleTestStatuses()>
  1503. Gets all possible statuses a test can be set to.
  1504. Returns ARRAYREF of status definition HASHREFs.
  1505. Caches the result for the lifetime of the TestRail::API object.
  1506. =cut
  1507. sub getPossibleTestStatuses {
  1508. state $check = compile(Object);
  1509. my ($self) = $check->(@_);
  1510. return $self->{'status_cache'} if $self->{'status_cache'};
  1511. $self->{'status_cache'} = $self->_doRequest('index.php?/api/v2/get_statuses');
  1512. return clone $self->{'status_cache'};
  1513. }
  1514. =head2 statusNamesToIds(names)
  1515. Convenience method to translate a list of statuses to TestRail status IDs.
  1516. The names referred to here are 'internal names' rather than the labels shown in TestRail.
  1517. =over 4
  1518. =item ARRAY C<NAMES> - Array of status names to translate to IDs.
  1519. =back
  1520. Returns ARRAY of status IDs in the same order as the status names passed.
  1521. Throws an exception in the case of one (or more) of the names not corresponding to a valid test status.
  1522. =cut
  1523. sub statusNamesToIds {
  1524. my ($self,@names) = @_;
  1525. return _X_in_my_Y($self,$self->getPossibleTestStatuses(),'id',@names);
  1526. };
  1527. =head2 statusNamesToLabels(names)
  1528. Convenience method to translate a list of statuses to TestRail status labels (the 'nice' form of status names).
  1529. This is useful when interacting with getRunSummary or getPlanSummary, which uses these labels as hash keys.
  1530. =over 4
  1531. =item ARRAY C<NAMES> - Array of status names to translate to IDs.
  1532. =back
  1533. Returns ARRAY of status labels in the same order as the status names passed.
  1534. Throws an exception in the case of one (or more) of the names not corresponding to a valid test status.
  1535. =cut
  1536. sub statusNamesToLabels {
  1537. my ($self,@names) = @_;
  1538. return _X_in_my_Y($self,$self->getPossibleTestStatuses(),'label',@names);
  1539. };
  1540. # Reduce code duplication with internal methods?
  1541. # It's more likely than you think
  1542. # Free PC check @ cpan.org
  1543. sub _X_in_my_Y {
  1544. state $check = compile(Object, ArrayRef, Str, slurpy ArrayRef[Str]);
  1545. my ($self,$search_arr,$key,$names) = $check->(@_);
  1546. my @ret;
  1547. foreach my $name (@$names) {
  1548. foreach my $member (@$search_arr) {
  1549. if ($member->{'name'} eq $name) {
  1550. push @ret, $member->{$key};
  1551. last;
  1552. }
  1553. }
  1554. }
  1555. confess("One or more names provided does not exist in TestRail.") unless scalar(@$names) == scalar(@ret);
  1556. return @ret;
  1557. }
  1558. =head2 B<createTestResults(test_id,status_id,comment,options,custom_options)>
  1559. Creates a result entry for a test.
  1560. =over 4
  1561. =item INTEGER C<TEST_ID> - ID of desired test
  1562. =item INTEGER C<STATUS_ID> - ID of desired test result status
  1563. =item STRING C<COMMENT> (optional) - Any comments about this result
  1564. =item HASHREF C<OPTIONS> (optional) - Various "Baked-In" options that can be set for test results. See TR docs for more information.
  1565. =item HASHREF C<CUSTOM OPTIONS> (optional) - Options to set for custom fields. See buildStepResults for a simple way to post up custom steps.
  1566. =back
  1567. Returns result definition HASHREF.
  1568. $options = {
  1569. elapsed => '30m 22s',
  1570. defects => ['TSR-3','BOOM-44'],
  1571. version => '6969'
  1572. };
  1573. $custom_options = {
  1574. step_results => [
  1575. {
  1576. content => 'Step 1',
  1577. expected => "Bought Groceries",
  1578. actual => "No Dinero!",
  1579. status_id => 2
  1580. },
  1581. {
  1582. content => 'Step 2',
  1583. expected => 'Ate Dinner',
  1584. actual => 'Went Hungry',
  1585. status_id => 2
  1586. }
  1587. ]
  1588. };
  1589. $res = $tr->createTestResults(1,2,'Test failed because it was all like WAAAAAAA when I poked it',$options,$custom_options);
  1590. =cut
  1591. sub createTestResults {
  1592. state $check = compile(Object, Int, Int, Optional[Maybe[Str]], Optional[Maybe[HashRef]], Optional[Maybe[HashRef]]);
  1593. my ($self,$test_id,$status_id,$comment,$opts,$custom_fields) = $check->(@_);
  1594. my $stuff = {
  1595. status_id => $status_id,
  1596. comment => $comment
  1597. };
  1598. #Handle options
  1599. if (defined($opts) && reftype($opts) eq 'HASH') {
  1600. $stuff->{'version'} = defined($opts->{'version'}) ? $opts->{'version'} : undef;
  1601. $stuff->{'elapsed'} = defined($opts->{'elapsed'}) ? $opts->{'elapsed'} : undef;
  1602. $stuff->{'defects'} = defined($opts->{'defects'}) ? join(',',@{$opts->{'defects'}}) : undef;
  1603. $stuff->{'assignedto_id'} = defined($opts->{'assignedto_id'}) ? $opts->{'assignedto_id'} : undef;
  1604. }
  1605. #Handle custom fields
  1606. if (defined($custom_fields) && reftype($custom_fields) eq 'HASH') {
  1607. foreach my $field (keys(%$custom_fields)) {
  1608. $stuff->{"custom_$field"} = $custom_fields->{$field};
  1609. }
  1610. }
  1611. return $self->_doRequest("index.php?/api/v2/add_result/$test_id",'POST',$stuff);
  1612. }
  1613. =head2 bulkAddResults(run_id,results)
  1614. Add multiple results to a run, where each result is a HASHREF with keys as outlined in the get_results API call documentation.
  1615. =over 4
  1616. =item INTEGER C<RUN_ID> - ID of desired run to add results to
  1617. =item ARRAYREF C<RESULTS> - Array of result HASHREFs to upload.
  1618. =back
  1619. Returns ARRAYREF of result definition HASHREFs.
  1620. =cut
  1621. sub bulkAddResults {
  1622. state $check = compile(Object, Int, ArrayRef[HashRef]);
  1623. my ($self,$run_id, $results) = $check->(@_);
  1624. return $self->_doRequest("index.php?/api/v2/add_results/$run_id", 'POST', { 'results' => $results });
  1625. }
  1626. =head2 bulkAddResultsByCase(run_id,results)
  1627. Basically the same as bulkAddResults, but instead of a test_id for each entry you use a case_id.
  1628. =cut
  1629. sub bulkAddResultsByCase {
  1630. state $check = compile(Object, Int, ArrayRef[HashRef]);
  1631. my ($self,$run_id, $results) = $check->(@_);
  1632. return $self->_doRequest("index.php?/api/v2/add_results_for_cases/$run_id", 'POST', { 'results' => $results });
  1633. }
  1634. =head2 B<getTestResults(test_id,limit,offset,filters)>
  1635. Get the recorded results for desired test, limiting output to 'limit' entries.
  1636. =over 4
  1637. =item INTEGER C<TEST_ID> - ID of desired test
  1638. =item POSITIVE INTEGER C<LIMIT> (OPTIONAL) - provide no more than this number of results.
  1639. =item INTEGER C<OFFSET> (OPTIONAL) - Offset to begin viewing result set at.
  1640. =item HASHREF C<FILTERS> (optional) - HASHREF describing parameters to filter test results by (other than limit/offset).
  1641. =back
  1642. See:
  1643. L<https://www.gurock.com/testrail/docs/api/reference/results#getresults>
  1644. for details as to the allowable filter keys.
  1645. Returns ARRAYREF of result definition HASHREFs.
  1646. =cut
  1647. sub getTestResults {
  1648. state $check = compile(Object, Int, Optional[Maybe[Int]], Optional[Maybe[Int]], Optional[Maybe[HashRef]]);
  1649. my ($self,$test_id,$limit,$offset,$filters) = $check->(@_);
  1650. my $url = "index.php?/api/v2/get_results/$test_id";
  1651. $url .= "&limit=$limit" if $limit;
  1652. $url .= "&offset=$offset" if defined($offset);
  1653. $url .= _convert_filters_to_string($filters);
  1654. return $self->_doRequest($url);
  1655. }
  1656. =head2 B<getResultsForCase(run_id,case_id,limit,offset,filters)>
  1657. Get the recorded results for a test run and case combination., limiting output to 'limit' entries.
  1658. =over 4
  1659. =item INTEGER C<RUN_ID> - ID of desired run
  1660. =item INTEGER C<CASE_ID> - ID of desired case
  1661. =item POSITIVE INTEGER C<LIMIT> (OPTIONAL) - provide no more than this number of results.
  1662. =item INTEGER C<OFFSET> (OPTIONAL) - Offset to begin viewing result set at.
  1663. =item HASHREF C<FILTERS> (optional) - HASHREF describing parameters to filter by (other than limit/offset).
  1664. =back
  1665. See:
  1666. L<https://www.gurock.com/testrail/docs/api/reference/results#getresultsforcase>
  1667. for details as to the allowable filter keys.
  1668. Returns ARRAYREF of result definition HASHREFs.
  1669. =cut
  1670. sub getResultsForCase {
  1671. state $check = compile(Object, Int, Int, Optional[Maybe[Int]], Optional[Maybe[Int]], Optional[Maybe[HashRef]]);
  1672. my ($self,$run_id,$case_id,$limit,$offset,$filters) = $check->(@_);
  1673. my $url = "index.php?/api/v2/get_results_for_case/$run_id/$case_id";
  1674. $url .= "&limit=$limit" if $limit;
  1675. $url .= "&offset=$offset" if defined($offset);
  1676. $url .= _convert_filters_to_string($filters);
  1677. return $self->_doRequest($url);
  1678. }
  1679. =head1 CONFIGURATION METHODS
  1680. =head2 B<getConfigurationGroups(project_id)>
  1681. Gets the available configuration groups for a project, with their configurations as children.
  1682. =over 4
  1683. =item INTEGER C<PROJECT_ID> - ID of relevant project
  1684. =back
  1685. Returns ARRAYREF of configuration group definition HASHREFs.
  1686. =cut
  1687. sub getConfigurationGroups {
  1688. state $check = compile(Object, Int);
  1689. my ($self,$project_id) = $check->(@_);
  1690. my $url = "index.php?/api/v2/get_configs/$project_id";
  1691. return $self->_doRequest($url);
  1692. }
  1693. =head2 B<getConfigurationGroupByName(project_id,name)>
  1694. Get the provided configuration group by name.
  1695. Returns false if the configuration group could not be found.
  1696. =cut
  1697. sub getConfigurationGroupByName {
  1698. state $check = compile(Object, Int, Str);
  1699. my ($self,$project_id,$name) = $check->(@_);
  1700. my $cgroups = $self->getConfigurationGroups($project_id);
  1701. return 0 if ref($cgroups) ne 'ARRAY';
  1702. @$cgroups = grep {$_->{'name'} eq $name} @$cgroups;
  1703. return 0 unless scalar(@$cgroups);
  1704. return $cgroups->[0];
  1705. }
  1706. =head2 B<addConfigurationGroup(project_id,name)>
  1707. New in TestRail 5.2.
  1708. Add a configuration group to the specified project.
  1709. =over 4
  1710. =item INTEGER C<PROJECT_ID> - ID of relevant project
  1711. =item STRING C<NAME> - Name for new configuration Group.
  1712. =back
  1713. Returns HASHREF with new configuration group.
  1714. =cut
  1715. sub addConfigurationGroup {
  1716. state $check = compile(Object, Int, Str);
  1717. my ($self,$project_id,$name) = $check->(@_);
  1718. my $url = "index.php?/api/v2/add_config_group/$project_id";
  1719. return $self->_doRequest($url,'POST',{'name' => $name});
  1720. }
  1721. =head2 B<editConfigurationGroup(config_group_id,name)>
  1722. New in TestRail 5.2.
  1723. Change the name of a configuration group.
  1724. =over 4
  1725. =item INTEGER C<CONFIG_GROUP_ID> - ID of relevant configuration group
  1726. =item STRING C<NAME> - Name for new configuration Group.
  1727. =back
  1728. Returns HASHREF with new configuration group.
  1729. =cut
  1730. sub editConfigurationGroup {
  1731. state $check = compile(Object, Int, Str);
  1732. my ($self,$config_group_id,$name) = $check->(@_);
  1733. my $url = "index.php?/api/v2/update_config_group/$config_group_id";
  1734. return $self->_doRequest($url,'POST',{'name' => $name});
  1735. }
  1736. =head2 B<deleteConfigurationGroup(config_group_id)>
  1737. New in TestRail 5.2.
  1738. Delete a configuration group.
  1739. =over 4
  1740. =item INTEGER C<CONFIG_GROUP_ID> - ID of relevant configuration group
  1741. =back
  1742. Returns BOOL.
  1743. =cut
  1744. sub deleteConfigurationGroup {
  1745. state $check = compile(Object, Int);
  1746. my ($self,$config_group_id) = $check->(@_);
  1747. my $url = "index.php?/api/v2/delete_config_group/$config_group_id";
  1748. return $self->_doRequest($url,'POST');
  1749. }
  1750. =head2 B<getConfigurations(project_id)>
  1751. Gets the available configurations for a project.
  1752. Mostly for convenience (no need to write a boilerplate loop over the groups).
  1753. =over 4
  1754. =item INTEGER C<PROJECT_ID> - ID of relevant project
  1755. =back
  1756. Returns ARRAYREF of configuration definition HASHREFs.
  1757. Returns result of getConfigurationGroups (likely -500) in the event that call fails.
  1758. =cut
  1759. sub getConfigurations {
  1760. state $check = compile(Object, Int);
  1761. my ($self,$project_id) = $check->(@_);
  1762. my $cgroups = $self->getConfigurationGroups($project_id);
  1763. my $configs = [];
  1764. return $cgroups unless (reftype($cgroups) || 'undef') eq 'ARRAY';
  1765. foreach my $cfg (@$cgroups) {
  1766. push(@$configs, @{$cfg->{'configs'}});
  1767. }
  1768. return $configs;
  1769. }
  1770. =head2 B<addConfiguration(configuration_group_id,name)>
  1771. New in TestRail 5.2.
  1772. Add a configuration to the specified configuration group.
  1773. =over 4
  1774. =item INTEGER C<CONFIGURATION_GROUP_ID> - ID of relevant configuration group
  1775. =item STRING C<NAME> - Name for new configuration.
  1776. =back
  1777. Returns HASHREF with new configuration.
  1778. =cut
  1779. sub addConfiguration {
  1780. state $check = compile(Object, Int, Str);
  1781. my ($self,$configuration_group_id,$name) = $check->(@_);
  1782. my $url = "index.php?/api/v2/add_config/$configuration_group_id";
  1783. return $self->_doRequest($url,'POST',{'name' => $name});
  1784. }
  1785. =head2 B<editConfiguration(config_id,name)>
  1786. New in TestRail 5.2.
  1787. Change the name of a configuration.
  1788. =over 4
  1789. =item INTEGER C<CONFIG_ID> - ID of relevant configuration.
  1790. =item STRING C<NAME> - New name for configuration.
  1791. =back
  1792. Returns HASHREF with new configuration group.
  1793. =cut
  1794. sub editConfiguration {
  1795. state $check = compile(Object, Int, Str);
  1796. my ($self,$config_id,$name) = $check->(@_);
  1797. my $url = "index.php?/api/v2/update_config/$config_id";
  1798. return $self->_doRequest($url,'POST',{'name' => $name});
  1799. }
  1800. =head2 B<deleteConfiguration(config_id)>
  1801. New in TestRail 5.2.
  1802. Delete a configuration.
  1803. =over 4
  1804. =item INTEGER C<CONFIG_ID> - ID of relevant configuration
  1805. =back
  1806. Returns BOOL.
  1807. =cut
  1808. sub deleteConfiguration {
  1809. state $check = compile(Object, Int);
  1810. my ($self,$config_id) = $check->(@_);
  1811. my $url = "index.php?/api/v2/delete_config/$config_id";
  1812. return $self->_doRequest($url,'POST');
  1813. }
  1814. =head2 B<translateConfigNamesToIds(project_id,configs)>
  1815. Transforms a list of configuration names into a list of config IDs.
  1816. =over 4
  1817. =item INTEGER C<PROJECT_ID> - Relevant project ID for configs.
  1818. =item ARRAY C<CONFIGS> - Array of config names
  1819. =back
  1820. Returns ARRAY of configuration names, with undef values for unknown configuration names.
  1821. =cut
  1822. sub translateConfigNamesToIds {
  1823. my ($self,$project_id,@names) = @_;
  1824. my $configs = $self->getConfigurations($project_id) or confess("Could not determine configurations in provided project.");
  1825. return _X_in_my_Y($self,$configs,'id',@names);
  1826. }
  1827. =head1 REPORT METHODS
  1828. =head2 getReports
  1829. Return the ARRAYREF of reports available for the provided project.
  1830. Requires you to mark a particular report as accessible in the API via the TestRail report interface.
  1831. =over 4
  1832. =item INTEGER C<PROJECT_ID> - Relevant project ID.
  1833. =back
  1834. =cut
  1835. sub getReports {
  1836. state $check = compile(Object, Int);
  1837. my ($self,$project_id) = $check->(@_);
  1838. my $url = "index.php?/api/v2/get_reports/$project_id";
  1839. return $self->_doRequest($url,'GET');
  1840. }
  1841. =head2 runReport
  1842. Compute the provided report using currently available data.
  1843. Returns HASHREF describing URLs to access completed reports.
  1844. =over 4
  1845. =item INTEGER C<REPORT_ID> - Relevant report ID.
  1846. =back
  1847. =cut
  1848. sub runReport {
  1849. state $check = compile(Object, Int);
  1850. my ($self,$report_id) = $check->(@_);
  1851. my $url = "index.php?/api/v2/run_report/$report_id";
  1852. return $self->_doRequest($url,'GET');
  1853. }
  1854. =head1 STATIC METHODS
  1855. =head2 B<buildStepResults(content,expected,actual,status_id)>
  1856. Convenience method to build the stepResult hashes seen in the custom options for getTestResults.
  1857. =over 4
  1858. =item STRING C<CONTENT> (optional) - The step itself.
  1859. =item STRING C<EXPECTED> (optional) - Expected result of test step.
  1860. =item STRING C<ACTUAL> (optional) - Actual result of test step
  1861. =item INTEGER C<STATUS ID> (optional) - Status ID of result
  1862. =back
  1863. =cut
  1864. #Convenience method for building stepResults
  1865. sub buildStepResults {
  1866. state $check = compile(Str, Str, Str, Int);
  1867. my ($content,$expected,$actual,$status_id) = $check->(@_);
  1868. return {
  1869. content => $content,
  1870. expected => $expected,
  1871. actual => $actual,
  1872. status_id => $status_id
  1873. };
  1874. }
  1875. # Convenience method for building filter string from filters Hashref
  1876. sub _convert_filters_to_string {
  1877. state $check = compile(Maybe[HashRef]);
  1878. my ($filters) = $check->(@_);
  1879. $filters //= {};
  1880. my @valid_keys = qw{ is_completed limit offset created_after created_before filter refs section_id updated_after updated_before refs_filter defects_filter };
  1881. my @valid_arrayref_keys = qw{ created_by milestone_id priority_id template_id type_id updated_by suite_id status_id };
  1882. my $filter_string = '';
  1883. foreach my $filter (keys(%$filters)) {
  1884. confess("Invalid filter key '$filter' passed") unless grep {$_ eq $filter} (@valid_keys, @valid_arrayref_keys);
  1885. if (ref $filters->{$filter} eq 'ARRAY') {
  1886. confess "$filter cannot be an ARRAYREF" if grep {$_ eq $filter} @valid_keys;
  1887. $filter_string .= "&$filter=".join(',',@{$filters->{$filter}});
  1888. } else {
  1889. $filter_string .= "&$filter=".$filters->{$filter} if defined($filters->{$filter});
  1890. }
  1891. }
  1892. return $filter_string;
  1893. }
  1894. 1;
  1895. __END__
  1896. =head1 SEE ALSO
  1897. L<HTTP::Request>
  1898. L<LWP::UserAgent>
  1899. L<JSON::MaybeXS>
  1900. L<http://docs.gurock.com/testrail-api2/start>
  1901. =head1 SPECIAL THANKS
  1902. Thanks to cPanel Inc, for graciously funding the creation of this module.