Problem p01089 - Generation 1

Orig Description

Deadlock Detection
In concurrent processing environments, a deadlock is an undesirable
situation where two or more threads are mutually waiting for others to
finish using some resources and cannot proceed further.
Your task is to detect whether there is any possibility of deadlocks
when multiple threads try to execute a given instruction sequence concurrently.
The instruction sequence consists of characters 'u' or digits from
'0' to '9', and each of them represents one instruction.
10 threads are trying to execute the same single instruction sequence.
Each thread starts its execution from the beginning of the sequence and
continues in the given order, until all the instructions are
executed.
There are 10 shared resources called locks from L0 to L9.
A digit k is the instruction for acquiring the lock Lk.
After one of the threads acquires a lock Lk,
it is kept by the thread until it is released by the instruction 'u'.
While a lock is kept, none of the threads, including one already acquired it,
cannot newly acquire the same lock Lk.
Precisely speaking, the following steps are repeated until all threads
finish.
One thread that has not finished yet is chosen arbitrarily.
The chosen thread tries to execute the next instruction that is not executed yet.
If the next instruction is a digit k and
		the lock Lk is not kept by any thread,
		the thread executes the instruction k and acquires Lk.
If the next instruction is a digit k and
		the lock Lk is already kept by some thread,
		the instruction k is not executed.
If the next instruction is 'u',
		the instruction is executed and all the locks currently kept by the thread
		are released.
After executing several steps, sometimes, it falls into the situation 
that the next instructions
of all unfinished threads are for acquiring already kept locks.
Once such a situation happens, no instruction will ever be executed no 
matter which thread is chosen. This situation is called a deadlock.
There are instruction sequences for which threads can never reach a
deadlock regardless of the execution order.
Such instruction sequences are called safe.
Otherwise, in other words, if there exists one or more execution orders
that lead to a deadlock, the execution sequence is called unsafe.
Your task is to write a program that tells whether the given instruction sequence is safe or unsafe.
Input
The input consists of at most 50 datasets, each in the following format.
n
s
n is the length of the instruction sequence and s is a string representing the sequence.
n is a positive integer not exceeding 10,000.
Each character of s is either a digit ('0' to '9') or 'u',
and s always ends with 'u'.
The end of the input is indicated by a line with a zero.
Output
For each dataset, if the given instruction sequence is safe, then print "SAFE" in a line.
If it is unsafe, then print "UNSAFE" in a line.
Sample Input
11
01u12u0123u
6
01u10u
8
201u210u
9
01u12u20u
3
77u
12
9u8u845u954u
0
Output for the Sample Input
SAFE
UNSAFE
SAFE
UNSAFE
UNSAFE
UNSAFE
The second input "01u10u" may possibly cause a deadlock.  After one
thread has executed the initial four instructions "01u1", the
thread keeps only one lock L1.  If another thread executes the first
instruction '0' at this time, this thread acquires L0.  Then, the
first thread tries to acquire L0, already kept by the second
thread, while the second thread tries to acquire L1, kept by the
first thread; This leads to a deadlock.
 →
 →
Figure 1: Why the Sample Input 2 "01u10u" is unsafe.
Contrarily, the third input "201u210u" is safe.
If one thread had executed up to "201u21" and another to "20", then
one may think it would lead to a deadlock, but this
can never happen because no two threads can simultaneously keep L2.

Extracted Specification

An integer n (1 ≤ n ≤ 10,000), representing some length.
A string s of length n, where each character is either a digit ('0' to '9') or 'u', and s always ends with 'u'.

### Example Input:

```
11
01u12u0123u
```

### Function Signature:
Write a function f(inputs) that takes in the input.
def f(inputs: List[Tuple[int, str]]):
    '''
    inputs: a list of tuples, where each tuple consists of an integer and a string
    '''

Test Case Generator

Sure! Here's a test case generator for the "Deadlock Detection" problem:

```python
import random
import string

def tcgen():
    # Generate a random length for the instruction sequence
    n = random.randint(1, 10000)
    
    # Generate the instruction sequence
    # Ensure the sequence ends with 'u'
    s = ''.join(random.choices('0123456789u', k=n-1)) + 'u'
    
    return n, s

# Generate multiple test cases
def generate_test_cases(num_cases):
    test_cases = []
    for _ in range(num_cases):
        n, s = tcgen()
        test_cases.append((n, s))
    return test_cases

# Example usage:
if __name__ == "__main__":
    num_cases = 5  # Number of test cases you want to generate
    test_cases = generate_test_cases(num_cases)
    
    for n, s in test_cases:
        print(n)
        print(s)
```

### Explanation:

1. **Random Length (`n`):** The length of the instruction sequence `n` is generated randomly between 1 and 10,000.
2. **Instruction Sequence (`s`):** The sequence is generated using `random.choices` to choose from digits `0-9` and `u`. The sequence is ensured to end with `u` by appending 'u' at the end.
3. **Multiple Test Cases:** The `generate_test_cases` function generates multiple test cases by calling `tcgen()` multiple times.
4. **Example Usage:** The example usage generates 5 test cases and prints them.

This will help you generate a variety of test cases that can be used to thoroughly test the deadlock detection logic.

Extract Arguments

def extract_arguments(fh):
    datasets = []
    while True:
        n = int(fh.readline().strip())
        if n == 0:
            break
        s = fh.readline().strip()
        datasets.append((n, s))
    return datasets