{
    "question": "You are given a tree consisting of $n$ vertices. A number is written on each vertex; the number on vertex $i$ is equal to $a_i$.\n\nLet's denote the function $g(x, y)$ as the greatest common divisor of the numbers written on the vertices belonging to the simple path from vertex $x$ to vertex $y$ (including these two vertices). Also let's denote $dist(x, y)$ as the number of vertices on the simple path between vertices $x$ and $y$, including the endpoints. $dist(x, x) = 1$ for every vertex $x$.\n\nYour task is calculate the maximum value of $dist(x, y)$ among such pairs of vertices that $g(x, y) > 1$.\n\n\n-----Input-----\n\nThe first line contains one integer $n$ \u2014 the number of vertices $(1 \\le n \\le 2 \\cdot 10^5)$.\n\nThe second line contains $n$ integers $a_1$, $a_2$, ..., $a_n$ $(1 \\le a_i \\le 2 \\cdot 10^5)$ \u2014 the numbers written on vertices.\n\nThen $n - 1$ lines follow, each containing two integers $x$ and $y$ $(1 \\le x, y \\le n, x \\ne y)$ denoting an edge connecting vertex $x$ with vertex $y$. It is guaranteed that these edges form a tree.\n\n\n-----Output-----\n\nIf there is no pair of vertices $x, y$ such that $g(x, y) > 1$, print $0$. Otherwise print the maximum value of $dist(x, y)$ among such pairs.\n\n\n-----Examples-----\nInput\n3\n2 3 4\n1 2\n2 3\n\nOutput\n1\n\nInput\n3\n2 3 4\n1 3\n2 3\n\nOutput\n2\n\nInput\n3\n1 1 1\n1 2\n2 3\n\nOutput\n0",
    "plan": "```python\nimport sys\n\ndef sieve_of_eratosthenes(limit):\n    \"\"\"\n    Generate a list of prime numbers up to the specified limit using the Sieve of Eratosthenes algorithm.\n\n    The Sieve of Eratosthenes is an efficient algorithm for finding all prime numbers up to a specified integer. \n    It works by iteratively marking the multiples of each prime number starting from 2.\n\n    Parameters:\n    - limit: An integer representing the upper bound up to which prime numbers are to be found.\n\n    Returns:\n    - primes: A list of integers where each integer is a prime number up to the specified limit.\n    \"\"\"\n\ndef factorize_numbers(a, primes):\n    \"\"\"\n    Factorize each number in the list 'a' using a given list of known prime numbers.\n\n    This function breaks down each number in the list 'a' into its prime factors. It uses the provided list of primes\n    to check divisibility, and constructs a list of tuples where each tuple contains a prime factor and its exponent.\n\n    Parameters:\n    - a: A list of integers representing the numbers to be factorized.\n    - primes: A list of known prime numbers used for factorization.\n\n    Returns:\n    - factorized: A list of lists where each inner list contains tuples representing the prime factors and their exponents for each number in 'a'.\n    \"\"\"\n\ndef find_leaves(neighbours):\n    \"\"\"\n    Find the leaf nodes in a tree.\n\n    Leaf nodes are vertices with only one adjacent vertex. This function identifies all such nodes in a tree represented by an adjacency list.\n\n    Parameters:\n    - neighbours: A list of lists where each inner list contains the indices of adjacent vertices for each vertex in the tree.\n\n    Returns:\n    - leaves: A list of integers where each integer represents the index of a leaf node in the tree.\n    \"\"\"\n\ndef update_max_distance(a, neighbours, leaves):\n    \"\"\"\n    Update and calculate the maximum distance between pairs of vertices with a GCD greater than 1.\n\n    This function processes the tree by repeatedly removing leaf nodes and updating the maximum distance (number of vertices) between pairs of vertices\n    whose path has a GCD greater than 1. It uses the factorized numbers of each vertex to compute the GCD and update distances accordingly.\n\n    Parameters:\n    - a: A list of lists representing the factorized numbers for each vertex, where each inner list contains tuples of prime factors and their exponents.\n    - neighbours: A list of lists representing the adjacency list of the tree.\n    - leaves: A list of integers representing the indices of leaf nodes in the tree.\n\n    Returns:\n    - answer: An integer representing the maximum distance between pairs of vertices such that the GCD of the numbers on the path between them is greater than 1.\n    \"\"\"\n```",
    "plan_code": "```python\nimport sys\n\ndef sieve_of_eratosthenes(limit):\n    \"\"\"\n    Generate a list of prime numbers up to the specified limit using the Sieve of Eratosthenes algorithm.\n\n    The Sieve of Eratosthenes is an efficient algorithm for finding all prime numbers up to a specified integer. \n    It works by iteratively marking the multiples of each prime number starting from 2.\n\n    Parameters:\n    - limit: An integer representing the upper bound up to which prime numbers are to be found.\n\n    Returns:\n    - primes: A list of integers where each integer is a prime number up to the specified limit.\n    \"\"\"\n\ndef factorize_numbers(a, primes):\n    \"\"\"\n    Factorize each number in the list 'a' using a given list of known prime numbers.\n\n    This function breaks down each number in the list 'a' into its prime factors. It uses the provided list of primes\n    to check divisibility, and constructs a list of tuples where each tuple contains a prime factor and its exponent.\n\n    Parameters:\n    - a: A list of integers representing the numbers to be factorized.\n    - primes: A list of known prime numbers used for factorization.\n\n    Returns:\n    - factorized: A list of lists where each inner list contains tuples representing the prime factors and their exponents for each number in 'a'.\n    \"\"\"\n\ndef find_leaves(neighbours):\n    \"\"\"\n    Find the leaf nodes in a tree.\n\n    Leaf nodes are vertices with only one adjacent vertex. This function identifies all such nodes in a tree represented by an adjacency list.\n\n    Parameters:\n    - neighbours: A list of lists where each inner list contains the indices of adjacent vertices for each vertex in the tree.\n\n    Returns:\n    - leaves: A list of integers where each integer represents the index of a leaf node in the tree.\n    \"\"\"\n\ndef update_max_distance(a, neighbours, leaves):\n    \"\"\"\n    Update and calculate the maximum distance between pairs of vertices with a GCD greater than 1.\n\n    This function processes the tree by repeatedly removing leaf nodes and updating the maximum distance (number of vertices) between pairs of vertices\n    whose path has a GCD greater than 1. It uses the factorized numbers of each vertex to compute the GCD and update distances accordingly.\n\n    Parameters:\n    - a: A list of lists representing the factorized numbers for each vertex, where each inner list contains tuples of prime factors and their exponents.\n    - neighbours: A list of lists representing the adjacency list of the tree.\n    - leaves: A list of integers representing the indices of leaf nodes in the tree.\n\n    Returns:\n    - answer: An integer representing the maximum distance between pairs of vertices such that the GCD of the numbers on the path between them is greater than 1.\n    \"\"\"\n```",
    "solution": "import sys\n\ndef sieve_of_eratosthenes(limit):\n    \"\"\"\n    Generate a list of prime numbers up to the specified limit using the Sieve of Eratosthenes algorithm.\n\n    The Sieve of Eratosthenes is an efficient algorithm for finding all prime numbers up to a specified integer. \n    It works by iteratively marking the multiples of each prime number starting from 2.\n\n    Parameters:\n    - limit: An integer representing the upper bound up to which prime numbers are to be found.\n\n    Returns:\n    - primes: A list of integers where each integer is a prime number up to the specified limit.\n    \"\"\"\n    primes = []\n    is_prime = [True] * (limit + 1)\n    for p in range(2, limit + 1):\n        if is_prime[p]:\n            primes.append(p)\n            for i in range(p * p, limit + 1, p):\n                is_prime[i] = False\n    return primes\n\n\ndef factorize_numbers(a, primes):\n    \"\"\"\n    Factorize each number in the list 'a' using a given list of known prime numbers.\n\n    This function breaks down each number in the list 'a' into its prime factors. It uses the provided list of primes\n    to check divisibility, and constructs a list of tuples where each tuple contains a prime factor and its exponent.\n\n    Parameters:\n    - a: A list of integers representing the numbers to be factorized.\n    - primes: A list of known prime numbers used for factorization.\n\n    Returns:\n    - factorized: A list of lists where each inner list contains tuples representing the prime factors and their exponents for each number in 'a'.\n    \"\"\"\n    factorized = []\n    for x in a:\n        factors = []\n        for p in primes:\n            if x % p == 0:\n                while x % p == 0:\n                    x //= p\n                factors.append((p, 1))\n        if x != 1:\n            factors.append((x, 1))\n        factorized.append(factors)\n    return factorized\n\n\ndef find_leaves(neighbours):\n    \"\"\"\n    Find the leaf nodes in a tree.\n\n    Leaf nodes are vertices with only one adjacent vertex. This function identifies all such nodes in a tree represented by an adjacency list.\n\n    Parameters:\n    - neighbours: A list of lists where each inner list contains the indices of adjacent vertices for each vertex in the tree.\n\n    Returns:\n    - leaves: A list of integers where each integer represents the index of a leaf node in the tree.\n    \"\"\"\n    leaves = []\n    for i, neighbours_list in enumerate(neighbours):\n        if len(neighbours_list) == 1:\n            leaves.append(i)\n    return leaves\n\n\ndef update_max_distance(a, neighbours, leaves):\n    \"\"\"\n    Update and calculate the maximum distance between pairs of vertices with a GCD greater than 1.\n\n    This function processes the tree by repeatedly removing leaf nodes and updating the maximum distance (number of vertices) between pairs of vertices\n    whose path has a GCD greater than 1. It uses the factorized numbers of each vertex to compute the GCD and update distances accordingly.\n\n    Parameters:\n    - a: A list of lists representing the factorized numbers for each vertex, where each inner list contains tuples of prime factors and their exponents.\n    - neighbours: A list of lists representing the adjacency list of the tree.\n    - leaves: A list of integers representing the indices of leaf nodes in the tree.\n\n    Returns:\n    - answer: An integer representing the maximum distance between pairs of vertices such that the GCD of the numbers on the path between them is greater than 1.\n    \"\"\"\n    answer = 1\n    while len(leaves) > 1:\n        x = leaves.pop()\n        y = neighbours[x][0]\n        neighbours[y].remove(x)\n\n        if len(neighbours[y]) == 1:\n            leaves.append(y)\n\n        for p in a[x]:\n            for q in a[y]:\n                if p[0] == q[0]:\n                    answer = max(answer, p[1] + q[1])\n                    q[1] = max(q[1], p[1] + 1)\n\n    return answer\n\n\nif __name__ == \"__main__\":\n    n = int(sys.stdin.readline().strip())\n    a = list(map(int, sys.stdin.readline().strip().split()))\n    edges = [list(map(int, sys.stdin.readline().strip().split())) for _ in range(n - 1)]\n    for i in range(len(edges)):\n        edges[i][0] -= 1\n        edges[i][1] -= 1\n\n    answer = 1\n    primes = sieve_of_eratosthenes(500)\n    a = factorize_numbers(a, primes)\n    neighbours = [[] for _ in range(n)]\n    for x, y in edges:\n        neighbours[x].append(y)\n        neighbours[y].append(x)\n\n    leaves = find_leaves(neighbours)\n\n    if sum(a) == n:\n        print(0)\n    else:\n        print(update_max_distance(a, neighbours, leaves))",
    "plain_solution": "import sys\n\ndef sieve_of_eratosthenes(limit):\n    primes = []\n    is_prime = [True] * (limit + 1)\n    for p in range(2, limit + 1):\n        if is_prime[p]:\n            primes.append(p)\n            for i in range(p * p, limit + 1, p):\n                is_prime[i] = False\n    return primes\n\ndef factorize_numbers(a, primes):\n    factorized = []\n    for x in a:\n        factors = []\n        for p in primes:\n            if x % p == 0:\n                while x % p == 0:\n                    x //= p\n                factors.append((p, 1))\n        if x != 1:\n            factors.append((x, 1))\n        factorized.append(factors)\n    return factorized\n\n\ndef find_leaves(neighbours):\n    leaves = []\n    for i, neighbours_list in enumerate(neighbours):\n        if len(neighbours_list) == 1:\n            leaves.append(i)\n    return leaves\n\n\ndef update_max_distance(a, neighbours, leaves):\n    answer = 1\n    while len(leaves) > 1:\n        x = leaves.pop()\n        y = neighbours[x][0]\n        neighbours[y].remove(x)\n\n        if len(neighbours[y]) == 1:\n            leaves.append(y)\n\n        for p in a[x]:\n            for q in a[y]:\n                if p[0] == q[0]:\n                    answer = max(answer, p[1] + q[1])\n                    q[1] = max(q[1], p[1] + 1)\n\n    return answer\n\n\nif __name__ == \"__main__\":\n    n = int(sys.stdin.readline().strip())\n    a = list(map(int, sys.stdin.readline().strip().split()))\n    edges = [list(map(int, sys.stdin.readline().strip().split())) for _ in range(n - 1)]\n    for i in range(len(edges)):\n        edges[i][0] -= 1\n        edges[i][1] -= 1\n\n    answer = 1\n    primes = sieve_of_eratosthenes(500)\n    a = factorize_numbers(a, primes)\n    neighbours = [[] for _ in range(n)]\n    for x, y in edges:\n        neighbours[x].append(y)\n        neighbours[y].append(x)\n\n    leaves = find_leaves(neighbours)\n\n    if sum(a) == n:\n        print(0)\n    else:\n        print(update_max_distance(a, neighbours, leaves))"
}
