• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# -*- coding: utf-8 -*-
2# Copyright 2013 Google Inc. All Rights Reserved.
3#
4# Licensed under the Apache License, Version 2.0 (the "License");
5# you may not use this file except in compliance with the License.
6# You may obtain a copy of the License at
7#
8#     http://www.apache.org/licenses/LICENSE-2.0
9#
10# Unless required by applicable law or agreed to in writing, software
11# distributed under the License is distributed on an "AS IS" BASIS,
12# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13# See the License for the specific language governing permissions and
14# limitations under the License.
15"""Tests for compose command."""
16
17from __future__ import absolute_import
18
19from gslib.commands.compose import MAX_COMPOSE_ARITY
20from gslib.cs_api_map import ApiSelector
21import gslib.tests.testcase as testcase
22from gslib.tests.testcase.integration_testcase import SkipForS3
23from gslib.tests.util import ObjectToURI as suri
24
25
26@SkipForS3('S3 does not support object composition.')
27class TestCompose(testcase.GsUtilIntegrationTestCase):
28  """Integration tests for compose command."""
29
30  def check_n_ary_compose(self, num_components):
31    """Tests composing num_components object."""
32    bucket_uri = self.CreateBucket()
33
34    data_list = ['data-%d,' % i for i in xrange(num_components)]
35    components = [self.CreateObject(bucket_uri=bucket_uri, contents=data).uri
36                  for data in data_list]
37
38    composite = bucket_uri.clone_replace_name(self.MakeTempName('obj'))
39
40    self.RunGsUtil(['compose'] + components + [composite.uri])
41    self.assertEqual(composite.get_contents_as_string(), ''.join(data_list))
42
43  def test_compose_too_many_fails(self):
44    components = ['gs://b/component-obj'] * (MAX_COMPOSE_ARITY + 1)
45    stderr = self.RunGsUtil(['compose'] + components + ['gs://b/composite-obj'],
46                            expected_status=1, return_stderr=True)
47    self.assertIn('command accepts at most', stderr)
48
49  def test_compose_too_few_fails(self):
50    stderr = self.RunGsUtil(
51        ['compose', 'gs://b/component-obj', 'gs://b/composite-obj'],
52        expected_status=1, return_stderr=True)
53    self.assertIn(
54        'CommandException: "compose" requires at least 2 component objects.\n',
55        stderr)
56
57  def test_compose_between_buckets_fails(self):
58    target = 'gs://b/composite-obj'
59    offending_obj = 'gs://alt-b/obj2'
60    components = ['gs://b/obj1', offending_obj]
61    stderr = self.RunGsUtil(['compose'] + components + [target],
62                            expected_status=1, return_stderr=True)
63    expected_msg = (
64        'CommandException: GCS does '
65        'not support inter-bucket composing.\n')
66    self.assertIn(expected_msg, stderr)
67
68  def test_versioned_target_disallowed(self):
69    stderr = self.RunGsUtil(
70        ['compose', 'gs://b/o1', 'gs://b/o2', 'gs://b/o3#1234'],
71        expected_status=1, return_stderr=True)
72    expected_msg = ('CommandException: A version-specific URL (%s) '
73                    'cannot be the destination for gsutil compose - abort.'
74                    % 'gs://b/o3#1234')
75    self.assertIn(expected_msg, stderr)
76
77  def test_simple_compose(self):
78    self.check_n_ary_compose(2)
79
80  def test_maximal_compose(self):
81    self.check_n_ary_compose(MAX_COMPOSE_ARITY)
82
83  def test_compose_with_wildcard(self):
84    """Tests composing objects with a wildcarded URI."""
85    bucket_uri = self.CreateBucket()
86
87    component1 = self.CreateObject(
88        bucket_uri=bucket_uri, contents='hello ', object_name='component1')
89    component2 = self.CreateObject(
90        bucket_uri=bucket_uri, contents='world!', object_name='component2')
91
92    composite = bucket_uri.clone_replace_name(self.MakeTempName('obj'))
93
94    self.RunGsUtil(['compose', component1.uri, component2.uri, composite.uri])
95    self.assertEqual(composite.get_contents_as_string(), 'hello world!')
96
97  def test_compose_with_precondition(self):
98    """Tests composing objects with a destination precondition."""
99    # Tests that cp -v option handles the if-generation-match header correctly.
100    bucket_uri = self.CreateVersionedBucket()
101    k1_uri = self.CreateObject(bucket_uri=bucket_uri, contents='data1')
102    k2_uri = self.CreateObject(bucket_uri=bucket_uri, contents='data2')
103    g1 = k1_uri.generation
104
105    gen_match_header = 'x-goog-if-generation-match:%s' % g1
106    # Append object 1 and 2
107    self.RunGsUtil(['-h', gen_match_header, 'compose', suri(k1_uri),
108                    suri(k2_uri), suri(k1_uri)])
109
110    # Second compose should fail the precondition.
111    stderr = self.RunGsUtil(['-h', gen_match_header, 'compose', suri(k1_uri),
112                             suri(k2_uri), suri(k1_uri)],
113                            return_stderr=True, expected_status=1)
114
115    self.assertIn('PreconditionException', stderr)
116
117  def test_compose_missing_second_source_object(self):
118    bucket_uri = self.CreateBucket()
119    object_uri = self.CreateObject(bucket_uri=bucket_uri, contents='foo')
120
121    # Compose with missing source object
122    stderr = self.RunGsUtil(['compose', suri(object_uri),
123                             suri(bucket_uri, 'nonexistent-obj'),
124                             suri(bucket_uri, 'valid-destination')],
125                            expected_status=1, return_stderr=True)
126    self.assertIn('NotFoundException', stderr)
127    if self.test_api == ApiSelector.JSON:
128      self.assertIn('One of the source objects does not exist', stderr)
129
130
131class TestCompatibleCompose(testcase.GsUtilIntegrationTestCase):
132
133  def test_compose_non_gcs_target(self):
134    stderr = self.RunGsUtil(['compose', 'gs://b/o1', 'gs://b/o2', 's3://b/o3'],
135                            expected_status=1, return_stderr=True)
136    expected_msg = ('CommandException: "compose" called on URL with '
137                    'unsupported provider (%s).\n' % 's3://b/o3')
138    self.assertIn(expected_msg, stderr)
139
140  def test_compose_non_gcs_component(self):
141    stderr = self.RunGsUtil(['compose', 'gs://b/o1', 's3://b/o2', 'gs://b/o3'],
142                            expected_status=1, return_stderr=True)
143    expected_msg = ('CommandException: "compose" called on URL with '
144                    'unsupported provider (%s).\n' % 's3://b/o2')
145    self.assertIn(expected_msg, stderr)
146
147