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 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976
|
NewPB
=====
This document describes the new PB protocol. This is a layer on top of Banana
which provides remote object access (method invocation and instance
transfer).
Fundamentally, PB is about one side keeping a ``RemoteReference`` to the
other side's "Referenceable" . The Referenceable has some methods that can be
invoked remotely: functionality it is offering to remote callers. Those
callers hold RemoteReferences which point to it. The RemoteReference object
offers a way to invoke those methods (generally through the ``callRemote``
method).
There are plenty of other details, starting with how the RemoteReference is
obtained, and how arguments and return values are communicated.
For the purposes of this document, we will designate the side that holds the
actual ``Referenceable`` object as "local" , and the side that holds the
proxy ``RemoteReference`` object as "remote" . This distinction is only
meaningful with respect to a single RemoteReference/Referenceable pair. One
program may hold Referenceable "A" and RemoteReference "B" , paired with
another that holds RemoteReference "A" and Referenceable "B" . Once
initialization is complete, PB is a symmetric protocol.
It is helpful to think of PB as providing a wire or pipe that connects two
programs. Objects are put into this pipe at one end, and something related to
the object comes out the other end. These two objects are said to correspond
to each other. Basic types (like lists and dictionaries) are handled by
Banana, but more complex types (like instances) are treated specially, so
that most of the time there is a "native" form (as present on the local side)
that goes into the pipe, and a remote form that comes out.
Initialization
--------------
The PB session begins with some feature negotiation and (generally) the
receipt of a VocabularyDict. Usually this takes place over an interactive
transport, like a TCP connection, but newpb can also be used in a more
batched message-oriented mode, as long as both the creator of the method call
request and its eventual consumer are in agreement about their shared state
(at least, this is the intention.. there are still pieces that need to be
implemented to make this possible).
The local side keeps a table which provides a bidirectional mapping between
``Referenceable`` objects and a connection-local "object-ID" number. This
table begins with a single object called the "Root" , which is implicitly
given ID number 0. Everything else is bootstrapped through this object. For
the typical PB Broker, this root object performs cred authentication and
returns other Referenceables as the cred Avatar.
The remote side has a collection of ``RemoteReference`` objects, each of
which knows the object-ID of the corresponding Referenceable, as well as the
Broker which provides the connection to the other Broker. The remote side
must do reference-tracking of these RemoteReferences, because as long as it
remains alive, the local-side Broker must maintain a reference to the
original Referenceable.
Method Calls
------------
The remote side invokes a remote method by calling ``ref.callRemote()`` on
its RemoteReference. This starts by validating the method name and arguments
against a "Schema" (described below). It then creates a new Request object
which will live until the method call has either completed successfully or
failed due to an exception (including the connection being lost).
``callRemote`` returns a Deferred, which does not fire until the request is
finished.
It then sends a ``call`` banana sequence over the wire. This sequence
indicates the request ID (used to match the request with the resulting
``answer`` or ``error`` response), the object ID of the Referenceable being
targeted, a string to indicate the name of the method being invoked, and the
arguments to be passed into the method.
All arguments are passed by name (i.e. keyword arguments instead of
positional parameters). Each argument is subject to the "argument
transformation" described below.
The local side receives the ``call`` sequence, uses the object-ID to look up
the Referenceable, finds the desired method, then applies the method's schema
to the incoming arguments. If they are acceptable, it invokes the method. A
normal return value it sent back immediately in an ``answer`` sequence
(subject to the same transformation as the inbound arguments). If the method
returns a Deferred, the answer will be sent back when the Deferred fires. If
the method raises an exception (or the Deferred does an errback), the
resulting Failure is sent back in a ``error`` sequence. Both the ``answer``
and the ``error`` start with the request-ID so they can be used to complete
the Request object waiting on the remote side.
The original Deferred (the one produced by ``callRemote`` ) is finally
callbacked with the results of the method (or errbacked with a Failure or
RemoteFailure object).
Example
~~~~~~~
This code runs on the "local" side: the one with the ``pb.Referenceable``
which will respond to a remote invocation.
.. code-block:: python
class Responder(pb.Referenceable):
def remote_add(self, a, b):
return a+b
and the following code runs on the "remote" side (the one which holds a
``pb.RemoteReference`` ):
.. code-block:: python
def gotAnswer(results):
print results
d = rr.callRemote("add", a=1, b=2)
d.addCallback(gotAnswer)
Note that the arguments are passed as named parameters: oldpb used both
positional parameters and named (keyword) arguments, but newpb prefers just
the keyword arguments. TODO: newpb will probably convert positional
parameters to keyword arguments (based upon the schema) before sending them
to the remote side.
Using RemoteInterfaces
~~~~~~~~~~~~~~~~~~~~~~
To nail down the types being sent across the wire, you can use a
``RemoteInterface`` to define the methods that are implemented by any
particular ``pb.Referenceable`` :
.. code-block:: python
class RIAdding(pb.RemoteInterface):
def add(a=int, b=int): return int
class Responder(pb.Referenceable):
implements(RIAdding)
def remote_add(self, a, b):
return a+b
# and on the remote side:
d = rr.callRemote(RIAdding['add'], a=1, b=2)
d.addCallback(gotAnswer)
In this example, the "RIAdding" remote interface defines a single method
"add" , which accepts two integer parameters and returns an integer. This
method (technically a classmethod) is used instead of the string form of the
method name. What does this get us?
- The calling side will pre-check its arguments against the constraints that
it believes to be imposed by the remote side. It will raise a Violation
rather than send parameters that it thinks will be rejected.
- The receiving side will enforce the constraints, causing the method
call to errback (with a Violation) if they are not met. This means the code
in ``remote_add`` does not need to worry about what strange types
it might be given, such as two strings, or two lists.
- The receiving side will pre-check its return argument before sending it
back. If the method returns a string, it will cause a Violation exception
to be raised. The caller will get this Violation as an errback instead of
whatever (illegal) value the remote method computed.
- The sending side will enforce the return-value constraint (raising a
Violation if it is not met). This means the calling side (in this case the
``gotAnswer`` callback function) does not need to worry about what strange
type the remote method returns.
You can use either technique: with RemoteInterfaces or without. To get the
type-checking benefits, you must use them. If you do not, PB cannot protect
you against memory consumption attacks.
RemoteInterfaces
~~~~~~~~~~~~~~~~
RemoteInterfaces are passed by name. Each side of a PB connection has a table
which maps names to RemoteInterfaces (subclasses of ``pb.RemoteInterface`` ).
Metaclass magic is used to add an entry to this table each time you define a
RemoteInterface subclass, using the ``__remote_name__`` attribute (or
reflect.qual() if that is not set).
Each ``Referenceable`` that goes over the wire is accompanied by the list of
RemoteInterfaces which it claims to implement. On the receiving side, these
RemoteInterface names are looked up in the table and mapped to actual (local)
RemoteInterface classes.
TODO: it might be interesting to serialize the RemoteInterface class and ship
it over the wire, rather than assuming both sides have a copy (and that they
agree). However, if one side does not have a copy, it is unlikely that it
will be able to do anything very meaningful with the remote end.
The syntax of RemoteInterface is still in flux. The basic idea is that each
method of the RemoteInterface defines a remotely invokable method, something
that will exist with a "remote_" prefix on any ``pb.Referenceable`` s which
claim to implement it.
Those methods are defined with a number of named parameters. The default
value of each parameter is something which can be turned into a
``Constraint`` according to the rules of schema.makeConstraint . This means
you can use things like ``(int, str, str)`` to mean a tuple of exactly those
three types.
Note that the methods of the RemoteInterface do *not* list "self" as a
parameter. As the zope.interface documentation points out, "self" is an
implementation detail, and does not belong in the interface specification.
Another way to think about it is that, when you write the code which calls a
method in this interface, you don't include "self" in the arguments you
provide, therefore it should not appear in the public documentation of those
methods.
The method is required to return a value which can be handled by
schema.makeConstraint: this constraint is then applied to the return value of
the remote method.
Other attributes of the method (perhaps added by decorators of some sort)
will, some day, be able to specify specialized behavior of the method. The
brainstorming sessions have come up with the following ideas:
- .wait=False: don't wait for an answer
- .reliable=False: feel free to send this over UDP
- .ordered=True: but enforce order between successive remote calls
- .priority=3: use priority queue / stream #3
- .failure=Full: allow/expect full Failure contents (stack frames)
- .failure=ErrorMessage: only allow/expect truncated CopiedFailures
We are also considering how to merge the RemoteInterface with other useful
interface specifications, in particular zope.interface and
formless.TypedInterface .
Argument Transformation
-----------------------
To understand this section, it may be useful to review the Banana
documentation on serializing object graphs. Also note that method arguments
and method return values are handled identically.
Basic types (lists, tuples, dictionaries) are serialized and unserialized as
you would expect: the resulting object would (if it existed in the sender's
address space) compare as equal (but of course not "identical" , because the
objects will exist at different memory locations).
Shared References, Serialization Scope
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Shared references to the same object are handled correctly. Banana is
responsible for noticing that a sharable object has been serialized before
(or at least has begun serialization) and inserts reference markers so that
the object graph can be reconstructed. This introduces the concept of
serialization scope: the boundaries beyond which shared references are not
maintained.
For PB, serialization is scoped to the method call. If an object is
referenced by two arguments to the same method call, that method will see two
references to the same object. If those arguments are containers of some
form, which (eventually) hold a reference to the same object, the object
graph will be preserved. For example:
.. code-block:: python
class Caller:
def start(self):
obj = [1, 2, 3]
self.remote.callRemote("both", obj, obj)
self.remote.callRemote("deeper", ["a", obj], (4, 5, obj))
class Called(pb.Referenceable):
def remote_both(self, arg1, arg2):
assert arg1 is arg2
assert arg1 == [1,2,3]
def remote_deeper(self, listarg, tuplearg):
ref1 = listarg[1]
ref2 = tuplearg[2]
assert ref1 is ref2
assert ref1 == [1,2,3]
But if the remote-side object is referenced in two distinct remote method
invocations, the local-side methods will see two separate objects. For
example:
.. code-block:: python
class Caller:
def start(self):
self.obj = [1, 2, 3]
d = self.remote.callRemote("one", self.obj)
d.addCallback(self.next)
def next(self, res):
self.remote.callRemote("two", self.obj)
class Called(pb.Referenceable):
def remote_one(self, ref1):
assert ref1 == [1,2,3]
self.ref1 = ref1
def remote_two(self, ref2):
assert ref2 == [1,2,3]
assert ref1 is not ref2 # not the same object
You can think of the method call itself being a node in the object graph,
with the method arguments as its children. The method call node is picked up
and the resulting sub-tree is serialized with no knowledge of anything
outside the sub-tree [#]_ .
The value returned by a method call is serialized by itself, without
reference to the arguments that were given to the method. If a remote method
is called with a list, and the method returns its argument unchanged, the
caller will get back a deep copy of the list it passed in.
Referenceables, RemoteReferences
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Referenceables are transformed into RemoteReferences when they are sent over
the wire. As one side traverses the object graph of the method arguments (or
the return value), each ``Referenceable`` object it encounters it serialized
with a ``my-reference`` sequence, that includes the object-ID number. When
the other side is unserializing the token stream, it creates a
``RemoteReference`` object, or uses one that already exists.
Likewise, if an argument (or return value) contains a ``RemoteReference`` ,
and it is being sent back to the Broker that holds the original
``Referenceable`` then it will be turned back into that Referenceable when it
arrives. In this case, the caller of a remote method which returns its
argument unchanged *will* see a a result that is identical to what it passed
in:
.. code-block:: python
class Target(pb.Referenceable):
pass
class Caller:
def start(self):
self.obj = Target()
d = self.remote.callRemote("echo", self.obj)
d.addCallback(self.next)
def next(self, res):
assert res is self.obj
class Called(pb.Referenceable):
def remote_echo(self, arg):
# arg is a RemoteReference to a Target() instance
return arg
These references have a serialization scope which extends across the entire
connection. As long as two method calls share the same ``Broker`` instance
(which generally means they share the same TCP socket), they will both
serialize ``Referenceable`` s into identical ``RemoteReference`` s. This also
means that both sides do reference-counting to insure that the Referenceable
doesn't get garbage-collected while a remote system holds a RemoteReference
that points to it.
In the future, there may be other classes which behave this way. In
particular, "Referenceable" and "Callable" may be distinct
qualities.
Copyable, RemoteCopy
~~~~~~~~~~~~~~~~~~~~
Some objects can be marked to indicate that they should be copied bodily each
time they traverse the wire (pass-by-value instead of pass-by-reference).
Classes which inherit from ``pb.Copyable`` are passed by value. Their
``getTypeToCopy`` and ``getStateToCopy`` methods are used to assemble the
data that will be serialized. These methods default to plain old
``reflect.qual`` (which provides the fully-qualified name of the class) and
the instance's attribute ``__dict__`` . You can override these to provide a
different (or smaller) set of state attributes to the remote end.
.. code-block:: python
class Source(pb.Copyable):
def getStateToCopy(self):
state = self.__dict__.copy()
del state['private']
state['children'] = []
return state
Rather than subclass ``pb.Copyable`` , you can also implement the
``flavors.ICopyable`` interface:
.. code-block:: python
from twisted.python import reflect
class Source2:
implements(flavors.ICopyable)
def getTypeToCopy(self):
return reflect.qual(self.__class__)
def getStateToCopy(self):
return self.__dict__
or register an ICopyable adapter. Using the adapter allows you to define
serialization behavior for third-party classes that are out of your control
(ones which you cannot rewrite to inherit from ``pb.Copyable`` ).
.. code-block:: python
class Source3:
pass
class Source3Copier:
implements(flavors.ICopyable)
def getTypeToCopy(self):
return 'foo.Source3'
def getStateToCopy(self):
orig = self.original
d = { 'foo': orig.foo, 'bar': orig.bar }
return d
registerAdapter(Source3Copier, Source3, flavors.ICopyable)
On the other end of the wire, the receiving side must register a
``RemoteCopy`` subclass under the same name as returned by the sender's
``getTypeToCopy`` value. This subclass is used as a factory to create
instances that correspond to the original ``Copyable`` . The registration can
either take place explicitly (with ``pb.registerRemoteCopy`` ), or
automatically (by setting the ``copytype`` attribute in the class
definition).
The default ``RemoteCopy`` behavior simply sets the instance's ``__dict__``
to the incoming state, which may be plenty if you are willing to let
outsiders arbitrarily manipulate your object state. If so, and you believe
both peers are importing the same source file, it is enough to create and
register the ``RemoteCopy`` at the same time you create the ``Copyable`` :
.. code-block:: python
class Source(pb.Copyable):
def getStateToCopy(self):
state = self.__dict__.copy()
del state['private']
state['children'] = []
return state
class Remote(pb.RemoteCopy):
copytype = reflect.qual(Source)
You can do something special with the incoming object state by overriding the
``setCopyableState`` method. This may allow you to do some sanity-checking on
the state before trusting it.
.. code-block:: python
class Remote(pb.RemoteCopy):
def setCopyableState(self, state):
state['count'] = 0
self.__dict__ = state
self.total = self.one + self.two
# show explicit registration, instead of using 'copytype' class attribute
pb.registerRemoteCopy(reflect.qual(Source), Remote)
You can also set a Constraint on the inbound object state, which provides a
way to enforce some type checking on the state components as they arrive.
This protects against resource-consumption attacks where someone sends you a
zillion-byte string as part of the object's state.
.. code-block:: python
class Remote(pb.RemoteCopy):
stateSchema = schema.AttributeDictConstraint(('foo', int),
('bar', str))
In this example, the object will only accept two attributes: "foo" (which
must be a number), and "bar" (which must be a string shorter than the default
limit of 1000 characters). Various classes from the``schema`` module can be
used to construct more complicated constraints.
Slicers, ISlicer
~~~~~~~~~~~~~~~~
Each object gets "Sliced" into a stream of tokens as they go over the wire:
Referenceable and Copyable are merely special cases. These classes have
Slicers which implement specific behaviors when the serialization process is
asked to send their instances to the remote side. You can implement your own
Slicers to take complete control over the serialization process. The most
useful reason to take advantage of this feature is to implement "streaming
slicers" , which can minimize in-memory buffering by only producing Banana
tokens on demand as space opens up in the transport.
Banana Slicers are documented in detail in the Banana documentation. Once you
create a Slicer class, you will want to "register" it, letting Banana know
that this Slicer is useful for conveying certain types of objects across the
wire. The registry maps a type to a Slicer class (which is really a slicer
factory), and is implemented by registering the slicer as a regular "adapter"
for the ``ISlicer`` interface. For example, lists are serialized by the
``ListSlicer`` class, so ``ListSlicer`` is registered as the slicer for the
``list`` type:
.. code-block:: python
class ListSlicer(BaseSlicer):
opentype = ("list",)
slices = list
Slicer registration can be either explicit or implicit. In this example, an
implicit registration is used: by setting the "slices" attribute to the
``list`` type, the BaseSlicer's metaclass automatically registers the mapping
from ``list`` to ListSlicer.
To explicitly register a slicer, just leave ``opentype`` set to None (to
disable auto-registration), and then register the slicer manually.
.. code-block:: python
class TupleSlicer(BaseSlicer):
opentype = ("tuple",)
slices = None
...
registerAdapter(TupleSlicer, tuple, pb.ISlicer)
As with ICopyable, registering an ISlicer adapter allows you to define
exactly how you wish to serialize third-party classes which you do not get to
modify.
Unslicers
~~~~~~~~~
On the other side of the wire, the incoming token stream is handed to an
``Unslicer`` , which is responsible for turning the set of tokens into a
single finished object. They are also responsible for enforcing limits on the
types and sizes of the tokens that make up the stream. Unslicers are also
described in greater detail in the Banana docs.
As with Slicers, Unslicers need to be registered to be useful. This registry
maps "opentypes" to Unslicer classes (i.e. factories which can produce an
unslicer instance each time the given opentype appears in the token stream).
Therefore it maps tuples to subclasses of ``BaseUnslicer`` .
Again, this registry can be either implicit or explicit. If the Unslicer has
a non-None class attribute named ``opentype`` , then it is automatically
registered. If it does not have this attribute (or if it is set to None),
then no registration is performed, and the Unslicer must be manually
registered:
.. code-block:: python
class MyUnslicer(BaseUnslicer):
...
pb.registerUnslicer(('myopentype',), MyUnslicer)
Also remember that this registry is global, and that you cannot register two
Unslicers for the same opentype (you'll get an exception at class-definition
time, which will probably result in an ImportError).
Slicer/Unslicer Example
~~~~~~~~~~~~~~~~~~~~~~~
The simplest kind of slicer has a ``sliceBody`` method (a generator) which
yields a series of tokens. To demonstrate how to build a useful Slicer, we'll
write one that can send large strings across the wire in pieces. Banana can
send arbitrarily long strings in a single token, but each token must be
handed to the transport layer in an indivisble chunk, and anything that
doesn't fit in the transmit buffers will be stored in RAM until some space
frees up in the socket. Practically speaking, this means that anything larger
than maybe 50kb will spend a lot of time in memory, increasing the RAM
footprint for no good reason.
Because of this, it is useful to be able to send large amounts of data in
smaller pieces, and let the remote end reassemble them. The following Slicer
is registered to handle all open files (perhaps not the best idea), and
simply emits the contents in 10kb chunks.
(readers familiar with oldpb will notice that this Slicer/Unslicer pair
provide similar functionality to the old FilePager class. The biggest
improvement is that newpb can accomplish this without the extra round-trip
per chunk. The downside is that, unless you enable streaming in your Broker,
no other methods can be invoked while the file is being transmitted. The
upside of the downside is that this lets you retain in-order execution of
remote methods, and that you don't have to worry changes to the contents of
the file causing corrupt data to be sent over the wire. The oter upside of
the downside is that, if you enable streaming, you can do whatever other
processing you wish between data chunks.)
.. code-block:: python
class BigFileSlicer(BaseSlicer):
opentype = ("bigfile",)
slices = types.FileType
CHUNKSIZE = 10000
def sliceBody(self, streamable, banana):
while 1:
chunk = self.obj.read(self.CHUNKSIZE)
if not chunk:
return
yield chunk
To receive this, you would use the following minimal Unslicer at the other
end. Note that this Unslicer does not do as much as it could in the way of
constraint enforcement: an attacker could easily make you consume as much
memory as they wished by simply sending you a never-ending series of chunks.
.. code-block:: python
class BigFileUnslicer(LeafUnslicer):
opentype = ("bigfile",)
def __init__(self):
self.chunks = []
def checkToken(self, typebyte, size):
if typebyte != tokens.STRING:
raise BananaError("BigFileUnslicer only accepts strings")
def receiveChild(self, obj):
self.chunks.append(obj)
def receiveClose(self):
return "".join(self.chunks)
The ``opentype`` attribute causes this Unslicer to be implicitly registered
to handle any incoming sequences with an "index tuple" of ``("bigfile",)`` ,
so each time BigFileSlicer is used, a BigFileUnslicer will be created to
handle the results.
A more complete example would want to write the file chunks to disk at they
arrived, or process them incrementally. It might also want to have some way
to limit the overall size of the file, perhaps by having the first chunk be
an integer with the promised file size. In this case, the example might look
like this somewhat contrived (and somewhat insecure) Unslicer:
.. code-block:: python
class SomewhatLargeFileUnslicer(LeafUnslicer):
opentype = ("bigfile",)
def __init__(self):
self.fileSize = None
self.size = 0
self.output = open("/tmp/bigfile.txt", "w")
def checkToken(self, typebyte, size):
if self.fileSize is None:
if typebyte != tokens.INT:
raise BananaError("fileSize must be an INT")
else:
if typebyte != tokens.STRING:
raise BananaError("BigFileUnslicer only accepts strings")
if self.size + size > self.fileSize:
raise BananaError("size limit exceeded")
def receiveChild(self, obj):
if self.fileSize is None:
self.fileSize = obj
# decide if self.fileSize is too big, raise error to refuse it
else:
self.output.write(obj)
self.size += len(obj)
def receiveClose(self):
self.output.close()
return open("/tmp/bigfile.txt", "r")
This constrained BigFileUnslicer uses the fact that each STRING token comes
with a size, which can be used to enforce the promised filesize that was
provided in the first token. The data is streamed to a disk file as it
arrives, so no more than CHUNKSIZE of memory is required at any given time.
Streaming Slicers
~~~~~~~~~~~~~~~~~
TODO: add example
The following slicer will, when the broker allows streaming, will yield the
CPU to other reactor events that want processing time. (This technique
becomes somewhat inefficient if there is nothing else contending for CPU
time, and if this matters you might want to use something which sends N
chunks before yielding, or yields only when some other known service
announces that it wants CPU time, etc).
.. code-block:: python
class BigFileSlicer(BaseSlicer):
opentype = ("bigfile",)
slices = types.FileType
CHUNKSIZE = 10000
def sliceBody(self, streamable, banana):
while 1:
chunk = self.obj.read(self.CHUNKSIZE)
if not chunk:
return
yield chunk
if streamable:
d = defer.Deferred()
reactor.callLater(0, d.callback, None)
yield d
The next example will deliver data as it becomes available from a
hypothetical slow process.
.. code-block:: python
class OutputSlicer(BaseSlicer):
opentype = ("output",)
def sliceBody(self, streamable, banana):
assert streamable # requires it
while 1:
if self.process.finished():
return
chunk = self.process.read(self.CHUNKSIZE)
if not chunk:
d = self.process.waitUntilDataIsReady()
yield d
else:
yield chunk
Streamability is required in this example because otherwise the Slicer is
required to provide chunks non-stop until the object has been completely
serialized. If the process cannot deliver data, it's not like the Slicer can
block waiting until it becomes ready. Prohibiting streamability is done to
ensure coherency of serialized state, and the only way to guarantee this is
to not let any non-Banana methods get CPU time until the object has been
fully processed.
Streaming Unslicers
~~~~~~~~~~~~~~~~~~~
On the receiving side, the Unslicer can be made streamable too. This is
considerably easier than on the sending side, because there are fewer
concerns about state coherency.
A streaming Unslicer is merely one that delivers some data directly from the
``receiveChild`` method, rather than accumulating it until the
``receiveClose`` method. The SomewhatLargeFileUnslicer example from above is
actually a streaming Unslicer. Nothing special needs to be done.
On the other hand, it can be tricky to know where exactly to deliver the data
being streamed. The streamed object is probably part of a larger structure
(like a method call), where the higher-level attribute can be used to
determine which object or method should be called with the incoming data as
it arrives. The current Banana model is that each completed object (as
returned by the child's ``receiveClose`` method) is handed to the parent's
``receiveChild`` method. The parent can do whatever it wants with the
results. To make streaming Unslicers more useful, the parent should be able
to set up a target for the data at the time the child Unslicer is created.
More work is needed in this area to figure out how this functionality should
be exposed.
Arbitrary Instances are NOT serialized
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Arbitrary instances (that is, anything which does not have an ``ISlicer``
adapter) are *not* serialized. If an argument to a remote method contains
one, you will get a Violation exception when you attempt to serialize it
(i.e., the Deferred that you get from ``callRemote`` will errback with a
Failure that contains a Violation exception). If the return value contains
one, the Violation will be logged on the local side, and the remote caller
will see an error just as if your method had raised a Violation itself.
There are two reasons for this. The first is a security precaution: you must
explicitly mark the classes that are willing to reveal their contents to the
world. This reduces the chance of leaking sensitive information.
The second is because it is not actually meaningful to send the contents of
an arbitrary object. The recipient only gets the class name and a dictionary
with the object's state. Which class should it use to create the
corresponding object? It could attempt to import one based upon the classname
(the approach pickle uses), but that would give a remote attacker
unrestricted access to classes which could do absolutely anything: very
dangerous.
Both ends must be willing to transport the object. The sending side expresses
this by marking the class (subclassing Copyable, or registering an ISlicer
adapter). The receiving side must register the class as well, by doing
registerUnslicer or using the ``opentype`` attribute in a suitable Unslicer
subclass definition.
PB Sequences
------------
There are several Banana sequences which are used to support the RPC
mechanisms of Perspective Broker. These are in addition to the usual ones
listed in the Banana docs.
Top-Level Sequences
~~~~~~~~~~~~~~~~~~~
These sequences only appear at the top-level (never inside another
object).
+-------------------------------+-------------------------------------------------------+
| method call (``callRemote`` ) | OPEN(call) INT(request-id) INT/STR(your-reference-id) |
| | STRING(interfacename) STRING(methodname) |
| | (STRING(argname),argvalue).. |
| | CLOSE |
+-------------------------------+-------------------------------------------------------+
| method response (success) | OPEN(answer) INT(request-id) value CLOSE |
+-------------------------------+-------------------------------------------------------+
| method response (exception) | OPEN(error) INT(request-id) value CLOSE |
+-------------------------------+-------------------------------------------------------+
| RemoteReference.__del__ | OPEN(decref) INT(your-reference-id) CLOSE |
+-------------------------------+-------------------------------------------------------+
Internal Sequences
~~~~~~~~~~~~~~~~~~
The following sequences are used to serialize PB-specific objects. They never
appear at the top-level, but only as the argument value or return value (or
somewhere inside them).
+--------------------+--------------------------------------------+
| pb.Referenceable | OPEN(my-reference) INT(clid) |
| | [OPEN(list) InterfaceList.. CLOSE] |
| | CLOSE |
+--------------------+--------------------------------------------+
| pb.RemoteReference | OPEN(your-reference) INT/STR(clid) |
| | CLOSE |
+--------------------+--------------------------------------------+
| pb.Copyable | OPEN(copyable) STRING(reflect.qual(class)) |
| | (attr,value).. CLOSE |
+--------------------+--------------------------------------------+
The first time a ``pb.Referenceable`` is sent, the second object is an
InterfaceList, which is a list of interfacename strings, and therefore
constrainable by a schema of ListOf(str) with some appropriate maximum-length
restrictions. This InterfaceList describes all the Interfaces that the
corresponding ``pb.Referenceable`` implements. The receiver uses this list to
look up local Interfaces (and therefore Schemas) to attach to the
``pb.RemoteReference`` . This is how method schemas are checked on the sender
side.
This implies that Interfaces must be registered, just as classes are for
``pb.Copyable`` . TODO: what happens if an unknown Interface is received?
Classes which wish to be passed by value should either inherit from
``pb.Copyable`` or have an ``ICopyable`` adapter registered for them. On the
receiving side, the ``registerRemoteCopy`` function must be used to register
a factory, which can be a ``pb.RemoteCopy`` subclass or something else which
implements ``IRemoteCopy`` .
``Failure`` objects are sent as a ``pb.Copyable`` with a class name of
"twisted.python.failure.Failure" .
Implementation notes
--------------------
Outgoing Referenceables
~~~~~~~~~~~~~~~~~~~~~~~
The side which holds the ``Referenceable`` uses a ReferenceableSlicer to
serialize it. Each ``Referenceable`` is tracked with a "process-Unique ID"
(abbreviated "puid" ). As the name implies, this number refers to a specific
object within a given process: it is scoped to the process (and is never sent
to another process), but it spans multiple PB connections (any given object
will have the same ``puid`` regardless of which connection is referring to
it). The ``puid`` is an integer, normally obtained with ``id(obj)`` , but you
can override the object's ``processUniqueID`` method to use something else
(this might be useful for objects that are really proxies for something
else). Any two objects with the same ``puid`` are serialized identically.
All Referenceables sent over the wire (as arguments or return values for
remote methods) are given a "connection-local ID" (``clid`` ) which is scoped
to one end of the connection. The Referenceable is serialized with this
number, using a banana sequence of ``(OPEN "my-reference" clid)`` . The
remote peer (the side that holds the ``RemoteReference`` ) knows the
``Referenceable`` by the ``clid`` sent to represent it. These are small
integers. From a security point of view, any object sent across the wire (and
thus given a ``clid`` ) is forever accessible to the remote end (or at least
until the connection is dropped).
The sending side uses the ``Broker.clids`` dict to map``puid`` to ``clid`` .
It uses the ``Broker.localObjects`` dict to map ``clid`` to ``Referenceable``
. The reference from ``.localObjects`` also has the side-effect of making
sure the Referenceable doesn't go out of scope while the remote end holds a
reference.
``Broker.currentLocalID`` is used as a counter to create ``clid`` values.
RemoteReference
~~~~~~~~~~~~~~~
In response to the incoming ``my-reference`` sequence, the receiving side
creates a ``RemoteReference`` that remembers its Broker and the ``clid``
value. The RemoteReference is stashed in the ``Broker.remoteReferences``
weakref dictionary (which maps from ``clid`` to ``RemoteReference`` ), to
make sure that a single ``Referenceable`` is always turned into the same
``RemoteReference`` . Note that this is not infallible: if the recipient
forgets about the ``RemoteReference`` , PB will too. But if they really do
forget about it, then they won't be able to tell that the replacement is not
the same as the original [#]_ . It will have a different ``clid`` . [#]_
This ``RemoteReference`` is where the ``.callRemote`` method lives. When used
to invoke remote methods, the ``clid`` is used as the second token of a
``call`` sequence. In this context, the ``clid`` is a "your-reference" : it
refers to the recipient's ``.localObjects`` table. The ``Referenceable``
-holder's ``my-reference-id`` is sent back to them as the
``your-reference-id`` argument of the ``call`` sequence.
The ``RemoteReference`` isn't always used to invoke remote methods: it could
appear in an argument or a return value instead: the goal is to have the
``Referenceable`` -holder see their same ``Referenceable`` come back to them.
In this case, the ``clid`` is used in a ``(OPEN "your-reference" clib)``
sequence. The ``Referenceable`` -holder looks up the ``clid`` in their
``.localObjects`` table and puts the result in the method argument or return
value.
URL References
~~~~~~~~~~~~~~
In addition to the implicitly-created numerically-indexed ``Referenceable``
instances (kept in the Broker's ``.localObjects`` dict), there are
explicitly-registered string-indexed ``Referenceable`` s kept in the
PBServerFactory's ``localObjects`` dictionary. This table is used to publish
objects to the outside world. These objects are the targets of the
``pb.getRemoteURL`` and ``pb.callRemoteURL`` functions.
To access these, a ``URLRemoteReference`` must be created that refers to a
string ``clid`` instead of a numeric one. This is a simple subclass of
``RemoteReference`` : it behaves exactly the same. The ``URLRemoteReference``
is created manually by ``pb.getRemoteURL`` , rather than being generated
automatically upon the receipt of a ``my-reference`` sequence. It also
assumes a list of RemoteInterface names (which are usually provided by the
holder of the ``Referenceable`` ).
To invoke methods on a URL-indexed object, a string token is used as the
``clid`` in the "your-reference-id" argument of a ``call`` sequence.
In addition, the ``clid`` of a ``your-reference`` sequence can be a string to
use URL-indexed objects as arguments or return values of method invocations.
This allows one side to send a ``URLRemoteReference`` to the other and have
it turn into the matching ``Referenceable`` when it arrives. Of course, if it
is invalid, the method call that tried to send it will fail.
Note that these ``URLRemoteReference`` objects wil not survive a roundtrip
like regular ``RemoteReference`` s do. The ``URLRemoteReference`` turns into
a ``Referenceable`` , but the ``Referenceable`` will turn into a regular
numeric (implicit) ``RemoteReference`` when it comes back. This may change in
the future as the URL-based referencing scheme is developed. It might also
become possible for string ``clid`` s to appear in ``my-reference``
sequences, giving ``Referenceable`` -holders the ability to publish URL
references explicitly.
It might also become possible to have these URLs point to other servers. In
this case, a ``remote`` sequence will probably be used, rather than the
``my-reference`` sequence used for implicit references.
Note that these URL-endpoints are per-Factory, so they are shared between
multiple connections (the implicitly-created references are only available on
the connection that created them). The PBServerFactory is created with a
"root object" , which is a URL-endpoint with a ``clid`` of an empty string.
.. rubric:: Footnotes
.. [#] This isn't quite true: for some objects, serialization is scoped to
the connection as a whole. Referenceables and RemoteReferences are
like this.
.. [#] unless they do something crazy like remembering the ``id(obj)`` of the
old object and check to see if it is the same as that of the new one.
But ``id(obj)`` is only unique among live objects anyway
.. [#] and note that I think there is a race condition here, in which the
reference is sent over the wire at the same time the other end forgets
about it
|