Difference between revisions of "LII:Medical Device Software Development with Continuous Integration/Continuous Integration: Part 2"

From LIMSWiki
Jump to navigationJump to search
(Created page with "==Theory== When embarking on a change, a developer takes a copy of the current code base on which to work. As other developers submit changed code to the code repository, this...")
 
m (Top spacing)
 
(2 intermediate revisions by the same user not shown)
Line 1: Line 1:
<div align="center">-----Return to [[LII:Medical Device Software Development with Continuous Integration|the beginning]] of this guide-----</div>
<br />
'''Note''': The following content originates from [[wikibooks::Introduction to Software Engineering/Tools/Continuous Integration|a separate source]]. However, it elaborates further on the original author's ideas in the [[LII:Medical Device Software Development with Continuous Integration/Continuous Integration|previous section]] and adds additional information regarding benefits, drawbacks, and tools other than Jenkins, Trac, and Redmine. It has been added and slightly modified under the [https://creativecommons.org/licenses/by-sa/3.0/ same license] as the rest of the content.
==Theory==
==Theory==
When embarking on a change, a developer takes a copy of the current code base on which to work. As other developers submit changed code to the code repository, this copy gradually ceases to reflect the repository code. When developers submit code to the repository, they must first update their code to reflect the changes in the repository since they took their copy. The more changes the repository contains, the more work developers must do before submitting their own changes.
When embarking on a change, a developer takes a copy of the current code base on which to work. As other developers submit changed code to the code repository, this copy gradually ceases to reflect the repository code. When developers submit code to the repository, they must first update their code to reflect the changes in the repository since they took their copy. The more changes the repository contains, the more work developers must do before submitting their own changes.
Line 7: Line 11:


==Recommended practices==
==Recommended practices==
Continuous integration - as the practice of frequently integrating one's new or changed code with the existing code repository - should occur frequently enough that no intervening window remains between commit and build, and such that no errors can arise without developers noticing them and correcting them immediately.<ref name="Fowler, Continuous Integration practices" >{{Cite web
Continuous integration should occur frequently enough that no intervening window remains between commit and build, and such that no errors can arise without developers noticing them and correcting them immediately.<ref name="FowlerCI00">{{cite web |url=http://www.martinfowler.com/articles/continuousIntegration.html |title=Continuous Integration |author=Fowler, M. |work=MartinFowler.com |date=01 May 2006 |accessdate=27 April 2016}}</ref> Normal practice is to trigger these builds by every commit to a repository, rather than a periodically scheduled build. The practicalities of doing this in a multi-developer environment of rapid commits are such that it's usual to trigger a short timer after each commit, then to start a build when either this timer expires, or after a rather longer interval since the last build. Automated tools such as CruiseControl or Jenkins offer this scheduling automatically.
  |url=http://martinfowler.com/articles/continuousIntegration.html#PracticesOfContinuousIntegration
  |title=Continuous Integration
  |accessdate=2009-11-11
  |last=Fowler |first=Martin  |authorlink=Martin Fowler
}}</ref> Normal practice is to trigger these builds by every commit to a repository, rather than a periodically scheduled build. The practicalities of doing this in a multi-developer environment of rapid commits are such that it's usual to trigger a short timer after each commit, then to start a build when either this timer expires, or after a rather longer interval since the last build. Automated tools such as CruiseControl or Hudson offer this scheduling automatically.


Another factor is the need for a version control system that supports atomic commits, i.e. all of a developer's changes may be seen as a single commit operation. There is no point in trying to build from only half of the changed files.
Another factor is the need for a version control system that supports atomic commits, i.e. all of a developer's changes may be seen as a single commit operation. There is no point in trying to build from only half of the changed files.


===Maintain a code repository===
===Maintain a code repository===
This practice advocates the use of a revision control system for the project's source code. All artifacts required to build the project should be placed in the repository. In this practice and in the revision control community, the convention is that the system should be buildable from a fresh checkout and not require additional dependencies. Extreme Programming advocate Martin Fowler also mentions that where branching is supported by tools, its use should be minimised {{Citation needed|date=March 2010}}. Instead, it is preferred that changes are integrated rather than creating multiple versions of the software that are maintained simultaneously. The mainline (or trunk) should be the place for the working version of the software.
This practice advocates the use of a revision control system for the project's source code. All artifacts required to build the project should be placed in the repository. In this practice and in the revision control community, the convention is that the system should be buildable from a fresh checkout and not require additional dependencies. Extreme Programming advocate Martin Fowler also mentions that where branching is supported by tools, its use should be minimized.<ref name="FowlerCI00" /> Instead, integrating changes is preferred rather than creating multiple versions of the software that are maintained simultaneously. The mainline (or trunk) should be the place for the working version of the software.


===Automate the build===
===Automate the build===
A single command should have the capability of building the system.  Many build-tools, such as make, have existed for many years. Other more recent tools like Ant, Maven, MSBuild or IBM Rational Build Forge are frequently used in continuous integration environments.  Automation of the build should include automating the integration, which often includes deployment into a production-like environment. In many cases, the build script not only compiles binaries, but also generates documentation, website pages, statistics and distribution media (such as Windows MSI files, RPM or DEB files).
A single command should have the capability of building the system.  Many build-tools, such as make, have existed for years. Other more recent tools like Ant, Maven, MSBuild or IBM Rational Build Forge are frequently used in continuous integration environments.  Automation of the build should include automating the integration, which often includes deployment into a production-like environment. In many cases, the build script not only compiles binaries, but also generates documentation, website pages, statistics, and distribution media (such as Windows MSI files, RPM or DEB files).


===Make the build self-testing===
===Make the build self-testing===
Line 26: Line 25:


===Everyone commits to the baseline every day===
===Everyone commits to the baseline every day===
By committing regularly, every committer can reduce the number of conflicting changes. Checking in a week's worth of work runs the risk of conflicting with other features and can be very difficult to resolve.  Early, small conflicts in an area of the system cause team members to communicate about the change they are making.
By committing regularly, every committer can reduce the number of conflicting changes. Checking in a week's worth of work runs the risk of conflicting with other features and can be very difficult to resolve.  Early, small conflicts in an area of the system cause team members to communicate about the change they are making.


Many programmers recommend committing all changes at least once a day (once per feature built), and in addition performing a nightly build.
Many programmers recommend committing all changes at least once a day (once per feature built), and in addition performing a nightly build.


===Every commit (to baseline) should be built===
===Every commit (to baseline) should be built===
The system should build commits to the current working version in order to verify that they integrate correctly. A common practice is to use Automated Continuous Integration, although this may be done manually. For many, continuous integration is synonymous with using Automated Continuous Integration where a continuous integration server or daemon monitors the version control system for changes, then automatically runs the build process.
The system should build commits to the current working version in order to verify that they integrate correctly. A common practice is to use automated continuous integration, although this may be done manually. For many, continuous integration is synonymous with using automated continuous integration where a continuous integration server or daemon monitors the version control system for changes, then automatically runs the build process.


===Keep the build fast===
===Keep the build fast===
Line 37: Line 36:


===Test in a clone of the production environment===
===Test in a clone of the production environment===
Having a test environment can lead to failures in tested systems when they deploy in the production environment, because the production environment may differ from the test environment in a significant way.  However, building a replica of a production environment is cost prohibitive. Instead, the pre-production environment should be built to be a scalable version of the actual production environment to both alleviate costs while maintaining technology stack composition and nuances.
Having a test environment can lead to failures in tested systems when they deploy in the production environment, because the production environment may differ from the test environment in a significant way.  However, building a replica of a production environment is cost prohibitive. Instead, the pre-production environment should be built to be a scalable version of the actual production environment to both alleviate costs while maintaining technology stack composition and nuances.


===Make it easy to get the latest deliverables===
===Make it easy to get the latest deliverables===
Line 46: Line 45:


===Automate deployment===
===Automate deployment===
Most CI systems allow the running of scripts after a build finishes. In most situations, it is possible to write a script to deploy the application to a live test server that everyone can look at. A further advance in this way of thinking is Continuous Deployment, which calls for the software to be deployed directly into production, often with additional automation to prevent defects or regressions<ref>See http://radar.oreilly.com/2009/03/continuous-deployment-5-eas.html and http://timothyfitz.wordpress.com/2009/02/10/continuous-deployment-at-imvu-doing-the-impossible-fifty-times-a-day/</ref>.
Most CI systems allow the running of scripts after a build finishes. In most situations, it is possible to write a script to deploy the application to a live test server that everyone can look at. A further advance in this way of thinking is the concept of "continuous deployment," which calls for the software to be deployed directly into production, often with additional automation to prevent defects or regressions.<ref name="RiesCont09">{{cite web |url=http://radar.oreilly.com/2009/03/continuous-deployment-5-eas.html |title=Continuous deployment in 5 easy steps |author=Ries, E. |work=O'Reilly Radar |publisher= O'Reilly Media, Inc |date=30 March 2009 |accessdate=27 April 2016}}</ref><ref name="FitzCont09">{{cite web |url=http://timothyfitz.com/2009/02/10/continuous-deployment-at-imvu-doing-the-impossible-fifty-times-a-day/ |title=Continuous Deployment at IMVU: Doing the impossible fifty times a day |author=Fitz, T. |work=timothyfitz.com |date=10 February 2009 |accessdate=27 April 2016}}</ref>
 
==History==
Continuous Integration emerged in the Extreme Programming (XP) community, and XP advocates Martin Fowler and Kent Beck first wrote about continuous integration circa 1999. Fowler's paper<ref name="Fowler, Continuous Integration" >{{Cite web
  |last=Fowler |first=Martin  |authorlink=Martin Fowler
  |url=http://www.martinfowler.com/articles/continuousIntegration.html
  |title=Continuous Integration
  |ref=Fowler, Continuous Integration
}}</ref> is a popular source of information on the subject. Beck's book ''Extreme Programming Explained''<ref name="Beck, Extreme Programming Explained" >{{Cite book
  |title=Extreme Programming Explained
  |last=Beck  |first=Kent  |authorlink=Kent Beck
  |year=1999
  |isbn=0-201-61641-6
  |ref=Beck, Extreme Programming Explained
}}</ref>, the original reference for Extreme Programming, also describes the term.


==Advantages and disadvantages==
==Advantages and disadvantages==
=== Advantages ===
===Advantages===
Continuous integration has many advantages:
Continuous integration has many advantages<ref name="FowlerCI00" />:


* when unit tests fail or a bug emerges, developers might revert the codebase back to a bug-free state, without wasting time debugging
* ability to revert the codebase back to a bug-free state, without wasting time debugging, when unit tests fail or a bug emerges;
* developers detect and fix integration problems continuously - avoiding last-minute chaos at release dates, (when everyone tries to check in their slightly incompatible versions).
* ability to detect and fix integration problems continuously, avoiding last-minute chaos at release dates (when everyone tries to check in their slightly incompatible versions);
* early warning of broken/incompatible code
* early warning of broken/incompatible code;
* early warning of conflicting changes
* early warning of conflicting changes;
* immediate unit testing of all changes
* immediate unit testing of all changes;
* constant availability of a "current" build for testing, demo, or release purposes
* constant availability of a "current" build for testing, demo, or release purposes;
* immediate feedback to developers on the quality, functionality, or system-wide impact of code they are writing
* immediate feedback to developers on the quality, functionality, or system-wide impact of code they are writing;
* frequent code check-in pushes developers to create modular, less complex code{{Citation needed|date=November 2009}}
* modular, less complex code often a result of frequent code check-in by developers; and
* metrics generated from automated testing and CI (such as metrics for code coverage, code complexity, and features complete) focus developers on developing functional, quality code, and help develop momentum in a team{{Citation needed|date=November 2009}}
* metrics generated from automated testing and CI (such as metrics for code coverage, code complexity, and features complete) focus developers on developing functional, quality code, and help develop momentum in a team.


===Disadvantages===
===Disadvantages===
* initial setup time required
* initial setup time required;
* well-developed test-suite required to achieve automated testing advantages
* well-developed test-suite required to achieve automated testing advantages;
* large-scale refactoring can be troublesome due to continuously changing code base
* large-scale refactoring can be troublesome due to continuously changing code base; and
* hardware costs for build machines can be significant
* hardware costs for build machines can be significant.


Many teams using CI report that the advantages of CI well outweigh the disadvantages.<ref>{{Cite conference
Many teams using CI report that the advantages of CI well outweigh the disadvantages.<ref name="RichardsonAgile08">{{cite conference |url=https://nofluffjuststuff.com/conference/boston/2008/09/session?id=11645 |title=''Agile Software Testing Strategies'' at No Fluff Just Stuff Conference |author=Richardson, J. |place=Boston, Massachusetts |date=14 September 2008 |accessdate=27 April 2016}}</ref> The effect of finding and fixing integration bugs early in the development process saves both time and money over the lifespan of a project.
  | first = Jared
  | last = Richardson
  | authorlink =
  | title = Agile Testing Strategies at No Fluff Just Stuff Conference
  | url = http://www.nofluffjuststuff.com
  | place = Boston, Massachusetts
  | year = September 2008
}}
</ref>
The effect of finding and fixing integration bugs early in the development process saves both time and money over the lifespan of a project.


==Software==
==Software==
Line 123: Line 98:
* Tinderbox — Mozilla-based product written in Perl
* Tinderbox — Mozilla-based product written in Perl
* Rational Team Concert — proprietary software development collaboration platform with built-in build engine by IBM including Rational Build Forge
* Rational Team Concert — proprietary software development collaboration platform with built-in build engine by IBM including Rational Build Forge
See comparison of continuous integration software for a more in depth feature matrix.
 
See the links to in-depth feature matrix in the external links for deeper comparisons.


==Further reading==
==Further reading==
* {{Cite book
* {{cite book |title=Continuous Integration. Improving Software Quality and Reducing Risk |author=Duvall, P.M. |publisher=Addison-Wesley |year=2007 |isbn=0321336380}}</ref>
| first = Paul M.
| last = Duvall
| authorlink = Paul Duvall
| year = 2007
| title = Continuous Integration. Improving Software Quality and Reducing Risk
| publisher = Addison-Wesley
| isbn = 0-321-33638-0
}}


==References==
==References==
Line 144: Line 112:
* [http://www.c2.com/cgi/wiki?ContinuousIntegration Continuous Integration at the Portland Pattern Repository] (a collegial discussion)
* [http://www.c2.com/cgi/wiki?ContinuousIntegration Continuous Integration at the Portland Pattern Repository] (a collegial discussion)
* [http://c2.com/cgi/wiki?CrossPlatformTesting Cross platform testing at the Portland Pattern Repository]
* [http://c2.com/cgi/wiki?CrossPlatformTesting Cross platform testing at the Portland Pattern Repository]
* [http://confluence.public.thoughtworks.org/display/CC/CI+Feature+Matrix Continuous Integration Server Feature Matrix] (a guide to tools)
* [https://web.archive.org/web/20150504203619/http://confluence.public.thoughtworks.org/display/CC/CI+Feature+Matrix Continuous Integration Server Feature Matrix] (archived version of guide to tools)
* [http://www.methodsandtools.com/archive/archive.php?id=42 Continuous Integration: The Cornerstone of a Great Shop] by Jared Richardson (an introduction)
* [http://www.methodsandtools.com/archive/archive.php?id=42 Continuous Integration: The Cornerstone of a Great Shop] by Jared Richardson (an introduction)
* [http://jayflowers.com/joomla/index.php?option=com_content&task=view&id=26 A Recipe for Build Maintainability and Reusability] by Jay Flowers
* [http://jayflowers.com/joomla/index.php?option=com_content&task=view&id=26 A Recipe for Build Maintainability and Reusability] by Jay Flowers
* [http://www.ibm.com/developerworks/java/library/j-ap11297/ Continuous Integration anti-patterns] by Paul Duvall
* [http://www.ibm.com/developerworks/java/library/j-ap11297/ Continuous Integration anti-patterns] by Paul Duvall
* [http://www.extremeprogramming.org/rules/integrateoften.html Extreme programming ]
* [http://www.extremeprogramming.org/rules/integrateoften.html Extreme programming ]

Latest revision as of 21:05, 27 April 2016

-----Return to the beginning of this guide-----


Note: The following content originates from a separate source. However, it elaborates further on the original author's ideas in the previous section and adds additional information regarding benefits, drawbacks, and tools other than Jenkins, Trac, and Redmine. It has been added and slightly modified under the same license as the rest of the content.

Theory

When embarking on a change, a developer takes a copy of the current code base on which to work. As other developers submit changed code to the code repository, this copy gradually ceases to reflect the repository code. When developers submit code to the repository, they must first update their code to reflect the changes in the repository since they took their copy. The more changes the repository contains, the more work developers must do before submitting their own changes.

Eventually, the repository may become so different from the developers' baselines that they enter what is sometimes called "integration hell",[1] where the time it takes to integrate exceeds the time it took to make their original changes. In a worst-case scenario, developers may have to discard their changes and completely redo the work.

Continuous integration involves integrating early and often, so as to avoid the pitfalls of "integration hell." The practice aims to reduce rework and thus reduce cost and time, particularly when automated as a best practice.[2][3]

Recommended practices

Continuous integration should occur frequently enough that no intervening window remains between commit and build, and such that no errors can arise without developers noticing them and correcting them immediately.[4] Normal practice is to trigger these builds by every commit to a repository, rather than a periodically scheduled build. The practicalities of doing this in a multi-developer environment of rapid commits are such that it's usual to trigger a short timer after each commit, then to start a build when either this timer expires, or after a rather longer interval since the last build. Automated tools such as CruiseControl or Jenkins offer this scheduling automatically.

Another factor is the need for a version control system that supports atomic commits, i.e. all of a developer's changes may be seen as a single commit operation. There is no point in trying to build from only half of the changed files.

Maintain a code repository

This practice advocates the use of a revision control system for the project's source code. All artifacts required to build the project should be placed in the repository. In this practice and in the revision control community, the convention is that the system should be buildable from a fresh checkout and not require additional dependencies. Extreme Programming advocate Martin Fowler also mentions that where branching is supported by tools, its use should be minimized.[4] Instead, integrating changes is preferred rather than creating multiple versions of the software that are maintained simultaneously. The mainline (or trunk) should be the place for the working version of the software.

Automate the build

A single command should have the capability of building the system. Many build-tools, such as make, have existed for years. Other more recent tools like Ant, Maven, MSBuild or IBM Rational Build Forge are frequently used in continuous integration environments. Automation of the build should include automating the integration, which often includes deployment into a production-like environment. In many cases, the build script not only compiles binaries, but also generates documentation, website pages, statistics, and distribution media (such as Windows MSI files, RPM or DEB files).

Make the build self-testing

Once the code is built, all tests should run to confirm that it behaves as the developers expect it to behave.

Everyone commits to the baseline every day

By committing regularly, every committer can reduce the number of conflicting changes. Checking in a week's worth of work runs the risk of conflicting with other features and can be very difficult to resolve. Early, small conflicts in an area of the system cause team members to communicate about the change they are making.

Many programmers recommend committing all changes at least once a day (once per feature built), and in addition performing a nightly build.

Every commit (to baseline) should be built

The system should build commits to the current working version in order to verify that they integrate correctly. A common practice is to use automated continuous integration, although this may be done manually. For many, continuous integration is synonymous with using automated continuous integration where a continuous integration server or daemon monitors the version control system for changes, then automatically runs the build process.

Keep the build fast

The build needs to complete rapidly, so that if there is a problem with integration, it is quickly identified.

Test in a clone of the production environment

Having a test environment can lead to failures in tested systems when they deploy in the production environment, because the production environment may differ from the test environment in a significant way. However, building a replica of a production environment is cost prohibitive. Instead, the pre-production environment should be built to be a scalable version of the actual production environment to both alleviate costs while maintaining technology stack composition and nuances.

Make it easy to get the latest deliverables

Making builds readily available to stakeholders and testers can reduce the amount of rework necessary when rebuilding a feature that doesn't meet requirements. Additionally, early testing reduces the chances that defects survive until deployment. Finding errors earlier also, in some cases, reduces the amount of work necessary to resolve them.

Everyone can see the results of the latest build

It should be easy to find out where/whether the build breaks and who made the relevant change.

Automate deployment

Most CI systems allow the running of scripts after a build finishes. In most situations, it is possible to write a script to deploy the application to a live test server that everyone can look at. A further advance in this way of thinking is the concept of "continuous deployment," which calls for the software to be deployed directly into production, often with additional automation to prevent defects or regressions.[5][6]

Advantages and disadvantages

Advantages

Continuous integration has many advantages[4]:

  • ability to revert the codebase back to a bug-free state, without wasting time debugging, when unit tests fail or a bug emerges;
  • ability to detect and fix integration problems continuously, avoiding last-minute chaos at release dates (when everyone tries to check in their slightly incompatible versions);
  • early warning of broken/incompatible code;
  • early warning of conflicting changes;
  • immediate unit testing of all changes;
  • constant availability of a "current" build for testing, demo, or release purposes;
  • immediate feedback to developers on the quality, functionality, or system-wide impact of code they are writing;
  • modular, less complex code often a result of frequent code check-in by developers; and
  • metrics generated from automated testing and CI (such as metrics for code coverage, code complexity, and features complete) focus developers on developing functional, quality code, and help develop momentum in a team.

Disadvantages

  • initial setup time required;
  • well-developed test-suite required to achieve automated testing advantages;
  • large-scale refactoring can be troublesome due to continuously changing code base; and
  • hardware costs for build machines can be significant.

Many teams using CI report that the advantages of CI well outweigh the disadvantages.[7] The effect of finding and fixing integration bugs early in the development process saves both time and money over the lifespan of a project.

Software

To support continuous integration, software tools such as automated build software can be employed.

Software tools for continuous integration include:

  • AnthillPro — continuous integration server by Urbancode
  • Apache Continuum — continuous integration server supporting Apache Maven and Apache Ant. Supports CVS, Subversion, Ant, Maven, and shell scripts
  • Apache Gump — continuous integration tool by Apache
  • Automated Build Studio — proprietary automated build, continuous integration and release management system by AutomatedQA
  • Bamboo — proprietary continuous integration server by Atlassian Software Systems
  • BuildBot — Python/Twisted-based continuous build system
  • BuildForge - proprietary automated build engine by IBM / Rational
  • BuildMaster — proprietary application lifecycle management and continuous integration tool by Inedo
  • CABIE - Continuous Automated Build and Integration Environment — open source, written in Perl; works with CVS, Subversion, AccuRev, Bazaar and Perforce
  • Cascade — proprietary continuous integration tool; provides a checkpointing facility to build and test changes before they are committed
  • codeBeamer — proprietary collaboration software with built-in continuous integration features
  • CruiseControl — Java-based framework for a continuous build process
  • CruiseControl.NET — .NET-based automated continuous integration server
  • CruiseControl.rb - Lightweight, Ruby-based continuous integration server that can build any codebase, not only Ruby, released under Apache Licence 2.0
  • ElectricCommander — proprietary continuous integration and release management solution from Electric Cloud
  • FinalBuilder Server — proprietary automated build and continuous integration server by VSoft Technologies
  • Go — proprietary agile build and release management software by Thoughtworks
  • Jenkins (formerly known as Hudson) — MIT-licensed, written in Java, runs in servlet container, supports CVS, Subversion, Mercurial, Git, StarTeam, Clearcase, Ant, NAnt, Maven, and shell scripts
  • Software Configuration and Library Manager — software configuration management system for z/OS by IBM Rational Software
  • QuickBuild - proprietary continuous integration server with free community edition featuring build life cycle management and pre-commit verification.
  • TeamCity — proprietary continuous-integration server by JetBrains with free professional edition
  • Team Foundation Server — proprietary continuous integration server and source code repository by Microsoft
  • Tinderbox — Mozilla-based product written in Perl
  • Rational Team Concert — proprietary software development collaboration platform with built-in build engine by IBM including Rational Build Forge

See the links to in-depth feature matrix in the external links for deeper comparisons.

Further reading

  • Duvall, P.M. (2007). Continuous Integration. Improving Software Quality and Reducing Risk. Addison-Wesley. ISBN 0321336380. 

</ref>

References

  1. Cunningham, W. (20 December 2012). "Integration Hell". WikiWikiWeb. http://c2.com/cgi/wiki?IntegrationHell. Retrieved 27 April 2016. 
  2. Brauneis, D.; Hüttermann, M. (16 January 2010). "[OSLC Possible new Working Group - Automation"]. open-services.net. http://open-services.net/pipermail/community_open-services.net/2010-January/000214.html. Retrieved 27 April 2016. 
  3. Taylor, B. (10 February 2009). "Rails Deployment and Automation with ShadowPuppet and Capistrano". Rails Machine. Archived from the original on 03 March 2011. https://web.archive.org/web/20110303225845/http://blog.railsmachine.com/articles/2009/02/10/rails-deployment-and-automation-with-shadowpuppet-and-capistrano. Retrieved 27 April 2016. 
  4. 4.0 4.1 4.2 Fowler, M. (1 May 2006). "Continuous Integration". MartinFowler.com. http://www.martinfowler.com/articles/continuousIntegration.html. Retrieved 27 April 2016. 
  5. Ries, E. (30 March 2009). "Continuous deployment in 5 easy steps". O'Reilly Radar. O'Reilly Media, Inc. http://radar.oreilly.com/2009/03/continuous-deployment-5-eas.html. Retrieved 27 April 2016. 
  6. Fitz, T. (10 February 2009). "Continuous Deployment at IMVU: Doing the impossible fifty times a day". timothyfitz.com. http://timothyfitz.com/2009/02/10/continuous-deployment-at-imvu-doing-the-impossible-fifty-times-a-day/. Retrieved 27 April 2016. 
  7. Richardson, J. (14 September 2008). "Agile Software Testing Strategies at No Fluff Just Stuff Conference". Boston, Massachusetts. https://nofluffjuststuff.com/conference/boston/2008/09/session?id=11645. Retrieved 27 April 2016. 

External links