TestRail-API.t 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. use strict;
  2. use warnings;
  3. use FindBin;
  4. use lib "$FindBin::Bin/lib";
  5. use TestRail::API;
  6. use Test::LWP::UserAgent::TestRailMock;
  7. use Test::More tests => 95;
  8. use Test::Fatal;
  9. use Test::Deep;
  10. use Scalar::Util ();
  11. use Capture::Tiny qw{capture};
  12. my $apiurl = $ENV{'TESTRAIL_API_URL'};
  13. my $login = $ENV{'TESTRAIL_USER'};
  14. my $pw = $ENV{'TESTRAIL_PASSWORD'};
  15. #Mock if nothing is provided
  16. my $is_mock = (!$apiurl && !$login && !$pw);
  17. like(exception {TestRail::API->new('trash','bogus','bogus');}, qr/invalid uri/i, "Non-URIs bounce constructor");
  18. #XXX for some insane reason 'hokum.bogus' seems to be popular with cpantesters
  19. my $bogoError = exception { capture { TestRail::API->new('http://hokum.bogus','lies','moreLies',undef,0); } };
  20. SKIP: {
  21. skip("Some CPANTesters like to randomly redirect all DNS misses to some other host, apparently", 1) if ($bogoError =~ m/404|302/);
  22. skip("Occasionally upon first run of tests, LWP mocking will just not work.",1) if ($bogoError =~ m/Unknown error occurred/i);
  23. like($bogoError, qr/Could not communicate with TestRail Server/i,"Bogus Testrail URI rejected");
  24. }
  25. SKIP: {
  26. skip("Testing authentication not supported with mock",2) if ($is_mock);
  27. like(exception {TestRail::API->new($apiurl,'lies','moreLies',undef,0,0); }, qr/Bad user credentials/i,"Bogus Testrail User rejected");
  28. like(exception {TestRail::API->new($apiurl,$login,'m043L13s ',undef,0); }, qr/Bad user credentials/i,"Bogus Testrail Password rejected");
  29. }
  30. ($apiurl,$login,$pw) = ('http://testrail.local','teodesian@cpan.org','fake') if $is_mock;
  31. my $tr = new TestRail::API($apiurl,$login,$pw,undef,1);
  32. #Mock if necesary
  33. $tr->{'debug'} = 0;
  34. $tr->{'browser'} = $Test::LWP::UserAgent::TestRailMock::mockObject if $is_mock;
  35. my $res;
  36. capture { $res = $tr->_doRequest('noSuchMethod') };
  37. is($res,-404,'Requesting bad URI returns 404');
  38. #Test USER methods
  39. my $userlist = $tr->getUsers();
  40. ok(@$userlist,"Get Users returns list");
  41. my $myuser = $tr->getUserByEmail($login);
  42. is($myuser->{'email'},$login,"Can get user by email");
  43. is($tr->getUserByID($myuser->{'id'})->{'id'},$myuser->{'id'},"Can get user by ID");
  44. is($tr->getUserByName($myuser->{'name'})->{'name'},$myuser->{'name'},"Can get user by Name");
  45. my @user_names = map {$_->{'name'}} @$userlist;
  46. my @user_ids = map {$_->{'id'}} @$userlist;
  47. my @cuser_ids = $tr->userNamesToIds(@user_names);
  48. cmp_deeply(\@cuser_ids,\@user_ids,"userNamesToIds functions correctly");
  49. isnt(exception {$tr->userNamesToIds(@user_names,'potzrebie'); }, undef, "Passing invalid user name throws exception");
  50. #Test CASE TYPE method
  51. my $caseTypes = $tr->getCaseTypes();
  52. is(ref($caseTypes),'ARRAY',"getCaseTypes returns ARRAY of case types");
  53. my @type_names = map {$_->{'name'}} @$caseTypes;
  54. my @type_ids = map {$_->{'id'}} @$caseTypes;
  55. is($tr->getCaseTypeByName($type_names[0])->{'id'},$type_ids[0],"Can get case type by name correctly");
  56. my @computed_type_ids = $tr->typeNamesToIds(@type_names);
  57. cmp_deeply(\@computed_type_ids,\@type_ids,"typeNamesToIds returns the correct type IDs in the correct order");
  58. #Test PRIORITY method
  59. my $priorities = $tr->getPriorities();
  60. is(ref($priorities),'ARRAY',"getPriorities returns ARRAY of priorities");
  61. my @priority_names = map {$_->{'name'}} @$priorities;
  62. my @priority_ids = map {$_->{'id'}} @$priorities;
  63. is($tr->getPriorityByName($priority_names[0])->{'id'},$priority_ids[0],"Can get case priority by name correctly");
  64. my @computed_priority_ids = $tr->priorityNamesToIds(@priority_names);
  65. cmp_deeply(\@computed_priority_ids,\@priority_ids,"priorityNamesToIds returns the correct priority IDs in the correct order");
  66. #Test PROJECT methods
  67. my $project_name = 'CRUSH ALL HUMANS';
  68. my $new_project = $tr->createProject($project_name,'Robo-Signed Soviet 5 Year Project');
  69. is($new_project->{'name'},$project_name,"Can create new project");
  70. ok($tr->getProjects(),"Get Projects returns list");
  71. is($tr->getProjectByName($project_name)->{'name'},$project_name,"Can get project by name");
  72. my $pjid = $tr->getProjectByID($new_project->{'id'});
  73. is(Scalar::Util::reftype($pjid) eq 'HASH' ? $pjid->{'id'} : $pjid,$new_project->{'id'},"Can get project by id");
  74. #Test TESTSUITE methods
  75. my $suite_name = 'HAMBURGER-IZE HUMANITY';
  76. my $new_suite = $tr->createTestSuite($new_project->{'id'},$suite_name,"Robo-Signed Patriotic People's TestSuite");
  77. is($new_suite->{'name'},$suite_name,"Can create new testsuite");
  78. ok($tr->getTestSuites($new_project->{'id'}),"Can get listing of testsuites for project");
  79. is($tr->getTestSuiteByName($new_project->{'id'},$new_suite->{'name'})->{'name'},$new_suite->{'name'},"Can get suite by name");
  80. is($tr->getTestSuiteByID($new_suite->{'id'})->{'id'},$new_suite->{'id'},"Can get suite by id");
  81. #Test SECTION methods -- roughly analogous to TESTSUITES in TL
  82. my $section_name = 'CARBON LIQUEFACTION';
  83. my $new_section = $tr->createSection($new_project->{'id'},$new_suite->{'id'},$section_name);
  84. is($new_section->{'name'},$section_name,"Can create new section");
  85. ok($tr->getSections($new_project->{'id'},$new_suite->{'id'}),"Can get section listing");
  86. is($tr->getSectionByName($new_project->{'id'},$new_suite->{'id'},$section_name)->{'name'},$section_name,"Can get section by name");
  87. is($tr->getSectionByID($new_section->{'id'})->{'id'},$new_section->{'id'},"Can get new section by id");
  88. my @cids = $tr->sectionNamesToIds($new_project->{'id'},$new_suite->{'id'},$section_name);
  89. is($cids[0],$new_section->{'id'},"sectionNamesToIds returns correct IDs");
  90. isnt(exception {$tr->sectionNamesToIds($new_project->{'id'},$new_suite->{'id'},"No such Section");},undef,"Passing bogus section to sectionNamesToIds throws exception");
  91. #Test CASE methods
  92. my $case_name = 'STROGGIFY POPULATION CENTERS';
  93. my $new_case = $tr->createCase($new_section->{'id'},$case_name);
  94. is($new_case->{'title'},$case_name,"Can create new test case");
  95. my $updated_case = $tr->updateCase($new_case->{'id'}, {'custom_preconds' => 'do some stuff'});
  96. is($updated_case->{'custom_preconds'},'do some stuff',"updateCase works");
  97. my $case_filters = {
  98. 'section_id' => $new_section->{'id'}
  99. };
  100. ok($tr->getCases($new_project->{'id'},$new_suite->{'id'},$case_filters),"Can get case listing");
  101. is($tr->getCaseByName($new_project->{'id'}, $new_suite->{'id'}, $case_name, $case_filters)->{'title'},$case_name,"Can get case by name");
  102. is($tr->getCaseByID($new_case->{'id'})->{'id'},$new_case->{'id'},"Can get case by ID");
  103. #Negative case
  104. $case_filters->{'hokum'} = 'bogus';
  105. isnt(exception {$tr->getCases($new_project->{'id'},$new_suite->{'id'},$case_filters)},undef,"Passing bogus filter croaks");
  106. #Test RUN methods
  107. my $run_name = 'SEND T-1000 INFILTRATION UNITS BACK IN TIME';
  108. my $new_run = $tr->createRun($new_project->{'id'},$new_suite->{'id'},$run_name,"ACQUIRE CLOTHES, BOOTS AND MOTORCYCLE");
  109. is($new_run->{'name'},$run_name,"Can create new run");
  110. ok($tr->getRuns($new_project->{'id'}),"Can get list of all runs in a project");
  111. ok($tr->getRuns($new_project->{'id'},{is_completed => 1,milestone_id => 3}),"Can get list of runs, filtered by milestone ID & completion status");
  112. is($tr->getRunByName($new_project->{'id'},$run_name)->{'name'},$run_name,"Can get run by name");
  113. is($tr->getRunByID($new_run->{'id'})->{'id'},$new_run->{'id'},"Can get run by ID");
  114. #Test MILESTONE methods
  115. my $milestone_name = "Humanity Exterminated";
  116. my $new_milestone = $tr->createMilestone($new_project->{'id'},$milestone_name,"Kill quota reached if not achieved in 5 years",time()+157788000); #It IS a soviet 5-year plan after all :)
  117. is($new_milestone->{'name'},$milestone_name,"Can create new milestone");
  118. ok($tr->getMilestones($new_project->{'id'}),"Can get list of milestones");
  119. is($tr->getMilestoneByName($new_project->{'id'},$milestone_name)->{'name'},$milestone_name,"Can get milestone by name");
  120. is($tr->getMilestoneByID($new_milestone->{'id'})->{'id'},$new_milestone->{'id'},"Can get milestone by ID");
  121. #Test PLAN methods
  122. my $plan_name = "GosPlan";
  123. my $new_plan = $tr->createPlan($new_project->{'id'},$plan_name,"Soviet 5-year agriculture plan to liquidate Kulaks",$new_milestone->{'id'},[{ suite_id => $new_suite->{'id'}, name => "Executing the great plan"}]);
  124. is($new_plan->{'name'},$plan_name,"Can create new plan");
  125. ok($tr->getPlans($new_project->{'id'}),"Can get list of all plans");
  126. ok($tr->getPlans($new_project->{'id'},{is_completed => 1,milestone_id => 3}),"Can get list of plans, filtered by milestone ID & completion status");
  127. my $namePlan = $tr->getPlanByName($new_project->{'id'},$plan_name);
  128. is($namePlan->{'name'},$plan_name,"Can get plan by name");
  129. is($tr->getPlanByID($new_plan->{'id'})->{'id'},$new_plan->{'id'},"Can get plan by ID");
  130. #Get runs per plan, create runs in plan
  131. my $prun = $new_plan->{'entries'}->[0]->{'runs'}->[0];
  132. is($tr->getRunByID($prun->{'id'})->{'name'},"Executing the great plan","Can get child run of plan by ID");
  133. is($tr->getChildRunByName($new_plan,"Executing the great plan", [], 9)->{'id'},$prun->{'id'},"Can find child run of plan by name filtering by testsuite");
  134. is($tr->getChildRunByName($new_plan,"Executing the great plan", [], 8),0,"Can't find child run of plan by name filtering by (bad) testsuite");
  135. SKIP: {
  136. skip("Cannot create configurations programattically in the API like in mocks",2) if !$is_mock;
  137. isnt($tr->getChildRunByName($namePlan,"Executing the great plan",['testConfig']),0,"Getting run by name returns child runs");
  138. is($tr->getChildRunByName($namePlan,"Executing the great plan"),0,"Getting run by name without sufficient configuration data returns child runs");
  139. }
  140. #Test createRunInPlan
  141. my $updatedPlan = $tr->createRunInPlan($new_plan->{'id'},$new_suite->{'id'},'Dynamic Plan Run');
  142. $prun = $updatedPlan->{'runs'}->[0];
  143. is($tr->getRunByID($prun->{'id'})->{'name'},"Dynamic Plan Run","Can get newly created child run of plan by ID");
  144. #Test TEST/RESULT methods
  145. my $tests = $tr->getTests($new_run->{'id'});
  146. ok($tests,"Can get tests");
  147. is($tr->getTestByName($new_run->{'id'},$tests->[0]->{'title'})->{'title'},$tests->[0]->{'title'},"Can get test by name");
  148. is($tr->getTestByID($tests->[0]->{'id'})->{'id'},$tests->[0]->{'id'},"Can get test by ID");
  149. my $resTypes = $tr->getTestResultFields();
  150. my $statusTypes = $tr->getPossibleTestStatuses();
  151. ok($resTypes,"Can get test result fields");
  152. ok($statusTypes,"Can get possible test statuses");
  153. my @status_names = map {$_->{'name'}} @$statusTypes;
  154. my @status_ids = map {$_->{'id'}} @$statusTypes;
  155. my @status_labels = map {$_->{'label'}} @$statusTypes;
  156. my @computed_ids = $tr->statusNamesToIds(@status_names);
  157. my @computed_labels = $tr->statusNamesToLabels(@status_names);
  158. cmp_deeply(\@computed_ids,\@status_ids,"statusNamesToIds functions correctly");
  159. cmp_deeply(\@computed_labels,\@status_labels,"statusNamesToLabels functions correctly");
  160. isnt(exception {$tr->statusNamesToIds(@status_names,'potzrebie'); }, undef, "Passing invalid status name throws exception in statusNamesToIds");
  161. isnt(exception {$tr->statusNamesToLabels(@status_names,'potzrebie'); }, undef, "Passing invalid status name throws exception in statusNamesToLabels");
  162. #TODO make more thorough tests for options, custom options
  163. my $result = $tr->createTestResults($tests->[0]->{'id'},$statusTypes->[0]->{'id'},"REAPER FORCES INBOUND");
  164. ok(defined($result->{'id'}),"Can add test results");
  165. my $results = $tr->getTestResults($tests->[0]->{'id'});
  166. is($results->[0]->{'id'},$result->{'id'},"Can get results for test");
  167. #Bulk add results
  168. $results = $tr->bulkAddResults($new_run->{'id'}, [{ 'test_id' => $tests->[0]->{'id'},'status_id' => $statusTypes->[0]->{'id'}, "comment" => "REAPER FORCES INBOUND" }]);
  169. ok(defined($results->[0]->{'id'}),"Can bulk add test results");
  170. #Test status and assignedto filtering
  171. my $filteredTests = $tr->getTests($new_run->{'id'},[$status_ids[0]]);
  172. is(scalar(@$filteredTests),1,"Test Filtering works: status id positive");
  173. $filteredTests = $tr->getTests($new_run->{'id'},[$status_ids[1]]);
  174. is(scalar(@$filteredTests),0,"Test Filtering works: status id negative");
  175. $filteredTests = $tr->getTests($new_run->{'id'},[$status_ids[0]],[$userlist->[0]->{'id'}]);
  176. is(scalar(@$filteredTests),0,"Test Filtering works: status id positive, user id negative");
  177. $filteredTests = $tr->getTests($new_run->{'id'},undef,[$userlist->[0]->{'id'}]);
  178. is(scalar(@$filteredTests),0,"Test Filtering works: status id undef, user id negative");
  179. #XXX there is no way to programmatically assign things :( so this will remain somewhat uncovered
  180. #Get run summary
  181. my $runs = $tr->getRuns($new_project->{'id'});
  182. my ($summary) = $tr->getRunSummary(@$runs); #I only care about the first one
  183. isnt($summary->{'run_status'},undef,"Can get run statuses correctly");
  184. is($summary->{'run_status'}->{'Passed'},int(!$is_mock),"Gets # of passed cases correctly");
  185. is($summary->{'run_status'}->{'Untested'},int($is_mock),"Gets # of untested cases correctly");
  186. #Get run results
  187. my $run_results = $tr->getRunResults($new_run->{'id'});
  188. is(scalar(@$run_results),3,"Correct # of results returned by getRunResults");
  189. #Test configuration methods
  190. my $configs = $tr->getConfigurations($new_project->{'id'});
  191. my $is_arr = is(Scalar::Util::reftype($configs),'ARRAY',"Can get configurations for a project");
  192. my (@config_names,@config_ids);
  193. if ($is_arr) {
  194. @config_names = map {$_->{'name'}} @$configs;
  195. @config_ids = map {$_->{'id'}} @$configs;
  196. }
  197. my @t_config_ids = $tr->translateConfigNamesToIds($new_project->{'id'},@config_names);
  198. is_deeply(\@config_ids,\@t_config_ids, "Can correctly translate Project names to IDs, and they are correctly sorted");
  199. my $grp = $tr->addConfigurationGroup($new_project->{'id'},"zippy");
  200. is($grp->{'name'},'zippy',"Can add configuration group successfully");
  201. my $fetchedgrp = $tr->getConfigurationGroupByName($new_project->{'id'},'zippy');
  202. is($fetchedgrp->{'id'},$grp->{'id'},"Can get configuration group by name");
  203. my $newgrp = $tr->editConfigurationGroup($grp->{'id'},"doodah");
  204. is($newgrp->{'name'},'doodah',"Can edit configuration group successfully");
  205. my $config = $tr->addConfiguration($grp->{'id'},"potzrebie");
  206. is($config->{'name'},"potzrebie","Can add configuration successfully");
  207. my $newconfig = $tr->editConfiguration($config->{'id'},"poyiut");
  208. is($newconfig->{'name'},"poyiut","Can edit configuration successfully");
  209. ok($tr->deleteConfiguration($config->{'id'}),"Can delete configuration successfully");
  210. ok($tr->deleteConfigurationGroup($grp->{'id'}),"Can delete group successfully");
  211. ############################################################
  212. # TestRail arbitrarily limits many calls to 250 result sets.
  213. # Let's make sure our getters actually get everything.
  214. ############################################################
  215. SKIP: {
  216. skip("Skipping slow tests...", 2) if $ENV{'TESTRAIL_SLOW_TESTS'};
  217. #Check get_plans
  218. foreach my $i (0..$tr->{'global_limit'}) {
  219. $tr->createPlan($new_project->{'id'},$plan_name,"PETE & RE-PIOTR");
  220. }
  221. is(scalar(@{$tr->getPlans($new_project->{'id'})}),($tr->{'global_limit'} + 2),"Can get list of plans beyond ".$tr->{'global_limit'});
  222. #Check get_runs
  223. foreach my $i (0..$tr->{'global_limit'}) {
  224. $tr->createRun($new_project->{'id'},$new_suite->{'id'},$run_name,"ACQUIRE CLOTHES, BOOTS AND MOTORCYCLE");
  225. }
  226. is(scalar(@{$tr->getRuns($new_project->{'id'})}),($tr->{'global_limit'} + 2),"Can get list of runs beyond ".$tr->{'global_limit'});
  227. }
  228. ##########
  229. # Clean up
  230. ##########
  231. #Delete a plan
  232. ok($tr->deletePlan($new_plan->{'id'}),"Can delete plan");
  233. #Delete a milestone
  234. ok($tr->deleteMilestone($new_milestone->{'id'}),"Can delete milestone");
  235. #Delete a run
  236. ok($tr->deleteRun($new_run->{'id'}),"Can delete run");
  237. #Delete a case
  238. ok($tr->deleteCase($new_case->{'id'}),"Can delete Case");
  239. #Delete a section
  240. ok($tr->deleteSection($new_section->{'id'}),"Can delete Section");
  241. #Delete a testsuite
  242. ok($tr->deleteTestSuite($new_suite->{'id'}),"Can delete TestSuite");
  243. #Delete project now that we are done with it
  244. ok($tr->deleteProject($new_project->{'id'}),"Can delete project");
  245. 1;