# SimQN: a discrete-event simulator for the quantum networks
# Copyright (C) 2021-2022 Lutong Chen, Jian Li, Kaiping Xue
# University of Science and Technology of China, USTC.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
from typing import List, Union
from qns.simulator import Simulator
from qns.simulator import Event
from qns.entity import Entity
from qns.entity.node.app import Application
[docs]class QNode(Entity):
"""
QNode is a quantum node in the quantum network
"""
def __init__(self, name: str = None, apps: List[Application] = None):
"""
Args:
name (str): the node's name
apps (List[Application]): the installing applications.
"""
super().__init__(name=name)
self.network = None
self.cchannels = []
self.qchannels = []
self.memories = []
self.operators = []
self.croute_table = []
self.qroute_table = []
self.requests = []
if apps is None:
self.apps: List[Application] = []
else:
self.apps: List[Application] = apps
[docs] def install(self, simulator: Simulator) -> None:
super().install(simulator)
# initiate sub-entities
for cchannel in self.cchannels:
from qns.entity import ClassicChannel
assert (isinstance(cchannel, ClassicChannel))
cchannel.install(simulator)
for qchannel in self.qchannels:
from qns.entity import QuantumChannel
assert (isinstance(qchannel, QuantumChannel))
qchannel.install(simulator)
for memory in self.memories:
from qns.entity import QuantumMemory
assert (isinstance(memory, QuantumMemory))
memory.install(simulator)
for operator in self.operators:
from qns.entity import QuantumOperator
assert (isinstance(operator, QuantumOperator))
operator.install(simulator)
# initiate applications
for app in self.apps:
app.install(self, simulator)
[docs] def handle(self, event: Event) -> None:
"""
This function will handle an `Event`.
This event will be passed to every applications in apps list in order.
Args:
event (Event): the event that happens on this QNode
"""
for app in self.apps:
skip = app.handle(self, event)
if skip:
break
[docs] def add_apps(self, app: Application):
"""
Insert an Application into the app list.
Args:
app (Application): the inserting application.
"""
self.apps.append(app)
[docs] def get_apps(self, app_type):
"""
Get an Application that is `app_type`
Args:
app_type: the class of app_type
"""
return [app for app in self.apps if isinstance(app, app_type)]
[docs] def add_memory(self, memory):
"""
Add a quantum memory in this QNode
Args:
memory (Memory): the quantum memory
"""
memory.node = self
self.memories.append(memory)
[docs] def get_memory(self, memory: Union[str, int]):
"""
Get the memory by index (in memories) or its name
Args:
memory (Union[str, int]): the index or name of the memory
"""
if isinstance(memory, str):
for m in self.memories:
if m.name == memory:
return m
return self.memories[memory]
[docs] def add_operator(self, operator):
"""
Add a quantum operator in this node
Args:
operator (QuantumOperator): the quantum operator
"""
operator.set_own(self)
self.operators.append(operator)
[docs] def add_cchannel(self, cchannel):
"""
Add a classic channel in this QNode
Args:
cchannel (ClassicChannel): the classic channel
"""
cchannel.node_list.append(self)
self.cchannels.append(cchannel)
[docs] def get_cchannel(self, dst: "QNode"):
"""
Get the classic channel that connects to the `dst`
Args:
dst (QNode): the destination
"""
for cchannel in self.cchannels:
if dst in cchannel.node_list and self in cchannel.node_list:
return cchannel
return None
[docs] def add_qchannel(self, qchannel):
"""
Add a quantum channel in this QNode
Args:
qchannel (QuantumChannel): the quantum channel
"""
qchannel.node_list.append(self)
self.qchannels.append(qchannel)
[docs] def get_qchannel(self, dst: "QNode"):
"""
Get the quantum channel that connects to the `dst`
Args:
dst (QNode): the destination
"""
for qchannel in self.qchannels:
if dst in qchannel.node_list and self in qchannel.node_list:
return qchannel
return None
[docs] def add_request(self, request):
"""
add a request to this node
Args:
request (Request): the inserting request
"""
self.requests.append(request)
[docs] def clear_request(self):
"""
clear all requests
"""
self.requests.clear()
[docs] def add_network(self, network):
"""
add a network object to this node
Args:
network (qns.network.network.Network): the network object
"""
self.network = network
def __repr__(self) -> str:
if self.name is not None:
return f"<node {self.name}>"
return super().__repr__()