[Docker] archive: appends '/' at end of directories

This ensure to correctly detect duplicate for directory entries
that might or might not contains ending slash.

--
MOS_MIGRATED_REVID=104502468
diff --git a/tools/build_defs/docker/BUILD b/tools/build_defs/docker/BUILD
index 921d347..91ea88c 100644
--- a/tools/build_defs/docker/BUILD
+++ b/tools/build_defs/docker/BUILD
@@ -36,6 +36,7 @@
     "//tools/build_defs/docker/testdata:data_path_image",
     "//tools/build_defs/docker/testdata:no_data_path_image",
     "//tools/build_defs/docker/testdata:dummy_repository",
+    "//tools/build_defs/docker/testdata:extras_with_deb",
 ]
 
 sh_test(
diff --git a/tools/build_defs/docker/archive.py b/tools/build_defs/docker/archive.py
index b2c2b38..0a32e40 100644
--- a/tools/build_defs/docker/archive.py
+++ b/tools/build_defs/docker/archive.py
@@ -156,6 +156,9 @@
 
   def _addfile(self, info, fileobj=None):
     """Add a file in the tar file if there is no conflict."""
+    if not info.name.endswith('/') and info.type == tarfile.DIRTYPE:
+      # Enforce the ending / for directories so we correctly deduplicate.
+      info.name += '/'
     if info.name not in self.members:
       self.tar.addfile(info, fileobj)
       self.members.add(info.name)
diff --git a/tools/build_defs/docker/build_test.sh b/tools/build_defs/docker/build_test.sh
index 9d9d142..409d6e2 100755
--- a/tools/build_defs/docker/build_test.sh
+++ b/tools/build_defs/docker/build_test.sh
@@ -372,4 +372,21 @@
     ./test/test
 }
 
+function test_extras_with_deb() {
+  local test_data="${TEST_DATA_DIR}/extras_with_deb.tar"
+  local sha=$(tar xOf ${test_data} ./top)
+
+  # The content of the layer should have no duplicate
+  local layer_listing="$(get_layer_listing "extras_with_deb" "${sha}" | sort)"
+  check_eq "${layer_listing}" \
+"./
+./etc/
+./etc/nsswitch.conf
+./tmp/
+./usr/
+./usr/bin/
+./usr/bin/java -> /path/to/bin/java
+./usr/titi"
+}
+
 run_suite "build_test"
diff --git a/tools/build_defs/docker/testdata/BUILD b/tools/build_defs/docker/testdata/BUILD
index 9ea8f16..c18fbfa 100644
--- a/tools/build_defs/docker/testdata/BUILD
+++ b/tools/build_defs/docker/testdata/BUILD
@@ -291,3 +291,25 @@
         "/usr/bin/java": "/bar",
     },
 )
+
+# Generate a dummy debian package with a test/ directory
+py_binary(
+    name = "gen_deb",
+    srcs = ["gen_deb.py"],
+)
+
+genrule(
+    name = "generate_deb",
+    outs = ["gen.deb"],
+    cmd = "$(location :gen_deb) $@",
+    tools = [":gen_deb"],
+)
+
+# Bot gen.deb and test-data has a test directory, it should appear
+# only once in the resulting layer.
+docker_build(
+    name = "extras_with_deb",
+    data_path = ".",
+    debs = [":gen.deb"],
+    tars = ["extras.tar"],
+)
diff --git a/tools/build_defs/docker/testdata/gen_deb.py b/tools/build_defs/docker/testdata/gen_deb.py
new file mode 100644
index 0000000..7468e22
--- /dev/null
+++ b/tools/build_defs/docker/testdata/gen_deb.py
@@ -0,0 +1,60 @@
+# Copyright 2015 The Bazel Authors. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#    http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+"""A simple cross-platform helper to create a dummy debian package."""
+from StringIO import StringIO
+import sys
+import tarfile
+
+
+def AddArFileEntry(fileobj, filename, content=''):
+  """Add a AR file entry to fileobj."""
+  fileobj.write((filename + '/').ljust(16))    # filename (SysV)
+  fileobj.write('0'.ljust(12))                 # timestamp
+  fileobj.write('0'.ljust(6))                  # owner id
+  fileobj.write('0'.ljust(6))                  # group id
+  fileobj.write('0644'.ljust(8))               # mode
+  fileobj.write(str(len(content)).ljust(10))   # size
+  fileobj.write('\x60\x0a')                    # end of file entry
+  fileobj.write(content)
+  if len(content) % 2 != 0:
+    fileobj.write('\n')  # 2-byte alignment padding
+
+if __name__ == '__main__':
+  # Create data.tar
+  tar = StringIO()
+  with tarfile.open('data.tar', mode='w', fileobj=tar) as f:
+    tarinfo = tarfile.TarInfo('usr/')
+    tarinfo.type = tarfile.DIRTYPE
+    f.addfile(tarinfo)
+    tarinfo = tarfile.TarInfo('usr/titi')
+    f.addfile(tarinfo, fileobj=StringIO('toto\n'))
+  data = tar.getvalue()
+  tar.close()
+  # Create control.tar
+  tar = StringIO()
+  with tarfile.open('control.tar', mode='w', fileobj=tar) as f:
+    tarinfo = tarfile.TarInfo('control')
+    f.addfile(tarinfo, fileobj=StringIO('\n'.join([
+        'Package: test'
+        'Description: Just a dummy test'
+        ])))
+  control = tar.getvalue()
+  tar.close()
+
+  # Write the final AR archive (the deb package)
+  with open(sys.argv[1], 'w') as f:
+    f.write('!<arch>\n')  # Magic AR header
+    AddArFileEntry(f, 'debian-binary', '2.0')
+    AddArFileEntry(f, 'control.tar', control)
+    AddArFileEntry(f, 'data.tar', data)