ant在持续集成的应用
一、Ant概述
Ant是一个Apache基金会下的跨平台的构建工具,它是一个将软件编译、测试、部署等步骤联系在一起加以自动化的一个工具。
在本文中,我主要让介绍Ant的命令、构建文件、最后部分以一个实例概述在持续集成工具jenkins中的应用。
二、为什么使用ant
开发过程中,我们一般是在windows下,利用eclipse可以很轻松地实现一键编译、测试、打包、部署,但是我们的应用一般是部署在linux下,我们不可能安装一个eclipse在linux上。那linux上有没有类似eclipse这样的工具呢?答案是肯定的,这个工具就是Ant,使用它可以实现自动化部署。除此之外,它还有很多优点。
1、跨平台性:Ant是纯Java语言编写的,所以具有很好的跨平台性。
2、操作简单:Ant运行时使用一个XML文件(构建文件)。 Ant通过调用target树,就可以执行各种task。每个task实现了特定接口对象。
3、集成方便:使用Ant就是搭建了一个一致的配置环境,每次集成都通过Ant自动化的构建(包括编译,发布,自动化测试)来验证,从而尽早地发现集成错误。
三、安装与配置
比较简单,这里请参见“使用jenkins实现持续集成V2.pdf” 2.2说明,注意安装好之后要配置下环境变量。
参见我的这篇文章:使用jenkins实现持续集成(http://www.cnblogs.com/zishengY/p/7170656.html)
四、Ant命令
Ant的构件文件是基于XML编写的,默认build.xml就是主要的ant脚本。
ant
在当前目录下的build.xml运行Ant,执行缺省的target。
ant -buildfile build-test.xml
在当前目录下的build-test.xml运行Ant,执行缺省的target。
ant -buildfile build-test.xml clean
在当前目录下的build-test.xml运行Ant,执行一个叫做clean的target。
ant -buildfile build-test.xml -Dbuild=build/classes clean
在当前目录下的build-test.xml运行Ant,执行一个叫做clean的target,并设定build属性的值为build/classes。
五、主要元素介绍。
参见“http://blog.csdn.net/D3Direct_1/article/details/53055236”
本部分主要介绍Ant构建文件(默认是是build.xml)的元素组成。
1、project 节点元素
project 元素是 Ant 构件文件的根元素, Ant 构件文件至少应该包含一个 project 元素,否则会发生错误。在每个 project 元素下,可包含多个 target 元素。接下来向读者展示一下 project 元素的各属性。
● name 属性:用于指定 project 元素的名称。
● default 属性:用于指定 project 默认执行的target 的名称。
● basedir 属性:用于指定基路径的位置。该属性没有指定时,使用 Ant 的构件文件的附目录作为基准目录。<?xml version="1.0" ?><project name="ant-project" default="print-dir" basedir="."><target name="print-dir"><echo message="The base dir is: ${basedir}" /></target></project>从上例可以看出,在这里定义了default 属性的值为print-dir,即当运行ant 命令时,如果没有指明执行的target,则将执行默认的target(print-dir)。此外,还定义了basedir 属性的值为 “.” ,.表示当前目录,进入当前目录后运行ant 命令,得一下结果:
2、target节点元素
target为ant的基本执行单元或是任务,它可以包含一个或多个具体的单元/任务。多个target 可以存在相互依赖关系。它有如下属性:
● name 属性:指定 target 元素的名称,这个属性在一个 project 元素中是唯一的。我们可以通过指定 target 元素的名称来指定某个 target 。
● depends 属性:用于描述 target 之间的依赖关系,若与多个 target 存在依赖关系时,需要以“,”间隔。 Ant 会依照 depends 属性中 target 出现的顺序依次执行每个 target ,被依赖的target 会先执行。
● if 属性:用于验证指定的属性是存在,若不存在,所在 target 将不会被执行。
● unless 属性:该属性的功能与 if 属性的功能正好相反,它也用于验证指定的属性是否存在,若不存在,所在 target 将会被执行。
● description 属性:该属性是关于 target 功能的简短描述和说明。
示例:<?xml version="1.0" ?><project name="ant-target" default="print"><target name="version" if="ant.java.version"><echo message="Java Version: ${ant.java.version}" /></target><target name="print" depends="version" unless="docs"><description>a depend example!</description><echo message="The base dir is: ${basedir}" /></target></project>从以下结果后可以看到,我们运行的是名为 print的target ,由于它依赖于version这个target任务,所以 version将首先被执行,同时因为系统配置了JDK,所以 ant.java.version 属性存在,执行了version,输出信息:"[echo] Java Version: 1.6 ",version执行完毕后,接着执行 print,因为docs不存在,而unless属性是在不存在时进入所在target 的,由此可知 print得以执行,输出信息:"[echo] The base dir is:D:\Workspace\AntExample\build"。
3、property属性节点元素
property元素可看作参量或者参数的定义,project 的属性可以通过 property 元素来设定,也可在 Ant 之外设定。若要在外部引入某文件,例如 build.properties 文件,可以通过如下内容将其引:
<property file="build.properties"/>
property 元素可用作 task 的属性值。在 task 中是通过将属性名放在${属性名}之间,并放在 task 属性值的位置来实现的。
Ant 提供了一些内置的属性,它能得到的系统属性的列表与 Java 文档中 System.getProperties() 方法得到的属性一致,这些系统属性可参考 sun 网站的说明。同时, Ant 还提供了一些它自己的内置属性,如下:
basedir: project 基目录的绝对路径;
ant.file: buildfile的绝对路径,上例中ant.file值为D:\Workspace\AntExample\build;
ant.version: Ant 的版本信息,本文为1.8.1 ;
ant.project.name: 当前指定的project的名字,即前文说到的project的name属性值;
ant.java.version: Ant 检测到的JDK版本,本文为 1.6 。举例说明如下:
<project name="ant-project" default="example"><property name="name" value="jojo" /><property name="age" value="25" /><target name="example"><echo message="name: ${name}, age: ${age}" /></target></project>上例中用户设置了名为name 和age的两个属性,这两个属性设置后,在下文中可以通过 ${name} 和 ${age} 分别取得这两个属性值。
4、copy命令
copy主要用来对文件和目录的复制功能。举例如下:
● 复制单个文件:
<copy file="old.txt" tofile="new.txt"/>● 对文件目录进行复制:
<copy todir="../dest_dir">
<fileset dir="src_dir"/>
</copy>● 将文件复制到另外的目录:
<copy file="src.txt" todir="c:/base"/>
5、delete命令
对文件或目录进行删除,举例如下:
● 删除某个文件:
<delete file="/res/image/cat.jpg"/>● 删除某个目录:
<delete dir="/res/image"/>● 删除所有的jar文件或空目录:
<delete includeEmptyDirs="true">
<fileset dir="." includes="**/*.jar"/>
</delete>
6、 mkdir 命令
创建目录。
<mkdir dir="/home/philander/build/classes"/>
7、 move 命令
移动文件或目录,举例如下:
● 移动单个文件:
<move file="sourcefile" tofile=”destfile”/>● 移动单个文件到另一个目录:
<move file="sourcefile" todir=”movedir”/>● 移动某个目录到另一个目录:
<move todir="newdir">
<fileset dir="olddir"/>
</move>
8、echo 命令
该任务的作用是根据日志或监控器的级别输出信息。它包括 message 、 file 、 append 和 level 四个属性,举例如下
<echo message="ant message" file="/logs/ant.log" append="true">
9、jar 标签节点元素
该标签用来生成一个JAR文件,其属性如下。
● destfile表示JAR文件名。
● basedir表示被归档的文件名。
● includes表示别归档的文件模式。
● exchudes表示被排除的文件模式。● compress表示是否压缩。
示例:
<jar destfile="${webRoot}/${ash_jar}" level="9" compress="true" encoding="utf-8" basedir="${dest}"><manifest><attribute name="Implementation-Version" value="Version: 2.2"/></manifest></jar>上面的mainfest是jar包中的MEAT-INF中的MANIFEST.MF中的文件内容
同样打包操作的的还有war、tgz,已经解压操作uzip
<!-- 创建zip --><zip basedir="${basedir}\classes" zipfile="temp\output.zip"/><!-- 创建tgz --><gzip src="classes\**\*.class" zipfile="output.class.gz"/><!-- 解压zip --><unzip src="output.class.gz" dest="extractDir"/><!-- 建立war包 --><war destfile="${webRoot}/ash.war" basedir="${basedir}/web" webxml="${basedir}/web/WEB-INF/web.xml"><exclude name="WEB-INF/classes/**"/><exclude name="WEB-INF/lib/**"/><exclude name="WEB-INF/work/_jsp/**"/><lib dir="${lib.dir}" includes="**/*.jar, **/*.so, **/*.dll"><exclude name="${webRoot}\${helloworld_jar}"/></lib><lib file="${webRoot}/${helloworld_jar}"/><classes dir="${dest}" includes="**/*.xml, **/*.properites, **/*.xsd"> </classes></war>
10、javac 标签节点元素
该标签用于编译一个或一组java文件,其属性如下。
● srcdir表示源程序的目录。
● destdir表示class文件的输出目录。
● include表示被编译的文件的模式。
● excludes表示被排除的文件的模式。
● classpath表示所使用的类路径。
● debug表示包含的调试信息。
● optimize表示是否使用优化。
● verbose 表示提供详细的输出信息。
● fileonerror表示当碰到错误就自动停止。示例
<javac srcdir="${src}" destdir="${dest}"/><!-- 设置jvm内存<javac srcdir="src" fork="true"/><javac srcdir="src" fork="true" executable="d:\sdk141\bin\javac"memoryMaximumSize="128m"/>-->
11、java 标签节点元素
该标签用来执行编译生成的.class文件,其属性如下。
● classname 表示将执行的类名。
● jar表示包含该类的JAR文件名。
● classpath所表示用到的类路径。
● fork表示在一个新的虚拟机中运行该类。
● failonerror表示当出现错误时自动停止。
● output 表示输出文件。
● append表示追加或者覆盖默认文件。示例
<java classname="com.hoo.test.HelloWorld" classpath="${hello_jar}"/>
12、arg 数据参数元素
由Ant构建文件调用的程序,可以通过<arg>元素向其传递命令行参数,如apply,exec和java任务均可接受嵌套<arg>元素,可以为各自的过程调用指定参数。以下是<arg>的所有属性。
● values 是一个命令参数。如果参数中有空格,但又想将它作为单独一个值,则使用此属性。
● file 表示一个参数的文件名。在构建文件中,此文件名相对于当前的工作目录。
● line 表示用空格分隔的多个参数列表。
● 表示路径,一个作为单个命令行变量的path-like的字符串;或作为分隔符,Ant会将其转变为特定平台的分隔符。
● pathref 引用的path(使用path元素节点定义path)的id
● prefix 前缀
● suffix 后缀例子
<arg value="-l -a"/>
是一个含有空格的单个的命令行变量。
<arg line="-l -a"/>
是两个空格分隔的命令行变量。
<arg path="/dir;/dir2:\dir3"/>
是一个命令行变量,其值在DOS系统上为\dir;\dir2;\dir3;在Unix系统上为/dir:/dir2:/dir3 。
13、ervironment 类型
由Ant构建文件调用的外部命令或程序,<env>元素制定了哪些环境变量要传递给正在执行的系统命令,<env>元素可以接受以下属性。
● file表示环境变量值的文件名。此文件名要被转换位一个绝对路径。
● path表示环境变量的路径。Ant会将它转换为一个本地约定。
● value 表示环境变量的一个直接变量。
● key 表示环境变量名。
注意 file path 或 value只能取一个。
14、filelist 文件集合列表
filelist 是一个支持命名的文件列表的数据类型,包含在一个filelist类型中的文件不一定是存在的文件。以下是其所有的属性。
● dir是用于计算绝对文件名的目录。
● files 是用逗号分隔的文件名列表。
● refid 是对某处定义的一个<filelist>的引用。
注意 dir 和 files 都是必要的,除非指定了refid(这种情况下,dir和files都不允许使用)。示例
<filelist id="docfiles" dir="${doc.src}" files="foo.xml,bar.xml"/>文件集合 ${doc.src}/foo.xml和${doc.src}/bar.xml. 这些文件也许还是不存在的文件.<filelist id="docfiles" dir="${doc.src}" files="foo.xml bar.xml"/><filelist refid="docfiles"/><filelist id="docfiles" dir="${doc.src}"><file name="foo.xml"/><file name="bar.xml"/></filelist>
15、fileset 文件类型
fileset 数据类型定义了一组文件,并通常表示为<fileset>元素。不过,许多ant任务构建成了隐式的fileset,这说明他们支持所有的fileset属性和嵌套元素。以下为fileset 的属性列表。
● dir表示fileset 的基目录。
● casesensitive的值如果为false,那么匹配文件名时,fileset不是区分大小写的,其默认值为true.
● defaultexcludes 用来确定是否使用默认的排除模式,默认为true。
● excludes 是用逗号分隔的需要派出的文件模式列表。
● excludesfile 表示每行包含一个排除模式的文件的文件名。
● includes 是用逗号分隔的,需要包含的文件模式列表。
● includesfile 表示每行包括一个包含模式的文件名。示例
<fileset id="lib.runtime" dir="${lib.path}/runtime"><include name="**/*.jar"/><include name="**/*.so"/><include name="**/*.dll"/></fileset><fileset id="lib.container" dir="${lib.path}/container"><include name="**/*.jar"/></fileset><fileset id="lib.extras" dir="${lib.path}"><include name="test/**/*.jar"/></fileset>
16、patternset 类型
fileset 是对文件的分组,而patternset是对模式的分组,他们是紧密相关的概念。
<patternset>支持4个属性:includes、excludex、includexfile、excludesfile,这些与fileset相同。
patternset 还允许以下嵌套元素:include,exclude,includefile 和 excludesfile.
示例
<!-- 黑白名单 --><patternset id="non.test.sources"><include name="**/*.java"/><!-- 文件名包含Test的排除 --><exclude name="**/*Test*"/></patternset><patternset id="sources"><include name="std/**/*.java"/><!-- 判断条件 存在professional就引入 --><include name="prof/**/*.java" if="professional"/><exclude name="**/*Test*"/></patternset><!-- 一组文件 --><patternset includesfile="some-file"/><patternset><includesfile name="some-file"/><patternset/><patternset><includesfile name="some-file"/><includesfile name="${some-other-file}" if="some-other-file"/><patternset/>
17、filterset 类型
filterset定义了一组过滤器,这些过滤器将在文件移动或复制时完成文件的文本替换。
主要属性如下:
● begintoken 表示嵌套过滤器所搜索的记号,这是标识其开始的字符串。
● endtoken 表示嵌套过滤器所搜索的记号这是标识其结束的字符串。
● id 是过滤器的唯一标志符。
● refid 是对构建文件中某处定义一个过滤器的引用。示例
<!-- 将目标文件build.dir目录中的version.txt文件内容中的@DATE@替换成TODAY当前日期的值,并把替换后的文件存放在dist.dir目录中 --><copy file="${build.dir}/version.txt" toFile="${dist.dir}/version.txt"><filterset><filter token="DATE" value="${TODAY}"/></filterset></copy><!-- 自定义变量的格式 --><copy file="${build.dir}/version.txt" toFile="${dist.dir}/version.txt"><!-- 从version.txt中的%位置开始搜索,到*位置结束,进行替换内容中的@DATE@替换成TODAY当前日期的值--><filterset begintoken="%" endtoken="*"><filter token="DATE" value="${TODAY}"/></filterset></copy><!-- 使用外部的过滤定义文件 --><copy toDir="${dist.dir}/docs"><fileset dir="${build.dir}/docs"><include name="**/*.html"></fileset><filterset begintoken="%" endtoken="*"><!-- 过来文件从外部引入,过来的属性和值配置在dist.properties文件中 --><filtersfile file="${user.dir}/dist.properties"/></filterset></copy><!-- 使用引用方式,重复利用过滤集 --><filterset id="myFilterSet" begintoken="%" endtoken="*"><filter token="DATE" value="${TODAY}"/></filterset><copy file="${build.dir}/version.txt" toFile="${dist.dir}/version.txt"><filterset refid="myFilterSet"/></copy>
18、path类型
path元素用来表示一个类路径,不过它还可以用于表示其他的路径。在用作几个属性时,路经中的各项用分号或冒号隔开。在构建的时候,此分隔符将代替当前平台中所有的路径分隔符,其拥有的属性如下。
● location 表示一个文件或目录。Ant在内部将此扩展为一个绝对路径。
● refid 是对当前构建文件中某处定义的一个path的引用。
● path表示一个文件或路径名列表。示例
<path id="buildpath"><fileset refid="lib.runtime"/><fileset refid="lib.container"/><fileset refid="lib.extras"/></path><path id="src.paths"><fileset id="srcs" dir="."><include name="src/**/*.java"/></fileset></path>
六、ant编译打包、运行工程
<?xml version="1.0" encoding="UTF-8"?><!-- name是当前工程的名称,default是默认执行的任务,basedir是工作目录(.代表当前根目录) --><project name="HelloWorld" default="run" basedir="."><!-- property类似于程序中定义简单的变量 --><property name="src" value="src"/><property name="dest" value="classes"/><property name="hello_jar" value="helloWorld.jar"/><!--target是一个事件、事情、任务, name是当前事情的名称,depends是依赖的上一件或是多件事情如果所依赖的事情没有执行,ant会先运行依赖事情,然后再运行当前事情--><!-- 初始化 --><target name="init"><!-- 建立classes目录 --><mkdir dir="${dest}"/><mkdir dir="temp"/><mkdir dir="temp2"/></target><!-- 编译 --><target name="compile" depends="init"><javac srcdir="${src}" destdir="${dest}"/><!-- 设置jvm内存<javac srcdir="src" fork="true"/><javac srcdir="src" fork="true" executable="d:\sdk141\bin\javac"memoryMaximumSize="128m"/>--></target><!-- 建立jar包 --><target name="build" depends="compile"><!--<jar jarfile="${hello_jar}" basedir="${dest}"/>创建一个名称是package.jar文件<jar destfile="package.jar" basedir="classes"/>--><jar destfile="${hello_jar}" basedir="classes"><!-- 向jar包中的main文件中添加内容 --><manifest><attribute name="Built-By" value="${user.name}"/><attribute name="Main-class" value="package.Main"/></manifest></jar><!-- 复制jar文件 todir="复制到目录"--><copy file="${hello_jar}" tofile="${dest}\temp.jar"/><copy todir="temp"><!-- 不按照默认方式 defaultexcludes="" --><fileset dir="src"><include name="**/*.java"/></fileset></copy><copy todir="temp2"><fileset dir="src"><and><contains text="main"/><size value="1" when="more"/></and></fileset></copy><!-- 移动jar文件 --><move file="${dest}\temp.jar" tofile="temp\move-temp.jar"/><!-- 创建zip --><zip basedir="${basedir}\classes" zipfile="temp\output.zip"/><!-- 创建tgz --><gzip src="classes\**\*.class" zipfile="output.class.gz"/><!-- 解压zip --><unzip src="output.class.gz" dest="extractDir"/><!--替换input.txt内容中的old为new<replace file="input.txt" token="old" value="new"/>--></target><!-- 运行 --><target name="run" depends="build"><java classname="com.hoo.test.HelloWorld" classpath="${hello_jar}"/></target><!-- 清除 --><target name="clean"><!-- 删除生成的文件 --><delete dir="${dest}"/><delete file="${hello_jar}"/></target><tstamp><format property="OFFSET_TIME"pattern="HH:mm:ss"offset="10" unit="minute"/></tstamp><!-- 重新运行 --><target name="rerun" depends="clean,run"><echo message="###${TSTAMP}#${TODAY}#${DSTAMP}###"/><aunt target="clean"/><aunt target="run"/></target></project>
要说的的东西基本上就那么多,ant还有很多内容没有提到,有兴趣的可以自己去研究研究。ant不难,你用它就像是在docs控制台输入命令行一样,只不过ant是将命令行转换为xml的脚本信息,可以进行重复的运行。在一定情况下,提高了效率和重复的工作。
ant在持续集成的应用的更多相关文章
- (转)使用CruiseControl+SVN+ANT实现持续集成之二
1. 环境搭建 1.1. 下载及目录介绍 从官方站点http://cruisecontrol.sourceforge.net/download.html下载一份最新的 CC 压缩包,最新的版本号为2. ...
- (转)使用SVN+CruiseControl+ANT实现持续集成之一
在前面的文章中, 介绍自己当时所在团队的处境(使用.NET开发),一个不到十个人的研发团队在保证正常开发进度同时需要并发支持四.五十个项目问题处理,经常为了程序版 本冲突.日常测试版本.发布版本提供等 ...
- Jenkins+svn+ant+tomcat持续集成
转载自 http://www.cnblogs.com/liuhaixia/p/7267473.html Jenkins是基于Java开发的一种持续集成工具,用于监控秩序重复的工作.通过Jenkins+ ...
- Jmeter+Ant+Jenkins持续集成方案改进
关于Jmeter+Ant+Jenkins如何搭建持续集成环境,网上资料一大把,就不多说了,本文主要谈一下期间的问题及扩展该持续集成方案. 其实核心的流程不复杂,Jenkins管理构建项目,Ant配置脚 ...
- Jenkins+JMeter+Ant 接口持续集成
JMeter安装 JMeter安装包: 下载地址:https://jmeter.apache.org/download_jmeter.cgi 依赖JDK环境 JDK环境配置: JAVA_HOME ...
- Jmeter接口测试自动化(jmeter+ant+jenkins持续集成)
Jmeter是压力测试.接口测试工具,Ant是基于Java的构建工具,具有跨平台的作用,jenkins是持续集成工具.将这三者结合起来可以搭建一套webservice接口测试的持续构建环境. 1. ...
- jenkins, ant, pmd持续集成
http://pmd.sourceforge.net/pmd-5.0.3/ant-task.html 在jenkins , ant , pmd进行集成的时候,build.xml模板如下,在网上找了一些 ...
- Jenkins+Jmeter+Ant接口持续集成
修改时间 修改内容 修改人 2016.5.22 创建 刘永志 2016.6.15 完成 刘永志 前言: 为什么要用Jmeter做接口测试: 当选择这套方案的时候,很多人会问,为什么选择Jmeter做C ...
- Jenkins+Jmeter+Ant 接口持续集成(转)
来源:https://testerhome.com/topics/5186 为什么要用Jmeter做接口测试 当选择这套方案的时候,很多人会问,为什么选择Jmeter做Case管理?为什么不自己写框架 ...
随机推荐
- Coursera 机器学习笔记(一)
主要是第一二周内容 机器学习概要 机器学习是什么? 生活在信息时代的我们,其实时时刻刻都离不开机器学习算法.比如日常使用的搜索引擎就涉及到很多学习算法. Arthur Samuel 给出第一个定义.他 ...
- XCode 8.3 Automatically manage signing 问题
记录iOS证书问题 无意中选择了Personal team,后台自动生成bundle identifier:com.xxx.project,导致不能在Organization team创建,而且在de ...
- 基于Jmeter的轻量级接口压力测试(一)
一.操作步骤: 1.在测试计划下新增一个线程组,并在线程组下新增一个http请求: 2.读取配置文件中的参数:在添加的http请求下添加配置元件-CSV DATA SET CONFIG 3.配置待测试 ...
- Bash Shell编程要点小结
一.case命令 case variable invalue1) command(s);; value2) command(s);; *) command(s);; esac 如果case变量没有被匹 ...
- Java中Bean是什么
javaBean在MVC设计模型中是model,又称模型层,在一般的程序中,我们称它为数据层,就是用来设置数据的属性和一些行为,然后我会提供获取属性和设置属性的get/set方法JavaBean是一种 ...
- thinkphp获取特定字段的两种方法
thinkphp getField( )和field( ) 2014年10月05日 ⁄ 综合 ⁄ 共 1509字 ⁄ 字号 小 中 大 ⁄ 评论关闭 做数据库查询的时候,比较经常用到这两个,总是查手册 ...
- 超好用的Redis管理及监控工具,使用后可大大提高你的工作效率!
Redis做为现在web应用开发的黄金搭担组合,大量的被应用,广泛用于存储session信息,权限信息,交易作业等热数据.做为一名有10年以上JAVA开发经验的程序员,工作中项目也是广泛使用了Redi ...
- MYSQL数据类型和where条件
MySQL中常见的数据类型 一.字符型 ① CHAR(N):固定N个字符长度的字符串,如果长度不够自动空格补齐; N的范围 0~255 ② VARCHAR(N): 存储可变长度的字符串,最常用 ③ T ...
- 怎么用jq封装插件
怎么用jq封装插件 以隔行变色为例 实现原理:1.找到表格的奇偶行,然后添加不同的class,激活行高亮显示也很简单,只要判断mouseover事件,然后添加一个class,mouseout的时候,再 ...
- 【LeetCode】112. Path Sum
题目: Given a binary tree and a sum, determine if the tree has a root-to-leaf path such that adding up ...