+ my $dxchan = shift;
+ my $origin = $thing->{user} || $thing->{origin};
+ my $chan_call = $dxchan->{call};
+
+ my $parent = Route::Node::get($origin);
+ unless ($parent) {
+ dbg("Thingy::Rt::cf: received from $thing->{origin}/$origin on $chan_call unknown") if isdbg('chanerr');
+ return;
+ }
+
+ # do nodes
+ my ($del, $add);
+ my %in;
+ if ($thing->{n}) {
+ %in = (map {my ($here, $call) = unpack("A1 A*", $_); ($call, $here)} split /:/, $thing->{n});
+ my ($tdel, $tadd) = $parent->diff_nodes(keys %in);
+ $add = $tadd;
+ $del = $tdel;
+ }
+ if ($thing->{a}) {
+ %in = (map {my ($here, $call) = unpack("A1 A*", $_); ($call, $here)} split /:/, $thing->{a});
+ my ($tdel, $tadd) = $parent->diff_nodes(keys %in);
+ push @$add, @$tadd;
+ push @$del, @$tdel;
+ }
+ if (@$add || @$del) {
+ my $call;
+
+ my @pc21;
+ foreach $call (@$del) {
+ RouteDB::delete($call, $chan_call);
+ my $ref = Route::Node::get($call);
+ push @pc21, $ref->del($parent) if $ref;
+ }
+ $thing->{pc21n} = \@pc21 if @pc21;
+
+ my @pc19;
+ foreach $call (@$add) {
+ RouteDB::update($call, $chan_call);
+ my $ref = Route::Node::get($call);
+ push @pc19, $parent->add($call, 0, $in{$call}) unless $ref;
+ }
+ $thing->{pc19n} = \@pc19 if @pc19;
+ }
+
+ # now users
+ if ($thing->{u}) {
+ %in = (map {my ($here, $call) = unpack "A1 A*", $_; ($call, $here)} split /:/, $thing->{u});
+ ($del, $add) = $parent->diff_users(keys %in);
+
+ my $call;
+
+ my @pc17;
+ foreach $call (@$del) {
+ RouteDB::delete($call, $chan_call);
+ my $ref = Route::User::get($call);
+ if ($ref) {
+ $parent->del_user($ref);
+ push @pc17, $ref;
+ } else {
+ dbg("Thingy::Rt::lcf: del user $call not known, ignored") if isdbg('chanerr');
+ next;
+ }
+ }
+ if (@pc17) {
+ $thing->{pc17n} = $parent;
+ $thing->{pc17u} = \@pc17;
+ }
+
+ my @pc16;
+ foreach $call (@$add) {
+ RouteDB::update($call, $chan_call);
+ push @pc16, _add_user($parent, $call, $in{$call});
+ }
+ if (@pc16) {
+ $thing->{pc16n} = $parent;
+ $thing->{pc16u} = \@pc16;
+ }