| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469 |
- package TestRail::API;
- {
- $TestRail::API::VERSION = '0.010';
- }
- =head1 NAME
- TestRail::API - Provides an interface to TestRail's REST api via HTTP
- =head1 SYNOPSIS
- use TestRail::API;
- my $tr = TestRail::API->new($username, $password, $host);
- =head1 DESCRIPTION
- 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.
- It is by no means exhaustively implementing every TestRail API function.
- =cut
- use 5.010;
- use strict;
- use warnings;
- use Carp;
- use Scalar::Util 'reftype';
- use Clone 'clone';
- use Try::Tiny;
- use JSON::XS;
- use HTTP::Request;
- use LWP::UserAgent;
- use Types::Serialiser; #Not necesarily shared by JSON::XS on all platforms
- =head1 CONSTRUCTOR
- =head2 B<new (api_url, user, password)>
- Creates new C<TestRail::API> object.
- =over 4
- =item STRING C<API URL> - base url for your TestRail api server.
- =item STRING C<USER> - Your testRail User.
- =item STRING C<PASSWORD> - Your TestRail password.
- =item BOOLEAN C<DEBUG> - Print the JSON responses from TL with your requests.
- =back
- Returns C<TestRail::API> object if login is successful.
- my $tr = TestRail::API->new('http://tr.test/testrail', 'moo','M000000!');
- =cut
- sub new {
- my ($class,$apiurl,$user,$pass,$debug) = @_;
- $user //= $ENV{'TESTRAIL_USER'};
- $pass //= $ENV{'TESTRAIL_PASSWORD'};
- $debug //= 0;
- my $self = {
- user => $user,
- pass => $pass,
- apiurl => $apiurl,
- debug => $debug,
- testtree => [],
- flattree => [],
- user_cache => [],
- type_cache => [],
- default_request => undef,
- browser => new LWP::UserAgent()
- };
- #Create default request to pass on to LWP::UserAgent
- $self->{'default_request'} = new HTTP::Request();
- $self->{'default_request'}->authorization_basic($user,$pass);
- bless $self, $class;
- return $self;
- }
- =head1 GETTERS
- =head2 B<browser>
- =head2 B<apiurl>
- =head2 B<debug>
- Accessors for these parameters you pass into the constructor, in case you forget.
- =cut
- #EZ access of obj vars
- sub browser {$_[0]->{'browser'}}
- sub apiurl {$_[0]->{'apiurl'}}
- sub debug {$_[0]->{'debug'}}
- #Convenient JSON-HTTP fetcher
- sub _doRequest {
- my ($self,$path,$method,$data) = @_;
- my $req = clone $self->{'default_request'};
- $method //= 'GET';
- $req->method($method);
- $req->url($self->apiurl.'/'.$path);
- warn "$method ".$self->apiurl."/$path" if $self->debug;
- #Data sent is JSON
- my $content = $data ? encode_json($data) : '';
- $req->content($content);
- $req->header( "Content-Type" => "application/json" );
- my $response = $self->browser->request($req);
- if ($response->code == 403) {
- warn "ERROR: Access Denied.";
- return 0;
- }
- if ($response->code != 200) {
- warn "ERROR: Arguments Bad: ".$response->content;
- return 0;
- }
- try {
- return decode_json($response->content);
- } catch {
- if ($response->code == 200 && !$response->content) {
- return 1; #This function probably just returns no data
- } else {
- warn "ERROR: Malformed JSON returned by API.";
- warn $@;
- if (!$self->debug) { #Otherwise we've already printed this, but we need to know if we encounter this
- warn "RAW CONTENT:";
- warn $response->content
- }
- return 0;
- }
- }
- }
- =head1 USER METHODS
- =head2 B<getUsers ()>
- Get all the user definitions for the provided Test Rail install.
- Returns ARRAYREF of user definition HASHREFs.
- =cut
- sub getUsers {
- my $self = shift;
- $self->{'user_cache'} = $self->_doRequest('index.php?/api/v2/get_users');
- return $self->{'user_cache'};
- }
- =head2 B<getUserByID(id)>
- =cut
- =head2 B<getUserByName(name)>
- =cut
- =head2 B<getUserByEmail(email)>
- Get user definition hash by ID, Name or Email.
- Returns user def HASHREF.
- =cut
- #I'm just using the cache for the following methods because it's more straightforward and faster past 1 call.
- sub getUserByID {
- my ($self,$user) = @_;
- $self->getUsers() if (!scalar(@{$self->{'user_cache'}}));
- foreach my $usr (@{$self->{'user_cache'}}) {
- return $usr if $usr->{'id'} == $user;
- }
- return 0;
- }
- sub getUserByName {
- my ($self,$user) = @_;
- $self->getUsers() if (!scalar(@{$self->{'user_cache'}}));
- foreach my $usr (@{$self->{'user_cache'}}) {
- return $usr if $usr->{'name'} eq $user;
- }
- return 0;
- }
- sub getUserByEmail {
- my ($self,$user) = @_;
- $self->getUsers() if (!scalar(@{$self->{'user_cache'}}));
- foreach my $usr (@{$self->{'user_cache'}}) {
- return $usr if $usr->{'email'} eq $user;
- }
- return 0;
- }
- =head1 PROJECT METHODS
- =head2 B<createProject (name, [description,send_announcement])>
- Creates new Project (Database of testsuites/tests).
- Optionally specify an announcement to go out to the users.
- Requires TestRail admin login.
- =over 4
- =item STRING C<NAME> - Desired name of project.
- =item STRING C<DESCRIPTION> (optional) - Description of project. Default value is 'res ipsa loquiter'.
- =item BOOLEAN C<SEND ANNOUNCEMENT> (optional) - Whether to confront users with an announcement about your awesome project on next login. Default false.
- =back
- Returns project definition HASHREF on success, false otherwise.
- $tl->createProject('Widgetronic 4000', 'Tests for the whiz-bang new product', true);
- =cut
- sub createProject {
- my ($self,$name,$desc,$announce) = @_;
- $desc //= 'res ipsa loquiter';
- $announce //= 0;
- my $input = {
- name => $name,
- announcement => $desc,
- show_announcement => $announce ? Types::Serialiser::true : Types::Serialiser::false
- };
- my $result = $self->_doRequest('index.php?/api/v2/add_project','POST',$input);
- return $result;
- }
- =head2 B<deleteProject (id)>
- Deletes specified project by ID.
- Requires TestRail admin login.
- =over 4
- =item STRING C<NAME> - Desired name of project.
- =back
- Returns BOOLEAN.
- $success = $tl->deleteProject(1);
- =cut
- sub deleteProject {
- my ($self,$proj) = @_;
- my $result = $self->_doRequest('index.php?/api/v2/delete_project/'.$proj,'POST');
- return $result;
- }
- =head2 B<getProjects ()>
- Get all available projects
- Returns array of project definition HASHREFs, false otherwise.
- $projects = $tl->getProjects;
- =cut
- sub getProjects {
- my $self = shift;
- my $result = $self->_doRequest('index.php?/api/v2/get_projects');
- #Save state for future use, if needed
- if (!scalar(@{$self->{'testtree'}})) {
- $self->{'testtree'} = $result if $result;
- }
- if ($result) {
- #Note that it's a project for future reference by recursive tree search
- for my $pj (@{$result}) {
- $pj->{'type'} = 'project';
- }
- }
- return $result;
- }
- =head2 B<getProjectByName ($project)>
- Gets some project definition hash by it's name
- =over 4
- =item STRING C<PROJECT> - desired project
- =back
- Returns desired project def HASHREF, false otherwise.
- $projects = $tl->getProjectByName('FunProject');
- =cut
- sub getProjectByName {
- my ($self,$project) = @_;
- confess "No project provided." unless $project;
- #See if we already have the project list...
- my $projects = $self->{'testtree'};
- $projects = $self->getProjects() unless scalar(@$projects);
- #Search project list for project
- for my $candidate (@$projects) {
- return $candidate if ($candidate->{'name'} eq $project);
- }
- return 0;
- }
- =head2 B<getProjectByID ($project)>
- Gets some project definition hash by it's ID
- =over 4
- =item INTEGER C<PROJECT> - desired project
- =back
- Returns desired project def HASHREF, false otherwise.
- $projects = $tl->getProjectByID(222);
- =cut
- sub getProjectByID {
- my ($self,$project) = @_;
- confess "No project provided." unless $project;
- #See if we already have the project list...
- my $projects = $self->{'testtree'};
- $projects = $self->getProjects() unless scalar(@$projects);
- #Search project list for project
- for my $candidate (@$projects) {
- return $candidate if ($candidate->{'id'} eq $project);
- }
- return 0;
- }
- =head1 TESTSUITE METHODS
- =head2 B<createTestSuite (project_id, name, [description])>
- Creates new TestSuite (folder of tests) in the database of test specifications under given project id having given name and details.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of project this test suite should be under.
- =item STRING C<NAME> - Desired name of test suite.
- =item STRING C<DESCRIPTION> (optional) - Description of test suite. Default value is 'res ipsa loquiter'.
- =back
- Returns TS definition HASHREF on success, false otherwise.
- $tl->createTestSuite(1, 'broken tests', 'Tests that should be reviewed');
- =cut
- sub createTestSuite {
- my ($self,$project_id,$name,$details) = @_;
- $details ||= 'res ipsa loquiter';
- my $input = {
- name => $name,
- description => $details
- };
- my $result = $self->_doRequest('index.php?/api/v2/add_suite/'.$project_id,'POST',$input);
- return $result;
- }
- =head2 B<deleteTestSuite (suite_id)>
- Deletes specified testsuite.
- =over 4
- =item INTEGER C<SUITE ID> - ID of testsuite to delete.
- =back
- Returns BOOLEAN.
- $tl->deleteTestSuite(1);
- =cut
- sub deleteTestSuite {
- my ($self,$suite_id) = @_;
- my $result = $self->_doRequest('index.php?/api/v2/delete_suite/'.$suite_id,'POST');
- return $result;
- }
- =head2 B<getTestSuites (project_id)>
- Gets the testsuites for a project
- =over 4
- =item STRING C<PROJECT ID> - desired project's ID
- =back
- Returns ARRAYREF of testsuite definition HASHREFs, 0 on error.
- $suites = $tl->getTestSuites(123);
- =cut
- sub getTestSuites {
- my ($self,$proj) = @_;
- return $self->_doRequest('index.php?/api/v2/get_suites/'.$proj);
- }
- =head2 B<getTestSuiteByName (project_id,testsuite_name)>
- Gets the testsuite that matches the given name inside of given project.
- =over 4
- =item STRING C<PROJECT ID> - ID of project holding this testsuite
- =item STRING C<TESTSUITE NAME> - desired parent testsuite name
- =back
- Returns desired testsuite definition HASHREF, false otherwise.
- $suites = $tl->getTestSuitesByName(321, 'hugSuite');
- =cut
- sub getTestSuiteByName {
- my ($self,$project_id,$testsuite_name) = @_;
- #TODO cache
- my $suites = $self->getTestSuites($project_id);
- return 0 if !$suites; #No suites for project, or no project
- foreach my $suite (@$suites) {
- return $suite if $suite->{'name'} eq $testsuite_name;
- }
- return 0; #Couldn't find it
- }
- =head2 B<getTestSuiteByID (testsuite_id)>
- Gets the testsuite with the given ID.
- =over 4
- =item STRING C<TESTSUITE_ID> - Testsuite ID.
- =back
- Returns desired testsuite definition HASHREF, false otherwise.
- $tests = $tl->getTestSuiteByID(123);
- =cut
- sub getTestSuiteByID {
- my ($self,$testsuite_id) = @_;
- my $result = $self->_doRequest('index.php?/api/v2/get_suite/'.$testsuite_id);
- return $result;
- }
- =head1 SECTION METHODS
- =head2 B<createSection(project_id,suite_id,name,[parent_id])>
- Creates a section.
- =over 4
- =item INTEGER C<PROJECT ID> - Parent Project ID.
- =item INTEGER C<SUITE ID> - Parent Testsuite ID.
- =item STRING C<NAME> - desired section name.
- =item INTEGER C<PARENT ID> (optional) - parent section id
- =back
- Returns new section definition HASHREF, false otherwise.
- $section = $tr->createSection(1,1,'nugs',1);
- =cut
- sub createSection {
- my ($self,$project_id,$suite_id,$name,$parent_id) = @_;
- my $input = {
- name => $name,
- suite_id => $suite_id
- };
- $input->{'parent_id'} = $parent_id if $parent_id;
- my $result = $self->_doRequest('index.php?/api/v2/add_section/'.$project_id,'POST',$input);
- return $result;
- }
- =head2 B<deleteSection (section_id)>
- Deletes specified section.
- =over 4
- =item INTEGER C<SECTION ID> - ID of section to delete.
- =back
- Returns BOOLEAN.
- $tr->deleteSection(1);
- =cut
- sub deleteSection {
- my ($self,$section_id) = @_;
- my $result = $self->_doRequest('index.php?/api/v2/delete_section/'.$section_id,'POST');
- return $result;
- }
- =head2 B<getSections (project_id,suite_id)>
- Gets sections for a given project and suite.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item INTEGER C<SUITE ID> - ID of suite to get sections for.
- =back
- Returns ARRAYREF of section definition HASHREFs.
- $tr->getSections(1,2);
- =cut
- sub getSections {
- my ($self,$project_id,$suite_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_sections/$project_id&suite_id=$suite_id");
- }
- =head2 B<getSectionByID (section_id)>
- Gets desired section.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item INTEGER C<SUITE ID> - ID of suite to get sections for.
- =back
- Returns section definition HASHREF.
- $tr->getSectionByID(344);
- =cut
- sub getSectionByID {
- my ($self,$section_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_section/$section_id");
- }
- =head2 B<getSectionByName (project_id,suite_id,name)>
- Gets desired section.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item INTEGER C<SUITE ID> - ID of suite to get section for.
- =item STRING C<NAME> - name of section to get
- =back
- Returns section definition HASHREF.
- $tr->getSectionByName(1,2,'nugs');
- =cut
- sub getSectionByName {
- my ($self,$project_id,$suite_id,$section_name) = @_;
- my $sections = $self->getSections($project_id,$suite_id);
- foreach my $sec (@$sections) {
- return $sec if $sec->{'name'} eq $section_name;
- }
- return 0;
- }
- =head1 CASE METHODS
- =head2 B<getCaseTypes ()>
- Gets possible case types.
- Returns ARRAYREF of case type definition HASHREFs.
- $tr->getCaseTypes();
- =cut
- sub getCaseTypes {
- my $self = shift;
- $self->{'type_cache'} = $self->_doRequest("index.php?/api/v2/get_case_types") if !$self->type_cache; #We can't change this with API, so assume it is static
- return $self->{'type_cache'};
- }
- =head2 B<getCaseTypeByName (name)>
- Gets case type by name.
- =over 4
- =item STRING C<NAME> - Name of desired case type
- =back
- Returns case type definition HASHREF.
- $tr->getCaseTypeByName();
- =cut
- sub getCaseTypeByName {
- #Useful for marking automated tests, etc
- my ($self,$name) = @_;
- my $types = $self->getCaseTypes();
- foreach my $type (@$types) {
- return $type if $type->{'name'} eq $name;
- }
- return 0;
- }
- =head2 B<createCase(section_id,title,type_id,options,extra_options)>
- Creates a test case.
- =over 4
- =item INTEGER C<SECTION ID> - Parent Project ID.
- =item STRING C<TITLE> - Case title.
- =item INTEGER C<TYPE_ID> - desired test type's ID.
- =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.
- =item HASHREF C<EXTRA OPTIONS> (optional) - contains priority_id, estimate, milestone_id and refs as possible keys. See TestRail API documentation for more info.
- =back
- Returns new case definition HASHREF, false otherwise.
- $custom_opts = {
- preconds => "Test harness installed",
- steps => "Do the needful",
- expected => "cubicle environment transforms into Dali painting"
- };
- $other_opts = {
- priority_id => 4,
- milestone_id => 666,
- estimate => '2m 45s',
- refs => ['TRACE-22','ON-166'] #ARRAYREF of bug IDs.
- }
- $case = $tr->createCase(1,'Do some stuff',3,$custom_opts,$other_opts);
- =cut
- sub createCase {
- my ($self,$section_id,$title,$type_id,$opts,$extras) = @_;
- my $stuff = {
- title => $title,
- type_id => $type_id
- };
- #Handle sort of optional but baked in options
- if (defined($extras) && reftype($extras) eq 'HASH') {
- $stuff->{'priority_id'} = $extras->{'priority_id'} if defined($extras->{'priority_id'});
- $stuff->{'estimate'} = $extras->{'estimate'} if defined($extras->{'estimate'});
- $stuff->{'milestone_id'} = $extras->{'milestone_id'} if defined($extras->{'milestone_id'});
- $stuff->{'refs'} = join(',',@{$extras->{'refs'}}) if defined($extras->{'refs'});
- }
- #Handle custom fields
- if (defined($opts) && reftype($opts) eq 'HASH') {
- foreach my $key (keys(%$opts)) {
- $stuff->{"custom_$key"} = $opts->{$key};
- }
- }
- my $result = $self->_doRequest("index.php?/api/v2/add_case/$section_id",'POST',$stuff);
- return $result;
- }
- =head2 B<deleteCase (case_id)>
- Deletes specified section.
- =over 4
- =item INTEGER C<CASE ID> - ID of case to delete.
- =back
- Returns BOOLEAN.
- $tr->deleteCase(1324);
- =cut
- sub deleteCase {
- my ($self,$case_id) = @_;
- my $result = $self->_doRequest("index.php?/api/v2/delete_case/$case_id",'POST');
- return $result;
- }
- =head2 B<getCases (project_id,suite_id,section_id)>
- Gets cases for provided section.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item INTEGER C<SUITE ID> - ID of parent suite.
- =item INTEGER C<SECTION ID> - ID of parent section
- =back
- Returns ARRAYREF of test case definition HASHREFs.
- $tr->getCases(1,2,3);
- =cut
- sub getCases {
- my ($self,$project_id,$suite_id,$section_id) = @_;
- my $url = "index.php?/api/v2/get_cases/$project_id&suite_id=$suite_id";
- $url .= "§ion_id=$section_id" if $section_id;
- return $self->_doRequest($url);
- }
- =head2 B<getCaseByName (project_id,suite_id,section_id,name)>
- Gets case by name.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item INTEGER C<SUITE ID> - ID of parent suite.
- =item INTEGER C<SECTION ID> - ID of parent section.
- =item STRING <NAME> - Name of desired test case.
- =back
- Returns test case definition HASHREF.
- $tr->getCaseByName(1,2,3,'nugs');
- =cut
- sub getCaseByName {
- my ($self,$project_id,$suite_id,$section_id,$name) = @_;
- my $cases = $self->getCases($project_id,$suite_id,$section_id);
- foreach my $case (@$cases) {
- return $case if $case->{'title'} eq $name;
- }
- return 0;
- }
- =head2 B<getCaseByID (case_id)>
- Gets case by ID.
- =over 4
- =item INTEGER C<CASE ID> - ID of case.
- =back
- Returns test case definition HASHREF.
- $tr->getCaseByID(1345);
- =cut
- sub getCaseByID {
- my ($self,$case_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_case/$case_id");
- }
- =head1 RUN METHODS
- =head2 B<createRun (project_id)>
- Create a run.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item INTEGER C<SUITE ID> - ID of suite to base run on
- =item STRING C<NAME> - Name of run
- =item STRING C<DESCRIPTION> (optional) - Description of run
- =item INTEGER C<MILESTONE_ID> (optional) - ID of milestone
- =item INTEGER C<ASSIGNED_TO_ID> (optional) - User to assign the run to
- =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.
- =back
- Returns run definition HASHREF.
- $tr->createRun(1,1345,'RUN AWAY','SO FAR AWAY',22,3,[3,4,5,6]);
- =cut
- #If you pass an array of case ids, it implies include_all is false
- sub createRun {
- my ($self,$project_id,$suite_id,$name,$desc,$milestone_id,$assignedto_id,$case_ids) = @_;
- my $stuff = {
- suite_id => $suite_id,
- name => $name,
- description => $desc,
- milestone_id => $milestone_id,
- assignedto_id => $assignedto_id,
- include_all => $case_ids ? Types::Serialiser::false : Types::Serialiser::true,
- case_ids => $case_ids
- };
- my $result = $self->_doRequest("index.php?/api/v2/add_run/$project_id",'POST',$stuff);
- return $result;
- }
- =head2 B<deleteRun (run_id)>
- Deletes specified run.
- =over 4
- =item INTEGER C<RUN ID> - ID of run to delete.
- =back
- Returns BOOLEAN.
- $tr->deleteRun(1324);
- =cut
- sub deleteRun {
- my ($self,$suite_id) = @_;
- my $result = $self->_doRequest("index.php?/api/v2/delete_run/$suite_id",'POST');
- return $result;
- }
- =head2 B<getRuns (project_id)>
- Get all runs for specified project.
- =over 4
- =item INTEGER C<PROJECT_ID> - ID of parent project
- =back
- Returns ARRAYREF of run definition HASHREFs.
- $allRuns = $tr->getRuns(6969);
- =cut
- sub getRuns {
- my ($self,$project_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_runs/$project_id");
- }
- =head2 B<getRunByName (project_id,name)>
- Gets run by name.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item STRING <NAME> - Name of desired run.
- =back
- Returns run definition HASHREF.
- $tr->getRunByName(1,'gravy');
- =cut
- sub getRunByName {
- my ($self,$project_id,$name) = @_;
- my $runs = $self->getRuns($project_id);
- foreach my $run (@$runs) {
- return $run if $run->{'name'} eq $name;
- }
- return 0;
- }
- =head2 B<getRunByID (run_id)>
- Gets run by ID.
- =over 4
- =item INTEGER C<RUN ID> - ID of desired run.
- =back
- Returns run definition HASHREF.
- $tr->getRunByID(7779311);
- =cut
- sub getRunByID {
- my ($self,$run_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_run/$run_id");
- }
- =head1 PLAN METHODS
- =head2 B<createPlan (project_id,name,description,milestone_id,entries)>
- Create a run.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item STRING C<NAME> - Name of plan
- =item STRING C<DESCRIPTION> (optional) - Description of plan
- =item INTEGER C<MILESTONE_ID> (optional) - ID of milestone
- =item ARRAYREF C<ENTRIES> (optional) - New Runs to initially populate the plan with -- See TestLink API documentation for more advanced inputs here.
- =back
- Returns test plan definition HASHREF, or false on failure.
- $entries = {
- suite_id => 345,
- include_all => Types::Serialiser::true,
- assignedto_id => 1
- }
- $tr->createPlan(1,'Gosplan','Robo-Signed Soviet 5-year plan',22,$entries);
- =cut
- sub createPlan {
- my ($self,$project_id,$name,$desc,$milestone_id,$entries) = @_;
- my $stuff = {
- name => $name,
- description => $desc,
- milestone_id => $milestone_id,
- entries => $entries
- };
- my $result = $self->_doRequest("index.php?/api/v2/add_plan/$project_id",'POST',$stuff);
- return $result;
- }
- =head2 B<deletePlan (plan_id)>
- Deletes specified plan.
- =over 4
- =item INTEGER C<PLAN ID> - ID of plan to delete.
- =back
- Returns BOOLEAN.
- $tr->deletePlan(8675309);
- =cut
- sub deletePlan {
- my ($self,$plan_id) = @_;
- my $result = $self->_doRequest("index.php?/api/v2/delete_plan/$plan_id",'POST');
- return $result;
- }
- =head2 B<getPlans (project_id)>
- Deletes specified plan.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =back
- Returns ARRAYREF of plan definition HASHREFs.
- $tr->getPlans(8);
- =cut
- sub getPlans {
- my ($self,$project_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_plans/$project_id");
- }
- =head2 B<getPlanByName (project_id,name)>
- Gets specified plan by name.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item STRING C<NAME> - Name of test plan.
- =back
- Returns plan definition HASHREF.
- $tr->getPlanByName(8,'GosPlan');
- =cut
- sub getPlanByName {
- my ($self,$project_id,$name) = @_;
- my $plans = $self->getPlans($project_id);
- foreach my $plan (@$plans) {
- return $plan if $plan->{'name'} eq $name;
- }
- return 0;
- }
- =head2 B<getPlanByID (plan_id)>
- Gets specified plan by ID.
- =over 4
- =item INTEGER C<PLAN ID> - ID of plan.
- =back
- Returns plan definition HASHREF.
- $tr->getPlanByID(2);
- =cut
- sub getPlanByID {
- my ($self,$plan_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_plan/$plan_id");
- }
- =head1 MILESTONE METHODS
- =head2 B<createMilestone (project_id,name,description,due_on)>
- Create a milestone.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item STRING C<NAME> - Name of milestone
- =item STRING C<DESCRIPTION> (optional) - Description of milestone
- =item INTEGER C<DUE_ON> - Date at which milestone should be completed. Unix Timestamp.
- =back
- Returns milestone definition HASHREF, or false on failure.
- $tr->createMilestone(1,'Patriotic victory of world perlism','Accomplish by Robo-Signed Soviet 5-year plan',time()+157788000);
- =cut
- sub createMilestone {
- my ($self,$project_id,$name,$desc,$due_on) = @_;
- my $stuff = {
- name => $name,
- description => $desc,
- due_on => $due_on # unix timestamp
- };
- my $result = $self->_doRequest("index.php?/api/v2/add_milestone/$project_id",'POST',$stuff);
- return $result;
- }
- =head2 B<deleteMilestone (milestone_id)>
- Deletes specified milestone.
- =over 4
- =item INTEGER C<MILESTONE ID> - ID of milestone to delete.
- =back
- Returns BOOLEAN.
- $tr->deleteMilestone(86);
- =cut
- sub deleteMilestone {
- my ($self,$milestone_id) = @_;
- my $result = $self->_doRequest("index.php?/api/v2/delete_milestone/$milestone_id",'POST');
- return $result;
- }
- =head2 B<getMilestones (project_id)>
- Get milestones for some project.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =back
- Returns ARRAYREF of milestone definition HASHREFs.
- $tr->getMilestones(8);
- =cut
- sub getMilestones {
- my ($self,$project_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_milestones/$project_id");
- }
- =head2 B<getMilestoneByName (project_id,name)>
- Gets specified milestone by name.
- =over 4
- =item INTEGER C<PROJECT ID> - ID of parent project.
- =item STRING C<NAME> - Name of milestone.
- =back
- Returns milestone definition HASHREF.
- $tr->getMilestoneByName(8,'whee');
- =cut
- sub getMilestoneByName {
- my ($self,$project_id,$name) = @_;
- my $milestones = $self->getMilestones($project_id);
- foreach my $milestone (@$milestones) {
- return $milestone if $milestone->{'name'} eq $name;
- }
- return 0;
- }
- =head2 B<getMilestoneByID (plan_id)>
- Gets specified milestone by ID.
- =over 4
- =item INTEGER C<MILESTONE ID> - ID of milestone.
- =back
- Returns milestione definition HASHREF.
- $tr->getMilestoneByID(2);
- =cut
- sub getMilestoneByID {
- my ($self,$milestone_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_milestone/$milestone_id");
- }
- =head1 TEST METHODS
- =head2 B<getTests (run_id)>
- Get tests for some run.
- =over 4
- =item INTEGER C<RUN ID> - ID of parent run.
- =back
- Returns ARRAYREF of test definition HASHREFs.
- $tr->getTests(8);
- =cut
- sub getTests {
- my ($self,$run_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_tests/$run_id");
- }
- =head2 B<getTestByName (run_id,name)>
- Gets specified test by name.
- =over 4
- =item INTEGER C<RUN ID> - ID of parent run.
- =item STRING C<NAME> - Name of milestone.
- =back
- Returns test definition HASHREF.
- $tr->getTestByName(36,'wheeTest');
- =cut
- sub getTestByName {
- my ($self,$run_id,$name) = @_;
- my $tests = $self->getTests($run_id);
- foreach my $test (@$tests) {
- return $test if $test->{'title'} eq $name;
- }
- return 0;
- }
- =head2 B<getTestByID (test_id)>
- Gets specified test by ID.
- =over 4
- =item INTEGER C<TEST ID> - ID of test.
- =back
- Returns test definition HASHREF.
- $tr->getTestByID(222222);
- =cut
- sub getTestByID {
- my ($self,$test_id) = @_;
- return $self->_doRequest("index.php?/api/v2/get_test/$test_id");
- }
- =head2 B<getTestResultFields()>
- Gets custom fields that can be set for tests.
- Returns ARRAYREF of result definition HASHREFs.
- =cut
- sub getTestResultFields {
- my $self = shift;
- return $self->_doRequest('index.php?/api/v2/get_result_fields');
- }
- =head2 B<getPossibleTestStatuses()>
- Gets all possible statuses a test can be set to.
- Returns ARRAYREF of status definition HASHREFs.
- =cut
- sub getPossibleTestStatuses {
- my $self = shift;
- return $self->_doRequest('index.php?/api/v2/get_statuses');
- }
- =head2 B<createTestResults(test_id,status_id,comment,options,custom_options)>
- Creates a result entry for a test.
- Returns result definition HASHREF.
- $options = {
- elapsed => '30m 22s',
- defects => ['TSR-3','BOOM-44'],
- version => '6969'
- };
- $custom_options = {
- step_results => [
- {
- content => 'Step 1',
- expected => "Bought Groceries",
- actual => "No Dinero!",
- status_id => 2
- },
- {
- content => 'Step 2',
- expected => 'Ate Dinner',
- actual => 'Went Hungry',
- status_id => 2
- }
- ]
- };
- $res = $tr->createTestResults(1,2,'Test failed because it was all like WAAAAAAA when I poked it',$options,$custom_options);
- =cut
- sub createTestResults {
- my ($self,$test_id,$status_id,$comment,$opts,$custom_fields) = @_;
- my $stuff = {
- status_id => $status_id,
- comment => $comment
- };
- #Handle options
- if (defined($opts) && reftype($opts) eq 'HASH') {
- $stuff->{'version'} = defined($opts->{'version'}) ? $opts->{'version'} : undef;
- $stuff->{'elapsed'} = defined($opts->{'elapsed'}) ? $opts->{'elapsed'} : undef;
- $stuff->{'defects'} = defined($opts->{'defects'}) ? join(',',@{$opts->{'defects'}}) : undef;
- $stuff->{'assignedto_id'} = defined($opts->{'assignedto_id'}) ? $opts->{'assignedto_id'} : undef;
- }
- #Handle custom fields
- if (defined($custom_fields) && reftype($custom_fields) eq 'HASH') {
- foreach my $field (keys(%$custom_fields)) {
- $stuff->{"custom_$field"} = $custom_fields->{$field};
- }
- }
- return $self->_doRequest("index.php?/api/v2/add_result/$test_id",'POST',$stuff);
- }
- =head2 B<getTestResults(test_id,limit)>
- Get the recorded results for desired test, limiting output to 'limit' entries.
- =over 4
- =item INTEGER C<TEST_ID> - ID of desired test
- =item POSITIVE INTEGER C<LIMIT> - provide no more than this number of results.
- =back
- Returns ARRAYREF of result definition HASHREFs.
- =cut
- sub getTestResults {
- my ($self,$test_id,$limit) = @_;
- my $url = "index.php?/api/v2/get_results/$test_id";
- $url .= "&limit=$limit" if defined($limit);
- return $self->_doRequest($url);
- }
- =head2 B<buildStepResults(content,expected,actual,status_id)>
- Convenience method to build the stepResult hashes seen in the custom options for getTestResults.
- =cut
- #Convenience method for building stepResults
- sub buildStepResults {
- my ($content,$expected,$actual,$status_id) = @_;
- return {
- content => $content,
- expected => $expected,
- actual => $actual,
- status_id => $status_id
- };
- }
- 1;
- __END__
- =head1 SEE ALSO
- L<Test::More>
- L<HTTP::Request>
- L<LWP::UserAgent>
- L<JSON::XS>
- L<http://docs.gurock.com/testrail-api2/start>
- =head1 AUTHOR
- George Baugh <teodesian@cpan.org>
- =head1 SPECIAL THANKS
- Thanks to cPanel Inc, for graciously funding the creation of this module.
- =head1 COPYRIGHT AND LICENSE
- This software is copyright (c) 2014 by George S. Baugh.
- This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.
|