Merge pull request #37813 from zhuzilin:autograph-loop-else-dev
PiperOrigin-RevId: 304383345 Change-Id: I9f5574e8c07edda607b8b43c662f32e045f0fba9
This commit is contained in:
commit
d6caf94b03
@ -80,28 +80,40 @@ class BreakTransformer(converter.Base):
|
|||||||
# A break in the else clause applies to the containing scope.
|
# A break in the else clause applies to the containing scope.
|
||||||
node.orelse = self.visit_block(node.orelse)
|
node.orelse = self.visit_block(node.orelse)
|
||||||
|
|
||||||
if break_used:
|
if not break_used:
|
||||||
# Python's else clause only triggers if the loop exited cleanly (e.g.
|
|
||||||
# break did not trigger).
|
|
||||||
guarded_orelse = self._guard_if_present(node.orelse, break_var)
|
|
||||||
|
|
||||||
template = """
|
template = """
|
||||||
var_name = False
|
while test:
|
||||||
while ag__.and_(lambda: test, lambda: ag__.not_(var_name)):
|
|
||||||
body
|
body
|
||||||
else:
|
orelse
|
||||||
orelse
|
|
||||||
"""
|
"""
|
||||||
node = templates.replace(
|
node = templates.replace(
|
||||||
template,
|
template, test=node.test, body=node.body, orelse=node.orelse)
|
||||||
var_name=break_var,
|
|
||||||
test=node.test,
|
|
||||||
body=node.body,
|
|
||||||
orelse=guarded_orelse)
|
|
||||||
|
|
||||||
new_while_node = node[1]
|
new_while_node = node[0]
|
||||||
anno.copyanno(original_node, new_while_node, anno.Basic.DIRECTIVES)
|
anno.copyanno(original_node, new_while_node, anno.Basic.DIRECTIVES)
|
||||||
|
|
||||||
|
return node
|
||||||
|
|
||||||
|
# Python's else clause only triggers if the loop exited cleanly (e.g.
|
||||||
|
# break did not trigger).
|
||||||
|
guarded_orelse = self._guard_if_present(node.orelse, break_var)
|
||||||
|
|
||||||
|
template = """
|
||||||
|
var_name = False
|
||||||
|
while ag__.and_(lambda: test, lambda: ag__.not_(var_name)):
|
||||||
|
body
|
||||||
|
orelse
|
||||||
|
"""
|
||||||
|
node = templates.replace(
|
||||||
|
template,
|
||||||
|
var_name=break_var,
|
||||||
|
test=node.test,
|
||||||
|
body=node.body,
|
||||||
|
orelse=guarded_orelse)
|
||||||
|
|
||||||
|
new_while_node = node[1]
|
||||||
|
anno.copyanno(original_node, new_while_node, anno.Basic.DIRECTIVES)
|
||||||
|
|
||||||
return node
|
return node
|
||||||
|
|
||||||
def visit_For(self, node):
|
def visit_For(self, node):
|
||||||
@ -115,37 +127,54 @@ class BreakTransformer(converter.Base):
|
|||||||
# A break in the else clause applies to the containing scope.
|
# A break in the else clause applies to the containing scope.
|
||||||
node.orelse = self.visit_block(node.orelse)
|
node.orelse = self.visit_block(node.orelse)
|
||||||
|
|
||||||
if break_used:
|
if not break_used:
|
||||||
# Python's else clause only triggers if the loop exited cleanly (e.g.
|
|
||||||
# break did not trigger).
|
|
||||||
guarded_orelse = self._guard_if_present(node.orelse, break_var)
|
|
||||||
extra_test = templates.replace_as_expression(
|
|
||||||
'ag__.not_(var_name)', var_name=break_var)
|
|
||||||
|
|
||||||
# The extra test is hidden in the AST, which will confuse the static
|
|
||||||
# analysis. To mitigate that, we insert a no-op statement that ensures
|
|
||||||
# the control variable is marked as used.
|
|
||||||
# TODO(mdan): Use a marker instead, e.g. ag__.condition_loop_on(var_name)
|
|
||||||
template = """
|
template = """
|
||||||
var_name = False
|
|
||||||
for target in iter_:
|
for target in iter_:
|
||||||
(var_name,)
|
|
||||||
body
|
body
|
||||||
else:
|
orelse
|
||||||
orelse
|
|
||||||
"""
|
"""
|
||||||
node = templates.replace(
|
node = templates.replace(
|
||||||
template,
|
template,
|
||||||
var_name=break_var,
|
|
||||||
iter_=node.iter,
|
iter_=node.iter,
|
||||||
target=node.target,
|
target=node.target,
|
||||||
body=node.body,
|
body=node.body,
|
||||||
orelse=guarded_orelse)
|
orelse=node.orelse)
|
||||||
|
|
||||||
new_for_node = node[1]
|
new_for_node = node[0]
|
||||||
anno.setanno(new_for_node, anno.Basic.EXTRA_LOOP_TEST, extra_test)
|
anno.copyanno(original_node, new_for_node, anno.Basic.EXTRA_LOOP_TEST)
|
||||||
anno.copyanno(original_node, new_for_node, anno.Basic.DIRECTIVES)
|
anno.copyanno(original_node, new_for_node, anno.Basic.DIRECTIVES)
|
||||||
|
|
||||||
|
return node
|
||||||
|
|
||||||
|
# Python's else clause only triggers if the loop exited cleanly (e.g.
|
||||||
|
# break did not trigger).
|
||||||
|
guarded_orelse = self._guard_if_present(node.orelse, break_var)
|
||||||
|
extra_test = templates.replace_as_expression(
|
||||||
|
'ag__.not_(var_name)', var_name=break_var)
|
||||||
|
|
||||||
|
# The extra test is hidden in the AST, which will confuse the static
|
||||||
|
# analysis. To mitigate that, we insert a no-op statement that ensures
|
||||||
|
# the control variable is marked as used.
|
||||||
|
# TODO(mdan): Use a marker instead, e.g. ag__.condition_loop_on(var_name)
|
||||||
|
template = """
|
||||||
|
var_name = False
|
||||||
|
for target in iter_:
|
||||||
|
(var_name,)
|
||||||
|
body
|
||||||
|
orelse
|
||||||
|
"""
|
||||||
|
node = templates.replace(
|
||||||
|
template,
|
||||||
|
var_name=break_var,
|
||||||
|
iter_=node.iter,
|
||||||
|
target=node.target,
|
||||||
|
body=node.body,
|
||||||
|
orelse=guarded_orelse)
|
||||||
|
|
||||||
|
new_for_node = node[1]
|
||||||
|
anno.setanno(new_for_node, anno.Basic.EXTRA_LOOP_TEST, extra_test)
|
||||||
|
anno.copyanno(original_node, new_for_node, anno.Basic.DIRECTIVES)
|
||||||
|
|
||||||
return node
|
return node
|
||||||
|
|
||||||
|
|
||||||
|
@ -0,0 +1,95 @@
|
|||||||
|
# Copyright 2020 The TensorFlow 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.
|
||||||
|
# ==============================================================================
|
||||||
|
"""Integration Tests for loop."""
|
||||||
|
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import division
|
||||||
|
from __future__ import print_function
|
||||||
|
|
||||||
|
from tensorflow.python.autograph.converters import break_statements
|
||||||
|
from tensorflow.python.autograph.converters import continue_statements
|
||||||
|
from tensorflow.python.autograph.converters import control_flow
|
||||||
|
from tensorflow.python.autograph.core import converter_testing
|
||||||
|
from tensorflow.python.framework import constant_op
|
||||||
|
from tensorflow.python.platform import test
|
||||||
|
|
||||||
|
|
||||||
|
class LoopIntegrationTest(converter_testing.TestCase):
|
||||||
|
|
||||||
|
def assertTransformedEquivalent(self, test_fn, *inputs):
|
||||||
|
with self.converted(test_fn,
|
||||||
|
[break_statements, continue_statements, control_flow],
|
||||||
|
{}, (constant_op.constant,)) as result:
|
||||||
|
self.assertEqual(test_fn(*inputs), result.test_fn(*inputs))
|
||||||
|
|
||||||
|
def test_while_loop_with_else(self):
|
||||||
|
|
||||||
|
def test_fn(x):
|
||||||
|
while x > 2:
|
||||||
|
x /= 2
|
||||||
|
else:
|
||||||
|
x += 1
|
||||||
|
return x
|
||||||
|
|
||||||
|
self.assertTransformedEquivalent(test_fn, 4)
|
||||||
|
self.assertTransformedEquivalent(test_fn, 2)
|
||||||
|
|
||||||
|
def test_while_loop_with_else_and_break(self):
|
||||||
|
|
||||||
|
def test_fn(cond1):
|
||||||
|
x = 8
|
||||||
|
while x > 2:
|
||||||
|
x /= 2
|
||||||
|
if cond1:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
x += 1
|
||||||
|
return x
|
||||||
|
|
||||||
|
self.assertTransformedEquivalent(test_fn, True)
|
||||||
|
self.assertTransformedEquivalent(test_fn, False)
|
||||||
|
|
||||||
|
def test_for_loop_with_else(self):
|
||||||
|
|
||||||
|
def test_fn(l):
|
||||||
|
res = 0
|
||||||
|
for x in l:
|
||||||
|
res += x
|
||||||
|
else:
|
||||||
|
res += 1
|
||||||
|
return res
|
||||||
|
|
||||||
|
self.assertTransformedEquivalent(test_fn, [])
|
||||||
|
self.assertTransformedEquivalent(test_fn, [1, 2])
|
||||||
|
|
||||||
|
def test_for_loop_with_else_and_break(self):
|
||||||
|
|
||||||
|
def test_fn(flag):
|
||||||
|
l = [1, 2, 3]
|
||||||
|
res = 0
|
||||||
|
for x in l:
|
||||||
|
res += x
|
||||||
|
if flag:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
res += 1
|
||||||
|
return res
|
||||||
|
|
||||||
|
self.assertTransformedEquivalent(test_fn, True)
|
||||||
|
self.assertTransformedEquivalent(test_fn, False)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
test.main()
|
Loading…
x
Reference in New Issue
Block a user