blob: ae750efb4fe387c7988bd402fa0ad7d2529e31e2 [file] [log] [blame]
#
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you 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
#
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
import os
import subprocess
import time
import unittest
import test_subprocess
# Check if we can run prlimit to control resources
try:
assert subprocess.check_call(["prlimit"], stdout=open(os.devnull, 'w')) == 0, 'prlimit is present, but broken'
prlimit_available = True
except OSError:
prlimit_available = False
class PRLimitedBroker(test_subprocess.Server):
def __init__(self, fdlimit, *args, **kwargs):
super(PRLimitedBroker, self).__init__(
['prlimit', '-n{0:d}:'.format(fdlimit), "broker", "", "0"], # `-n 256:` sets only soft limit to 256
stdout=subprocess.PIPE, universal_newlines=True, *args, **kwargs)
self.fdlimit = fdlimit
class FdLimitTest(unittest.TestCase):
devnull = open(os.devnull, 'w')
@classmethod
def tearDownClass(cls):
if cls.devnull:
cls.devnull.close()
@unittest.skipUnless(prlimit_available, "prlimit not available")
def test_fd_limit_broker(self):
"""Check behaviour when running out of file descriptors on accept"""
# Not too many FDs but not too few either, some are used for system purposes.
fdlimit = 256
with PRLimitedBroker(fdlimit, kill_me=True) as b:
receivers = []
# Start enough receivers to use all FDs
# NOTE: broker does not log a file descriptor related error at any point in the test, only
# PN_TRANSPORT_CLOSED: amqp:connection:framing-error: connection aborted
# PN_TRANSPORT_CLOSED: proton:io: Connection reset by peer - disconnected :5672 (connection aborted)
for i in range(fdlimit):
receiver = subprocess.Popen(["receive", "", b.port, str(i)], stdout=self.devnull, stderr=subprocess.STDOUT)
receivers.append(receiver)
# Allow these subprocesses time to establish ahead of the upcoming test sender.
time.sleep(1)
# All FDs are now in use, new send should not succeed. May fail by hanging (epoll) or by
# immediate failure (libuv). But poll() should never be 0.
sender = subprocess.Popen(["send", "", b.port, "x"],
stdout=self.devnull, stderr=subprocess.STDOUT)
time.sleep(1) # polling for None immediately would always succeed, regardless whether send hangs or not
self.assertNotEqual(sender.poll(), 0)
# Kill receivers to free up FDs
for r in receivers:
r.kill()
for r in receivers:
r.wait()
# Sender completes on its own
sender.wait()
# Additional send/receive should succeed now
self.assertIn("10 messages sent", test_subprocess.check_output(
["send", "", b.port], universal_newlines=True))
self.assertIn("10 messages received", test_subprocess.check_output(
["receive", "", b.port], universal_newlines=True))
if __name__ == "__main__":
unittest.main()