The present disclosure relates to source code management, and more particularly to automatically checking in source code to a plurality of code branches of a software application.
A rising trend in software development is to change from the more traditional software development methodology known as “waterfall” to so-called “agile” software development methods. Due to this paradigm shift, product development lifecycle span is changing from months to weeks. Agile software development is known to be very accommodating of customer needs, and to adapt to changes in customer requirements, because it enables continuous delivery of a series of releases. The waterfall method, in contrast, delivers fewer releases that are spaced further apart in time. However, use of agile software development it also poses a number of challenges.
According to one aspect of the present disclosure, a computer-implemented method is disclosed which detects that new source code linked to a defect identifier has been checked in to a source code repository for a given one of a plurality of code branches of a software application. The new source code comprises new application source code and new test case source code. The defect identifier identifies a defect in the software application that the new application source code resolves and the new test case source code tests. The new source code is automatically checked in to additional ones of the plurality of code branches based on the defect identifier. For each additional code branch into which the new source code is automatically checked in:
According to another aspect of the present disclosure, a computing device is disclosed which comprises a memory circuit and a processing circuit operatively connected the memory circuit. The processing circuit is configured to detect that new source code linked to a defect identifier has been checked in to a source code repository for a given one of a plurality of code branches of a software application. The new source code comprises new application source code and new test case source code. The defect identifier identifies a defect in the software application that the new application source code resolves and the new test case source code tests. The processing circuit is further configured to automatically check in the new source code to additional ones of the plurality of code branches based on the defect identifier. The processing circuit is further configured to, for each additional code branch into which the new source code is automatically checked in:
According to another aspect of the present disclosure, a computer program product is disclosed which comprises a computer readable storage medium having computer readable program code embodied therewith. The computer readable program code comprises computer readable program code configured to detect that new source code linked to a defect identifier has been checked in to a source code repository for a given one of a plurality of code branches of a software application. The new source code comprises new application source code and new test case source code. The defect identifier identifies a defect in the software application that the new application source code resolves and the new test case source code tests. The computer readable program code comprises computer readable program code configured to automatically check in the new source code to additional ones of the plurality of code branches based on the defect identifier. The computer readable program code further comprises computer readable program code configured to, for each additional code branch into which the new source code is automatically checked in:
Of course, the present disclosure is not limited to the above features and advantages. Indeed, those skilled in the art will recognize additional features and advantages upon reading the following detailed description, and upon viewing the accompanying drawings.
Aspects of the present disclosure are illustrated by way of example and are not limited by the accompanying figures with like references indicating like elements.
As will be appreciated by one skilled in the art, aspects of the present disclosure may be illustrated and described herein in any of a number of patentable classes or context including any new and useful process, machine, manufacture, or composition of matter, or any new and useful improvement thereof. Accordingly, aspects of the present disclosure may be implemented entirely as hardware, entirely as software (including firmware, resident software, micro-code, etc.) or combining software and hardware implementation that may all generally be referred to herein as a “circuit,” “module,” “component,” or “system.” Furthermore, aspects of the present disclosure may take the form of a computer program product embodied in one or more computer readable media having computer readable program code embodied thereon.
Any combination of one or more computer readable media may be utilized. The computer readable media may be a computer readable signal medium or a computer readable storage medium. A computer readable storage medium may be, for example, but not limited to, an electronic, magnetic, optical, electromagnetic, or semiconductor system, apparatus, or device, or any suitable combination of the foregoing. More specific examples (a non-exhaustive list) of the computer readable storage medium would include the following: a portable computer diskette, a hard disk, a random access memory (RAM), a read-only memory (ROM), an erasable programmable read-only memory (EPROM or Flash memory), an appropriate optical fiber with a repeater, a portable compact disc read-only memory (CD-ROM), an optical storage device, a magnetic storage device, or any suitable combination of the foregoing. In the context of this document, a computer readable storage medium may be any tangible medium that can contain, or store a program for use by or in connection with an instruction execution system, apparatus, or device.
A computer readable signal medium may include a propagated data signal with computer readable program code embodied therein, for example, in baseband or as part of a carrier wave. Such a propagated signal may take any of a variety of forms, including, but not limited to, electro-magnetic, optical, or any suitable combination thereof. A computer readable signal medium may be any computer readable medium that is not a computer readable storage medium and that can communicate, propagate, or transport a program for use by or in connection with an instruction execution system, apparatus, or device. Program code embodied on a computer readable signal medium may be transmitted using any appropriate medium, including but not limited to wireless, wireline, optical fiber cable, RF, etc., or any suitable combination of the foregoing.
Computer program code for carrying out operations for aspects of the present disclosure may be written in any combination of one or more programming languages, including an object oriented programming language such as Java, Scala, Smalltalk, Eiffel, JADE, Emerald, C++, C#, VB.NET, Python or the like, conventional procedural programming languages, such as the “C” programming language, Visual Basic, Fortran 2003, Perl, COBOL 2002, PHP, ABAP, dynamic programming languages such as Python, Ruby and Groovy, or other programming languages. The program code may execute entirely on the user's computer, partly on the user's computer, as a stand-alone software package, partly on the user's computer and partly on a remote computer or entirely on the remote computer or server. In the latter scenario, the remote computer may be connected to the user's computer through any type of network, including a local area network (LAN) or a wide area network (WAN), or the connection may be made to an external computer (for example, through the Internet using an Internet Service Provider) or in a cloud computing environment or offered as a service such as a Software as a Service (SaaS).
Aspects of the present disclosure are described herein with reference to flowchart illustrations and/or block diagrams of methods, apparatuses (systems) and computer program products according to embodiments of the disclosure. It will be understood that each block of the flowchart illustrations and/or block diagrams, and combinations of blocks in the flowchart illustrations and/or block diagrams, can be implemented by computer program instructions. These computer program instructions may be provided to a processor of a general purpose computer, special purpose computer, or other programmable data processing apparatus to produce a machine, such that the instructions, which execute via the processor of the computer or other programmable instruction execution apparatus, create a mechanism for implementing the functions/acts specified in the flowchart and/or block diagram block or blocks.
These computer program instructions may also be stored in a non-transitory computer readable medium that when executed can direct a computer, other programmable data processing apparatus, or other devices to function in a particular manner, such that the instructions when stored in the computer readable medium produce an article of manufacture including instructions which when executed, cause a computer to implement the function/act specified in the flowchart and/or block diagram block or blocks. The computer program instructions may also be loaded onto a computer, other programmable instruction execution apparatus, or other devices to cause a series of operational steps to be performed on the computer, other programmable apparatuses or other devices to produce a computer implemented process such that the instructions which execute on the computer or other programmable apparatus provide processes for implementing the functions/acts specified in the flowchart and/or block diagram block or blocks.
The present disclosure describes techniques for automatically checking in source code (including application source code and test case source code) to a plurality of code branches of a software application, while also monitoring code coverage levels to ensure that code coverage of the software application is maintained at an acceptable level. If code coverage decreases below an acceptable threshold, then a notification can be transmitted to indicate the deficiency. These techniques can be used despite the disparate software development tools from different vendors that are often used by software developers.
When agile software development is used, short release cycles are implemented, which means that there can be multiple releases of a product to be maintained by developers. These multiple releases can take the form of code branches in a source code repository. This increase in code branches correspondingly increases the likelihood that a software defect (i.e., bugs) from a released version as reported by a customer may apply to previous or subsequent releases. This places a considerable burden on developers to handle software defects and updates effectively. The techniques discussed herein automatically identify impacted code branches and promote source code changes among affected code branches to more effectively handle software defect resolution.
Referring now to
The defect management system 14 is used to track defects in the various code branches of the software application. For a given defect, the defect management system 14 can indicate a defect identifier (ID), a defect description, an indication of what code branches the defect is present in, and an indication of whether the defect has been resolved with new source code. Some example defect management systems include RTC and JIRA.
The build system 16 is used to compile the various code branches of the software application, including application and test case source code, from the source code repository 12. The build system 16 may also be used to execute tests cases to test the software application (e.g., using compiled test cast source code). To test the application, the build system may also use data from the software testing management platform 18. Some example build systems include TEAM CITY and MAVEN.
The software testing management platform 18 is used to track the results of executed test cases for the software application (e.g., whether tests have passed or failed for a given code branch). An example software testing management platform 18 includes QUALITY CENTER from HP.
Software developers 24 can interact with the source code repository 12 and defect management system 14 via their respective computing devices 28 to track software defects, and fix them by checking in new source code. Quality assurance (QA) engineers 26 can interact with the build system 16 and software testing management platform 18 via their respective computing devices 28 to ensure that a given code branch of the software application is running properly and has been adequately tested.
In a typical software development environment, the components 12-18 are executed on different servers. Quite commonly they are provided by different vendors and lack out of the box integration. The patch integrator 20 can provide missing integration between these components.
The patch integrator 20 communicates with the various components 12, 14, 16, 18 to detect when new source code is checked in to the source code repository 12 for a given one of a plurality of code branches of a software application, and to automatically check in that new source code (including application source code and test case source code) to other ones of the plurality of code branches, as will be described below. The patch integrator 20 is operatively connected to a repository 22 of historic code coverage data that is indicative of the extent to which various code branches of a software application are tested by their corresponding test case source code at various points in time. The patch integrator may also maintain an exclusion list that identifies code branches into which source code should not be automatically checked in.
As used herein, the term “change set” will be used to refer to the files of application source code of a given code branch that are changed by a check in, or that are added by a check in. A given change set will include one or more files that are updated versions of files in a code branch. The new versions of existing files will be called “updated files.” Also, the change set may also include new files that are added to the code branch and that were not already present in a code branch. These new files will be called “added files” because they are being added instead of updated.
After creating application source code that resolves the defect in the given code branch, the developer checks in that new application source code into the source code repository 12 via the same or another computing device 28 (206). The new application source code is tagged with the defect ID to indicate that it resolves the defect identified by the defect ID. The developer 24 also checks in new test case source code into the source code repository 12 via the developer computing device 24 (208). The new test case source code tests the new application source code, and is also tagged with the defect ID. The new test case source code can be used to check for regression errors to ensure that the defect identified with the defect ID does not get reintroduced into the code branch of the software application in the future. The developer also transmits a notification to the defect management system 14 that the defect identified by the defect ID has been fixed (210). The notification of 210 can be transmitted via the same or another developer computing device 28.
The developer (or perhaps a quality assurance “QA” engineer) creates records for new test cases in the software testing management platform 18 (212). The new records identify tests in the new test case source code, and are also tagged with the defect ID. The software testing management platform 18 can use these records to store test results for the various tests.
The build system 16 detects that new source code has been checked in to the source code repository 12 for the given code branch (214), and compiles the source code for the given code branch (216), including the application source code and the test case source code. The build system 16 executes tests for the given code branch, and determines code coverage data for the given code branch (218), which is then reported to the patch integrator 20 (220). The patch integrator 20 stores the code coverage data in its code coverage repository 22 (222).
The patch integrator 20 determines that source code has been checked into the given code branch (224). This may be performed using Application Programming Interfaces (APIs) of the source code repository 12. The APIs could be used to obtain metadata of the source code (e.g., the defect ID that the code is associated with). The patch integrator 20 also determines a subset of the impacted code branches that the new source code should be merged into (226). This determination is performed at least partially based on the defect ID corresponding to the new source code.
Once the subset of code branches is determined, the patch integrator 20 automatically checks in the new source code into the subset of additional code branches (228). Each file in the change set that is an added file will be added to the subset of additional code branches. Each file in the change set that is an updated file will be updated in the subset of additional code branches. This updating could include merging changes into the existing file in the code branch, or could include replacing the existing file in the code branch.
The patch integrator 20 may also be configured to generate, based on the first and second code coverage metrics, a dashboard that displays historical code coverage data for the software application. The dashboard could be shown on an electronic display, for example, and may in some embodiments be accessible remotely (e.g., via a world wide web-based interface). The dashboard could be used to look up code current coverage data for a code branch and/or to view graphs of how code coverage has changed over time for a code branch, for example. In the same or another embodiment, the dashboard could be used to indicate a quantity of automatic updates that have been performed and/or a quantity of documented software defects that currently exist in each code branch. Thus, in some embodiments the dashboard could be used to indicate which check-ins improve code coverage numbers for code branches of a software application. In the same or other embodiments, the dashboard could be used to provide a development team with a list of defects reported by customers and their applicability in remaining versions of product—defects which could be automatically fixed by the patch integrator 20 across many code branches (228) once fixed in a single code branch (206).
Items 216-222 are repeated for each additional code branch in the subset of additional code branches (230). Thus, for each additional code branch, the application and test source code (including the new source code that has been automatically checked in) is compiled, testing is performed against the new build to determine code coverage data, and the patch integrator 20 stores that code coverage data in the code coverage repository 22. A code coverage analysis is performed for the given code branch and the subset of additional code branches, and if needed a deficiency notification is transmitted (232). Also, the software testing management platform 18 is automatically updated to reflect that additional test cases, which correspond to the new test case source code, are available for the additional ones of the plurality of code branches (234).
In one or more embodiments, performance of items 226-234 is contingent on the tests performed in item 218 being successful. If one or more of the tests failed, that could be indicative of a regression error being introduced in the new application source code checked in at 206. If one or more regression errors were present, it may be undesirable to check in the new application source code into other code branches, as that could cause the regression error to be propagated among the code branches. In one or more embodiments, the build of item 216 is a so-called “smoke build”, and the new application source code and test case source code are not fully committed to the given code branch until the tests performed in item 218 are determined to be successful.
When checking in source code to the source code repository, developers may also check in resource files (e.g., images or configuration files) and/or binary files (e.g., dependencies such as DLLs) along with the new source code. In one or more embodiments, the patch integrator 20 is configured to check in these companion files along with the new source code into the subset of additional code branches.
Although the items 22-234 are shown in a particular order in
Items 226 and 232 of
In block 266, a determination is made of whether the defect ID has already been fixed for the code branch. If the defect ID has already been fixed for the code branch (a “Yes” to block 266), then the code branch is excluded from the subset (block 262). Otherwise, if the defect ID has not already been fixed for the code branch (a “No” to block 266), then a determination is made of whether a previous version of any of the updated file(s) is present in the code branch (block 268). If no previous version of any of the updated files is present in the code branch (a “No” to block 268), then the code branch is excluded from the subset (block 262). Otherwise, if a previous version of any of the updated files is present in the code branch (a “Yes” to block 268), then the code branch is included in the subset (block 270).
Thus, in some embodiments of the method 100 of
In some such embodiments, the determining of which additional code branches that the new source code will be automatically checked in to comprises determining that the new source code should not be automatically checked in to one of the plurality of code branches if none of the files in the set of files are present in the code branch (a “No” to block 268 in
Also, in some embodiments, the determining of which additional code branches the new source code will be automatically checked in to is further based on an exclusion list which indicates code branches into which application source code should not be automatically checked in (see block 266 in
Also, how the automatic check in is performed may depend on whether a file is an updated file or an added file. In some embodiments of the method 100, the method 100 includes determining whether the new application source code includes any new files that were not included in the given code branch before the check in of the new source code (i.e., any “added files”). In such embodiments, the automatic check in of the new source code to additional ones of the plurality of code branches based on the defect identifier (block 104) comprises:
Performing a merge instead of a file replacement may be advantageous in certain scenarios. For example, it is possible that only a very small portion of a rather large file may be updated. Also, the file in question may vary between branches to some extent (e.g., there may be different dependencies or different formatting preferences being used). In such instances, merging the changes into the existing file may be preferable to simply replacing the file.
As with
The “code coverage analysis” of item 232 of
As part of the code coverage analysis, a difference is determined between the first and second code coverage metrics (a “code coverage delta”). If the code coverage delta indicates that the second code coverage metric is greater than the first code coverage metric, then the patch integrator 20 assumes that a code coverage of the code branch being analyzed is sufficient. However, if the delta code coverage indicates that second code coverage metric is less than the first code coverage metric by an amount that exceeds a predefined threshold, then the patch integrator determines that the code branch being analyzed suffers from a code coverage deficiency, and a notification is transmitted (e.g., to QA engineer 26) to provide a notification.
In some embodiments, the predefined code coverage threshold is reconfigurable in the patch integrator 20. For example, for some entities a code coverage of 80% is considered acceptable, whereas for other entities that number may be too high or too low. Having reconfigurable code coverage thresholds can accommodate varying requirements among users of the patch integrator 20.
In some embodiments, the code coverage metric is limited to the change set. In such embodiments, the method 100 includes determining the change set comprising files in the given code branch that are either updated or added by the check in of the new source code. In such embodiments, neither of the first and second code coverage metrics reflects code coverage of files that are not present in the change set. To elaborate, the first code coverage metric reflects a code coverage of files of the change set before the automatic check in and does not reflect a code coverage of files outside of the change set. Additionally, the second code coverage metric reflects a code coverage of files of the change set after the automatic check in and does not reflect a code coverage of files outside of the change set. Thus, if there are 100 files of source code in a code branch and the change set includes 2 files, the code coverage data does not consider the remaining 98 files in the code branch. Of course, other embodiments could also be possible in which all source code was considered in determining the code coverage metrics.
Code coverage data may also be expressed at the class level (
Referring to
The first and second code coverage metrics of
New application source may be checked in to the source code repository 12, with the new application source code being tagged with a defect ID, but no corresponding test case source code is checked in for the defect ID. In some embodiments, when that occurs the patch integrator 20 detects that additional source code linked to an additional defect identifier has been checked into the source code repository for a particular one of the plurality of code branches, wherein the additional source code comprises application source code but does not include any test case source code. Based on this, the patch integrator 20 automatically checks in the additional source code to additional ones of the plurality of code branches based on the additional defect identifier, and queries the software testing management platform 18 to determine if any test case source code is available for the additional defect identifier. If no indication is received from the software testing management platform 18 that indicates that test case source code is available for the additional defect identifier after either a predetermined amount of time or a predetermined amount of querying, the patch integrator 20 transmits a notification that indicates a test case deficiency for the additional defect identifier (e.g., to QA engineer 26). An example of this is shown in
The method 400 can be used to determine which files in the v3.4 code branch will be updated. The files of source code (i.e., “source code files”) that are updated by the new application source code in a first code branch (here the v3.3 code branch) are identified (block 402). These are source code files that previously existed in the v3.3 code branch that are updated by the new application source code. Assume that there are 100 source code files, 3 of which were updated by a check in to the v3.3 branch. The analysis of block 402 would identify those three source code files.
For each identified source code file, a determination is made of whether a corresponding file in the second code branch (here the v3.4 branch) needs an update by comparing the existing source file in the v3.4 code branch to the updated file in the v3.3 branch, and determining a delta change (block 404). It may occur that one of the 3 files has already been updated. If the delta change for any of the 3 files indicated that those files did not need an update, then no update would be attempted for that file (but the remaining two files could still be updated). However, for the discussion below assume that each of the 3 files had a delta change indicating that an update was needed.
The patch integrator 22 attempts to merge the changes of new application source code into these 3 source code files in the v3.4 branch (block 406). If the merge of any of the source code files was unsuccessful for some reason (a “No” to block 408), then a notification could be transmitted that manual conflict resolution is needed (block 410). Thus, in one or more embodiments the method 100 includes, if an attempted automatic check in of the new source code into a particular one of the plurality of code branches fails, transmitting a notification that manual (i.e., non-automatic) resolution is required to incorporate the new source code into the particular code branch. If that occurred, the automatic check in for the branch may be aborted (even if updates had not yet been attempted on all files identified in block 402).
However, if all attempted merges for the v3.4 branch succeed (a “Yes” to block 408), then the new test case source code of the v3.3 branch (if any exists) would also be incorporated into the v3.4 branch (block 412), and the patch integrator 20 would instructs a test environment (e.g., build system 16) to execute test cases for the updated v3.4 code branch (block 414), and the code coverage data for the updated v3.4 code branch is stored in the code coverage repository 22.
The processing circuit 502 is configured to detecting that new source code linked to a defect identifier has been checked in to source code repository 12 for a given one of a plurality of code branches of a software application, wherein the new source code comprises new application source code and new test case source code, and wherein the defect identifier identifies a defect in the software application that the new application source code resolves and the new test case source code tests.
The processing circuit 502 is further configured to automatically check in the new source code to additional ones of the plurality of code branches based on the defect identifier. For each additional code branch into which the new source code is automatically checked in, the processing circuit 502 is configured to:
The computing device 500 also includes a memory circuit 506 which is a computer readable storage medium that stores instructions for operation of the patch integrator 20. The memory circuit 506 may include the code coverage repository 22 and the exclusion list discussed above. Alternatively, the code coverage repository 22 could be stored in an external memory circuit (not shown) that is accessible by the computing device 500. In one or more embodiments, the memory circuit 506 includes a computer program product 508. The computer program product 508 comprises a computer readable storage medium (e.g., memory circuit 506, or some other storage device) having computer readable program code embodied therewith. The computer readable program code comprises computer readable program code configured to detect that new source code linked to a defect identifier has been checked in to a source code repository for a given one of a plurality of code branches of a software application, wherein the new source code comprises new application source code and new test case source code, and wherein the defect identifier identifies a defect in the software application that the new application source code resolves and the new test case source code tests.
The computer readable program code further comprises computer readable program code configured to automatically check in the new source code to additional ones of the plurality of code branches based on the defect identifier. The computer readable program code further comprises computer readable program code configured to, for each additional code branch into which the new source code is automatically checked in:
In some embodiments, the computer program product also includes additional computer readable program code for configuring the computing device 500 to implement any combination of the techniques discussed above.
The methods and apparatus described herein may provide a number of benefits over prior art systems by helping effectively track, identify, merge and manage bug fixes across releases. The automatic identification and promotion of change sets among source code branches, along with the optional exclusion of certain branches from receiving such automatic updates, can alleviate some of the problems of maintaining multiple code branches and tracking bugs across those branches. Thus, these techniques may be particularly useful for agile software development where releases occur frequently. Of course, it is understood that agile software development is a non-limiting example, and that these techniques could even be used in conjunction with the previous “waterfall” software development methodology, for example.
Instead of requiring code to be manually merged into many code branches, which is tedious and error prone, the patch integrator 20 can perform this automatically, freeing up software development resources. The patch integrator 20 can effectively manage resolution of software defects that may span multiple builds and can enable automated delivery of fixes across build images with minimal overhead.
The unified tagging methodology described above, whereby software defects and the source code that resolve those defects can be tagged with a shared defect ID, can be used to provide integration among disparate software development tools from different vendors. Also, the techniques described above for identifying missing test cases and/or low code coverage can provide convenient notifications of occasions when more development effort needs to be focused on preparing test case source code.
The present embodiments may, of course, be carried out in other ways than those specifically set forth herein without departing from essential characteristics of the disclosure. For example, it should be noted that the flowchart and block diagrams in the Figures illustrate the architecture, functionality, and operation of possible implementations of systems, methods and computer program products according to various aspects of the present disclosure. In this regard, each block in the flowchart or block diagrams may represent a module, segment, or portion of code, which comprises one or more executable instructions for implementing the specified logical function(s). It should also be noted that, in some alternative implementations, the functions noted in the block may occur out of the order noted in the figures. For example, two blocks shown in succession may, in fact, be executed substantially concurrently, or the blocks may sometimes be executed in the reverse order, depending upon the functionality involved. It will also be noted that each block of the block diagrams and/or flowchart illustration, and combinations of blocks in the block diagrams and/or flowchart illustration, can be implemented by special purpose hardware-based systems that perform the specified functions or acts, or combinations of special purpose hardware and computer instructions.
The terminology used herein is for the purpose of describing particular aspects only and is not intended to be limiting of the disclosure. As used herein, the singular forms “a”, “an” and “the” are intended to include the plural forms as well, unless the context clearly indicates otherwise. It will be further understood that the terms “comprises” and/or “comprising,” when used in this specification, specify the presence of stated features, integers, steps, operations, elements, and/or components, but do not preclude the presence or addition of one or more other features, integers, steps, operations, elements, components, and/or groups thereof.
The corresponding structures, materials, acts, and equivalents of any means or step plus function elements in the claims below are intended to include any disclosed structure, material, or act for performing the function in combination with other claimed elements as specifically claimed. The description of the present disclosure has been presented for purposes of illustration and description, but is not intended to be exhaustive or limited to the disclosure in the form disclosed. Many modifications and variations will be apparent to those of ordinary skill in the art without departing from the scope and spirit of the disclosure. The aspects of the disclosure herein were chosen and described in order to best explain the principles of the disclosure and the practical application, and to enable others of ordinary skill in the art to understand the disclosure with various modifications as are suited to the particular use contemplated.
Thus, the foregoing description and the accompanying drawings represent non-limiting examples of the methods and apparatus taught herein. As such, the present disclosure is not limited by the foregoing description and accompanying drawings. Instead, the present disclosure is limited only by the following claims and their legal equivalents.
This application is a continuation application of U.S. patent application Ser. No. 14/673,967, which was filed on Mar. 31, 2015 and is incorporated by reference herein in its entirety.
Number | Date | Country | |
---|---|---|---|
Parent | 14673967 | Mar 2015 | US |
Child | 15621019 | US |