-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathservice.py
624 lines (519 loc) · 23.7 KB
/
service.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
"""Framework to host measurement service."""
from __future__ import annotations
import json
import sys
import threading
import warnings
from enum import Enum, EnumMeta
from os import path
from pathlib import Path
from types import TracebackType
from typing import (
TYPE_CHECKING,
Any,
Callable,
Dict,
Iterable,
List,
Literal,
Optional,
Type,
TypeVar,
Union,
)
import grpc
from deprecation import deprecated
from google.protobuf.descriptor import EnumDescriptor
from ni_measurement_plugin_sdk_service import _datatypeinfo
from ni_measurement_plugin_sdk_service._annotations import (
ENUM_VALUES_KEY,
TYPE_SPECIALIZATION_KEY,
)
from ni_measurement_plugin_sdk_service._internal import grpc_servicer
from ni_measurement_plugin_sdk_service._internal.parameter import (
metadata as parameter_metadata,
)
from ni_measurement_plugin_sdk_service._internal.service_manager import GrpcService
from ni_measurement_plugin_sdk_service.discovery import DiscoveryClient, ServiceLocation
from ni_measurement_plugin_sdk_service.grpc.channelpool import ( # re-export
GrpcChannelPool as GrpcChannelPool,
)
from ni_measurement_plugin_sdk_service.measurement.info import (
DataType,
MeasurementInfo,
ServiceInfo,
TypeSpecialization,
)
from ni_measurement_plugin_sdk_service.session_management import (
MultiSessionReservation,
PinMapContext,
SessionManagementClient,
SingleSessionReservation,
)
if TYPE_CHECKING:
from google.protobuf.internal.enum_type_wrapper import _EnumTypeWrapper
if sys.version_info >= (3, 10):
from typing import TypeGuard
else:
from typing_extensions import TypeGuard
if sys.version_info >= (3, 11):
from typing import Self
else:
from typing_extensions import Self
SupportedEnumType = Union[Type[Enum], _EnumTypeWrapper]
class MeasurementContext:
"""Proxy for the Measurement Service's context-local state."""
@property
def grpc_context(self) -> grpc.ServicerContext:
"""Get the context for the RPC."""
return grpc_servicer.measurement_service_context.get().grpc_context
@property
def pin_map_context(self) -> PinMapContext:
"""Get the pin map context for the RPC."""
return grpc_servicer.measurement_service_context.get().pin_map_context
def add_cancel_callback(self, cancel_callback: Callable[[], None]) -> None:
"""Add a callback which is invoked when the RPC is canceled."""
grpc_servicer.measurement_service_context.get().add_cancel_callback(cancel_callback)
def cancel(self) -> None:
"""Cancel the RPC."""
grpc_servicer.measurement_service_context.get().cancel()
@property
def time_remaining(self) -> float:
"""Get the time remaining for the RPC."""
return grpc_servicer.measurement_service_context.get().time_remaining
def abort(self, code: grpc.StatusCode, details: str) -> None:
"""Aborts the RPC."""
grpc_servicer.measurement_service_context.get().abort(code, details)
@property
def _measurement_service(self) -> MeasurementService:
owner = grpc_servicer.measurement_service_context.get().owner
assert isinstance(owner, MeasurementService)
return owner
def reserve_session(
self,
pin_or_relay_names: Union[str, Iterable[str]],
timeout: Optional[float] = 0.0,
) -> SingleSessionReservation:
"""Reserve a single session.
Reserve the session matching the given pins, sites, and instrument type ID and return
the information needed to create or access the session.
Args:
pin_or_relay_names: One or multiple pins, pin groups, relays, or relay groups to
use for the measurement.
timeout: Timeout in seconds.
Allowed values: 0 (non-blocking, fails immediately if resources cannot be
reserved), -1 (infinite timeout), or any other positive numeric value (wait for
that number of seconds)
Returns:
A reservation object with which you can query information about the session and
unreserve it.
"""
if not pin_or_relay_names:
raise ValueError("You must specify at least one pin or relay name.")
return self._measurement_service.session_management_client.reserve_session(
context=self.pin_map_context, pin_or_relay_names=pin_or_relay_names, timeout=timeout
)
def reserve_sessions(
self,
pin_or_relay_names: Union[str, Iterable[str]],
timeout: Optional[float] = 0.0,
) -> MultiSessionReservation:
"""Reserve multiple sessions.
Reserve sessions matching the given pins, sites, and instrument type ID and return the
information needed to create or access the sessions.
Args:
pin_or_relay_names: One or multiple pins, pin groups, relays, or relay groups to use
for the measurement.
timeout: Timeout in seconds.
Allowed values: 0 (non-blocking, fails immediately if resources cannot be
reserved), -1 (infinite timeout), or any other positive numeric value (wait for
that number of seconds)
Returns:
A reservation object with which you can query information about the sessions and
unreserve them.
"""
if not pin_or_relay_names:
raise ValueError("You must specify at least one pin or relay name.")
return self._measurement_service.session_management_client.reserve_sessions(
context=self.pin_map_context, pin_or_relay_names=pin_or_relay_names, timeout=timeout
)
_F = TypeVar("_F", bound=Callable)
class MeasurementService:
"""Class that supports registering and hosting a python function as a gRPC service."""
def __init__(
self,
service_config_path: Path,
version: str = "",
ui_file_paths: List[Path] = [],
service_class: Optional[str] = None,
) -> None:
"""Initialize the Measurement Service object.
Uses the specified .serviceconfig file, version, and UI file paths
to initialize a Measurement Service object.
Args:
service_config_path (Path): Path to the .serviceconfig file.
version (str): Version of the measurement service. Do not use this
parameter. Instead, specify the "version" field in the
.serviceconfig file. Default value is "".
ui_file_paths (List[Path]): List of paths to supported UIs.
Default value is [].
service_class (str): The service class from the .serviceconfig to use.
Default value is None, which will use the first service in the
.serviceconfig file.
"""
if not path.exists(service_config_path):
raise RuntimeError(f"File does not exist. {service_config_path}")
with open(service_config_path) as service_config_file:
service_config = json.load(service_config_file)
if service_class is None:
service = next(iter(service_config["services"]), None)
else:
service = next(
(s for s in service_config["services"] if s["serviceClass"] == service_class), None
)
if not service:
raise RuntimeError(
f"Service class '{service_class}' not found in '{service_config_file}'"
)
config_version = service.get("version")
if config_version is not None:
if version and version != config_version:
raise RuntimeError(
f"Version mismatch: .serviceconfig version is '{config_version}', but version parameter is '{version}'.",
)
version = config_version
# Note: sphinx-autoapi uses the public attributes' type hints and docstrings.
self.measurement_info: MeasurementInfo = MeasurementInfo(
display_name=service["displayName"],
version=version,
ui_file_paths=ui_file_paths,
)
"""Information about the measurement performed by this service."""
def convert_value_to_str(value: object) -> str:
if isinstance(value, str):
return value
return json.dumps(value, separators=(",", ":"))
self.service_info: ServiceInfo = ServiceInfo(
display_name=service["displayName"],
service_class=service["serviceClass"],
description_url=service["descriptionUrl"],
provided_interfaces=service["providedInterfaces"],
versions=[version],
annotations={
key: convert_value_to_str(value)
for key, value in service.get("annotations", {}).items()
},
)
"""Information about this service."""
self.context: MeasurementContext = MeasurementContext()
"""Accessor for context-local state."""
self._configuration_parameter_list: List[parameter_metadata.ParameterMetadata] = []
self._output_parameter_list: List[parameter_metadata.ParameterMetadata] = []
self._measure_function: Callable = self._raise_measurement_method_not_registered
self._initialization_lock = threading.RLock()
self._channel_pool: Optional[GrpcChannelPool] = None
self._discovery_client: Optional[DiscoveryClient] = None
self._grpc_service: Optional[GrpcService] = None
self._session_management_client: Optional[SessionManagementClient] = None
def _raise_measurement_method_not_registered(self) -> Any:
raise RuntimeError(
"Measurement method not registered. Use the register_measurement decorator to register it."
)
@property
def channel_pool(self) -> GrpcChannelPool:
"""Pool of gRPC channels used by the service."""
if self._channel_pool is None:
with self._initialization_lock:
if self._channel_pool is None:
self._channel_pool = GrpcChannelPool()
return self._channel_pool
@property
def discovery_client(self) -> DiscoveryClient:
"""Client for accessing the NI Discovery Service."""
if self._discovery_client is None:
with self._initialization_lock:
if self._discovery_client is None:
self._discovery_client = DiscoveryClient(grpc_channel_pool=self.channel_pool)
return self._discovery_client
@property
@deprecated(
deprecated_in="1.3.0-dev0",
details="This property should not be public and will be removed in a later release.",
)
def configuration_parameter_list(self) -> List[Any]:
"""List of configuration parameters."""
return self._configuration_parameter_list
@property
@deprecated(
deprecated_in="1.3.0-dev0",
details="This property should not be public and will be removed in a later release.",
)
def grpc_service(self) -> Optional[GrpcService]:
"""The gRPC service object. This is a private implementation detail."""
return self._grpc_service
@property
@deprecated(
deprecated_in="1.3.0-dev0",
details="This property should not be public and will be removed in a later release.",
)
def measure_function(self) -> Callable:
"""Registered measurement function."""
return self._measure_function
@property
@deprecated(
deprecated_in="1.3.0-dev0",
details="This property should not be public and will be removed in a later release.",
)
def output_parameter_list(self) -> List[Any]:
"""List of output parameters."""
return self._output_parameter_list
@property
def service_location(self) -> ServiceLocation:
"""The location of the service on the network."""
with self._initialization_lock:
if self._grpc_service is None:
raise RuntimeError("Measurement service not running")
return self._grpc_service.service_location
@property
def session_management_client(self) -> SessionManagementClient:
"""Client for accessing the measurement plug-in session management service."""
if self._session_management_client is None:
with self._initialization_lock:
if self._session_management_client is None:
self._session_management_client = SessionManagementClient(
discovery_client=self.discovery_client,
grpc_channel_pool=self.channel_pool,
)
return self._session_management_client
def register_measurement(self, measurement_function: _F) -> _F:
"""Register a function as the measurement function for a measurement service.
To declare a measurement function, use this idiom::
@measurement_service.register_measurement
@measurement_service.configuration("Configuration 1", ...)
@measurement_service.configuration("Configuration 2", ...)
@measurement_service.output("Output 1", ...)
@measurement_service.output("Output 2", ...)
def measure(configuration1, configuration2):
...
return (output1, output2)
See also: :func:`.configuration`, :func:`.output`
"""
self._measure_function = measurement_function
return measurement_function
def configuration(
self,
display_name: str,
type: DataType,
default_value: Any,
*,
instrument_type: str = "",
enum_type: Optional[SupportedEnumType] = None,
) -> Callable[[_F], _F]:
"""Add a configuration parameter to a measurement function.
This decorator maps the measurement service's configuration parameters
to Python positional parameters. To add multiple configuration parameters
to the same measurement function, use this decorator multiple times.
The order of decorator calls must match the order of positional parameters.
See also: :func:`.register_measurement`
Args:
display_name (str): Display name of the configuration.
type (DataType): Data type of the configuration.
default_value (Any): Default value of the configuration.
instrument_type (Optional[str]):
Filter pins by instrument type. This is only supported when configuration type
is DataType.IOResource or DataType.Pin (deprecated).
For NI instruments, use instrument type id constants defined by
:py:mod:`ni_measurement_plugin_sdk_service.session_management`, such as
:py:const:`~ni_measurement_plugin_sdk_service.session_management.INSTRUMENT_TYPE_NI_DCPOWER`
or
:py:const:`~ni_measurement_plugin_sdk_service.session_management.INSTRUMENT_TYPE_NI_DMM`.
For custom instruments, use the instrument type id defined in the pin map file.
enum_type (Optional[SupportedEnumType]):
Defines the enum type associated with this configuration parameter. This is only
supported when configuration type is DataType.Enum or DataType.EnumArray1D.
Returns:
Callable: Callable that takes in Any Python Function
and returns the same python function.
"""
if type == DataType.Pin:
warnings.warn(
"DataType.Pin is deprecated. Use DataType.IOResource instead.", DeprecationWarning
)
if type == DataType.PinArray1D:
warnings.warn(
"DataType.PinArray1D is deprecated. Use DataType.IOResourceArray1D instead.",
DeprecationWarning,
)
data_type_info = _datatypeinfo.get_type_info(type)
annotations = self._make_annotations_dict(
data_type_info.type_specialization, instrument_type=instrument_type, enum_type=enum_type
)
parameter = parameter_metadata.ParameterMetadata.initialize(
display_name,
data_type_info.grpc_field_type,
data_type_info.repeated,
default_value,
annotations,
data_type_info.message_type,
enum_type,
)
self._configuration_parameter_list.append(parameter)
def _configuration(func: _F) -> _F:
return func
return _configuration
def output(
self,
display_name: str,
type: DataType,
*,
enum_type: Optional[SupportedEnumType] = None,
) -> Callable[[_F], _F]:
"""Add an output parameter to a measurement function.
This decorator maps the measurement service's output parameters to
the elements of the tuple returned by the measurement function.
To add multiple output parameters to the same measurement function,
use this decorator multiple times.
The order of decorator calls must match the order of elements
returned by the measurement function.
See also: :func:`.register_measurement`
Args:
display_name (str): Display name of the output.
type (DataType): Data type of the output.
enum_type (Optional[SupportedEnumType]:
Defines the enum type associated with this configuration parameter. This is only
supported when configuration type is DataType.Enum or DataType.EnumArray1D.
Returns:
Callable: Callable that takes in Any Python Function and
returns the same python function.
"""
if type == DataType.Pin:
warnings.warn(
"DataType.Pin is deprecated. Use DataType.IOResource instead.", DeprecationWarning
)
if type == DataType.PinArray1D:
warnings.warn(
"DataType.PinArray1D is deprecated. Use DataType.IOResourceArray1D instead.",
DeprecationWarning,
)
data_type_info = _datatypeinfo.get_type_info(type)
annotations = self._make_annotations_dict(
data_type_info.type_specialization, enum_type=enum_type
)
parameter = parameter_metadata.ParameterMetadata.initialize(
display_name,
data_type_info.grpc_field_type,
data_type_info.repeated,
None,
annotations,
data_type_info.message_type,
enum_type,
)
self._output_parameter_list.append(parameter)
def _output(func: _F) -> _F:
return func
return _output
def host_service(self) -> MeasurementService:
"""Host the registered measurement method as a gRPC measurement service.
Returns:
MeasurementService: Context manager that can be used with a with-statement to close
the service.
Raises:
Exception: If register measurement methods not available.
"""
with self._initialization_lock:
if self._measure_function is self._raise_measurement_method_not_registered:
self._raise_measurement_method_not_registered()
if self._grpc_service is not None:
raise RuntimeError("Measurement service already running.")
self._grpc_service = GrpcService(self.discovery_client)
self._grpc_service.start(
self.measurement_info,
self.service_info,
self._configuration_parameter_list,
self._output_parameter_list,
self._measure_function,
owner=self,
)
return self
def _make_annotations_dict(
self,
type_specialization: TypeSpecialization,
*,
instrument_type: str = "",
enum_type: Optional[SupportedEnumType] = None,
) -> Dict[str, str]:
annotations: Dict[str, str] = {}
if type_specialization == TypeSpecialization.NoType:
return annotations
annotations[TYPE_SPECIALIZATION_KEY] = type_specialization.value
if type_specialization == TypeSpecialization.Pin:
if instrument_type != "" or instrument_type is not None:
annotations["ni/pin.instrument_type"] = instrument_type
if type_specialization == TypeSpecialization.IOResource:
if instrument_type != "" or instrument_type is not None:
annotations["ni/ioresource.instrument_type"] = instrument_type
if type_specialization == TypeSpecialization.Enum:
if enum_type is not None:
annotations[ENUM_VALUES_KEY] = self._enum_to_annotations_value(enum_type)
else:
raise ValueError("enum_type is required for enum parameters.")
return annotations
def _enum_to_annotations_value(self, enum_type: SupportedEnumType) -> str:
enum_values = {}
# For protobuf enums, enum_type is an instance of EnumTypeWrapper at run time, so passing
# it to issubclass() would raise an error.
if self._is_protobuf_enum(enum_type):
if 0 not in enum_type.values():
raise ValueError("The enum does not have a value for 0.")
for name, value in enum_type.items():
enum_values[name] = value
elif isinstance(enum_type, EnumMeta):
if not any(member.value == 0 for member in enum_type):
raise ValueError("The enum does not have a value for 0.")
for member in enum_type:
enum_values[member.name] = member.value
return json.dumps(enum_values)
def _is_protobuf_enum(self, enum_type: SupportedEnumType) -> TypeGuard[_EnumTypeWrapper]:
# Use EnumDescriptor to check for protobuf enums at run time without using
# google.protobuf.internal.
return isinstance(getattr(enum_type, "DESCRIPTOR", None), EnumDescriptor)
def close_service(self) -> None:
"""Stop the gRPC measurement service.
This method stops the gRPC server, unregisters with the discovery service, and cleans up
the cached discovery client and gRPC channel pool.
After calling close_service(), you may call host_service() again.
Exiting the measurement service's runtime context automatically calls close_service().
"""
with self._initialization_lock:
if self._grpc_service is not None:
self._grpc_service.stop()
if self._channel_pool is not None:
self._channel_pool.close()
self._grpc_service = None
self._channel_pool = None
self._discovery_client = None
def __enter__(self: Self) -> Self:
"""Enter the runtime context related to the measurement service."""
return self
def __exit__(
self,
exc_type: Optional[Type[BaseException]],
exc_val: Optional[BaseException],
traceback: Optional[TracebackType],
) -> Literal[False]:
"""Exit the runtime context related to the measurement service."""
self.close_service()
return False
def get_channel(self, provided_interface: str, service_class: str = "") -> grpc.Channel:
"""Return gRPC channel to specified service.
Args:
provided_interface (str): The gRPC Full Name of the service.
service_class (str): The service "class" that should be matched.
Returns:
grpc.Channel: A channel to the gRPC service.
Raises:
Exception: If service_class is not specified and there is more than one matching service
registered.
"""
service_location = self.discovery_client.resolve_service(provided_interface, service_class)
return self.channel_pool.get_channel(service_location.insecure_address)