changeset 202:5146e17a7b76

IMPL::Web::Application::RestResource fixes, documentation
author sergey
date Wed, 25 Apr 2012 02:49:23 +0400 (2012-04-24)
parents 0c018a247c8a
children 68a59c3358ff
files Lib/IMPL/Serialization.pm Lib/IMPL/Web/Application/RestBaseResource.pm Lib/IMPL/Web/Application/RestCustomResource.pm Lib/IMPL/Web/Application/RestResource.pm Lib/IMPL/Web/Handler/RestController.pm
diffstat 5 files changed, 266 insertions(+), 183 deletions(-) [+]
line wrap: on
line diff
--- a/Lib/IMPL/Serialization.pm	Tue Apr 24 19:52:07 2012 +0400
+++ b/Lib/IMPL/Serialization.pm	Wed Apr 25 02:49:23 2012 +0400
@@ -242,7 +242,7 @@
       return 1;
     }
     
-    my $refObj = $this->{$ObjectFactory} ?$this->{$ObjectFactory}->($rhObject->{'Type'},$rhObject->{'Data'},$rhObject->{'Id'} ? $this->{$Context}->{$rhObject->{'Id'}} : undef) : DefaultFactory($rhObject->{'Type'},$rhObject->{'Data'} || [],$rhObject->{'Id'} ? $this->{$Context}->{$rhObject->{'Id'}} : undef);
+    my $refObj = $this->{$ObjectFactory} ?$this->{$ObjectFactory}->($rhObject->{'Type'},$rhObject->{'Data'},$rhObject->{'Id'} ? $this->{$Context}->{$rhObject->{'Id'}} : undef) : DefaultFactory($rhObject->{'Type'},$rhObject->{'Data'},$rhObject->{'Id'} ? $this->{$Context}->{$rhObject->{'Id'}} : undef);
       
     die new Exception("Trying to close a non existing oject") if not $rhObject;
   
@@ -329,6 +329,7 @@
       return \$Data;
     }
   } elsif ($Type eq 'ARRAY') {
+  	$Data ||= [];
     die new Exception("Invalid a deserialization context when deserializing ARRAY") if not ref $Data and defined $Data;
     if (not ref $refSurogate) {
       my @Array;
@@ -339,6 +340,7 @@
     }
     return $refSurogate;
   } elsif ($Type eq 'HASH') {
+  	$Data ||= [];
     die new Exception("Invalid a deserialization context when deserializing HASH") if not ref $Data and defined $Data;
     if (not ref $refSurogate) {
       $refSurogate = {};
@@ -348,6 +350,7 @@
     }
     return $refSurogate;
   } elsif ($Type eq 'REF') {
+  	$Data ||= [];
     die new Exception("Invalid a deserialization context when deserializing REF") if not ref $Data and defined $Data;
     if (not ref $refSurogate) {
       my $ref = $Data->[1]; 
--- a/Lib/IMPL/Web/Application/RestBaseResource.pm	Tue Apr 24 19:52:07 2012 +0400
+++ b/Lib/IMPL/Web/Application/RestBaseResource.pm	Wed Apr 25 02:49:23 2012 +0400
@@ -81,61 +81,6 @@
 	return undef;
 }
 
-sub InvokeMember {
-    my ($this,$method,$action) = @_;
-    
-    die ArgumentException->new("method","No method information provided") unless $method;
-    
-    #normalize method info
-    if (not ref $method) {
-        $method = {
-            method => $method
-        };
-    }
-    
-    if (ref $method eq 'HASH') {
-        my $member = $method->{method} or die InvalidOpException->new("A member name isn't specified");
-        
-        my @args;
-    
-        if (my $params = $method->{parameters}) {
-            if (ref $params eq 'HASH') {
-                @args = map {
-                    $_,
-                    $this->MakeParameter($params->{$_},$action)
-                } keys %$params;                
-            } elsif (ref $params eq 'ARRAY') {
-                @args = map $this->MakeParameter($_,$action), @$params;
-            } else {
-                @args = ($this->MakeParameter($params,$action)); 
-            }
-        }
-        return $this->target->$member(@args);
-    } elsif (ref $method eq TResolve) {
-        return $method->Invoke($this->target);
-    } elsif (ref $method eq 'CODE') {
-        return $method->($this,$action);
-    } else {
-        die InvalidOpException->new("Unsupported type of the method information", ref $method);
-    }
-}
-
-sub MakeParameter {
-    my ($this,$param,$action) = @_;
-    
-    if ($param) {
-        if (is $param, TTransform ) {
-            return $param->Transform($this,$action->query);
-        } elsif ($param and not ref $param) {
-            return $action->query->param($param);
-        } else {
-        	die new InvalidOpException->new("Unsupported parameter mapping", $param);
-        }
-    } else {
-        return undef;
-    }
-}
-
 
 1;
 
--- a/Lib/IMPL/Web/Application/RestCustomResource.pm	Tue Apr 24 19:52:07 2012 +0400
+++ b/Lib/IMPL/Web/Application/RestCustomResource.pm	Wed Apr 25 02:49:23 2012 +0400
@@ -21,12 +21,6 @@
 	public property delete => PROP_GET | PROP_OWNERSET;
 }
 
-sub CTOR {
-	my ($this) = @_;
-	
-	die ArgumentException->new("parent") unless $this->parent; 
-}
-
 sub FetchChildResource {
 	my ($this,$id,$action) = @_;
 	
@@ -60,4 +54,10 @@
     return $this->InvokeMember($method,$action);
 }
 
+sub InvokeMember {
+	my ($this,$method,$action) = @_;
+	
+	return $this->$method($action);
+}
+
 1;
\ No newline at end of file
--- a/Lib/IMPL/Web/Application/RestResource.pm	Tue Apr 24 19:52:07 2012 +0400
+++ b/Lib/IMPL/Web/Application/RestResource.pm	Wed Apr 25 02:49:23 2012 +0400
@@ -1,7 +1,7 @@
 package IMPL::Web::Application::RestResource;
 use strict;
 
-use IMPL::lang qw(:declare :constants is);
+use IMPL::lang qw(:declare :constants is :hash);
 use IMPL::Exception();
 
 use IMPL::declare {
@@ -12,7 +12,7 @@
 		ArgumentException => '-IMPL::InvalidArgumentException',
 		TTransform => '-IMPL::Transform',
 		TResolve => '-IMPL::Config::Resolve',
-		CustomResource => 'IMPL::Web::Application::CustomResource'
+		CustomResource => 'IMPL::Web::Application::RestCustomResource'
 	},
 	base => {
 		'IMPL::Web::Application::RestCustomResource' => '@_'
@@ -37,46 +37,50 @@
 	die ArgumentException->new("target") unless $this->target;
 	
 	$this->final($this->childRegex ? 0 : 1);
+	$this->methods({}) unless $this->methods;
 	
 	if ($this->enableForms) {
         $this->methods->{create} = {
-        	get => \&_ParentGet,
-        	post => \&_ParentPost,
+        	get => $this->get,
+        	post => $this->post,
         	final => 1 # this resource doesn't have any children
-        };
+        } if $this->post;
         
         $this->methods->{edit} = {
-        	get => \&_ParentGet,
-        	post => \&_ParentPut,
+        	get => $this->get,
+        	post => $this->put,
         	final => 1 # this resource doesn't have any children
-        };
+        } if $this->put;
         
         $this->methods->{delete} {
-        	get => \&_ParentGet,
-        	post => \&_ParentDelete
+        	get => $this->get,
+        	post => $this->delete,
         	final => 1 # this resource doesn't have any children
-        }
+        } if $this->delete;
 	}
 }
 
-sub _ParentGet {
-	my ($this,$action) = @_;
-	return $this->parent->GetImpl($action);
-}
-
-sub _ParentPut {
-    my ($this,$action) = @_;
-    return $this->parent->PutImpl($action);
-}
-
-sub _ParentPost {
-    my ($this,$action) = @_;
-    return $this->parent->PostImpl($action);
-}
-
-sub _ParentDelete {
-    my ($this,$action) = @_;
-    return $this->parent->DeleteImpl($action);
+# создает дочерний ресурс из описания, однако все методы созданного
+# ресурса переадресуются к его родителю, это нужно, чтобы публиковать
+# методы и свойства объекта
+sub _CreateSubResource {
+	my ($this,$resource,$id) = @_;
+	
+	my %methods = map {
+        my $method = $resource->{$_};
+        $_ => sub {
+            my ($this,$action) = @_;
+            return $this->parent->InvokeMember($method,$action);
+        };           
+    } grep $resource->{$_}, qw(get post put delete);
+        
+    return CustomResource->new(
+        %methods,
+        final => $resource->{final},
+        parent => $this,
+        id => $id,
+        contract => $this->contract
+    );
 }
 
 sub FetchChildResource {
@@ -93,21 +97,19 @@
 		
 		$res = $this->InvokeMember($method,$action);
 		
-	} elsif (my $resource = $this->methods->{$id}) {
-		return CustomResource->new(
-            get => $resource->{get},
-            post => $resource->{post},
-            put => $resource->{put},
-            delete => $resource->{delete},
-            parent => $this,
-            id => $id,
-            target => $this->target
-		);
-		 
-	} elsif ($rx and $id =~ m/^$rx$/ and $method = $this->fetch) {
+	} elsif ($this->methods and my $resource = $this->methods->{$id}) {
+		
+		return $this->_CreateSubResource($resource,$id);
+		
+	} elsif ($rx and $id =~ m/^$rx$/ and my $method = $this->fetch) {
+		
+		$method = {
+			method => $method,
+			parameters => 'id'
+		} unless ref $method;
+		
 		$res = $this->InvokeMember($method,$action, { id => $id } );
-	} else {
-		die ForbiddenException->new();
+		
 	}
 	
     die NotFoundException->new() unless defined $res;
@@ -115,6 +117,74 @@
     return $this->contract->Transform($res, {parent => $this, id => $id} );
 }
 
+sub InvokeMember {
+    my ($this,$method,$action,$predefined) = @_;
+    
+    die ArgumentException->new("method","No method information provided") unless $method;
+    
+    #normalize method info
+    if (not ref $method) {
+        $method = {
+            method => $method
+        };
+    }
+    
+    if (ref $method eq 'HASH') {
+        my $member = $method->{method} or die InvalidOpException->new("A member name isn't specified");
+        
+        $member = $member->Invoke($this) if eval { $member->isa(TResolve) };
+        
+        my @args;
+    
+        if (my $params = $method->{parameters}) {
+            if (ref $params eq 'HASH') {
+                @args = map {
+                    $_,
+                    $this->MakeParameter($params->{$_},$action,$predefined)
+                } keys %$params;                
+            } elsif (ref $params eq 'ARRAY') {
+                @args = map $this->MakeParameter($_,$action,$predefined), @$params;
+            } else {
+                @args = ($this->MakeParameter($params,$action,$predefined)); 
+            }
+        }
+        return $this->target->$member(@args);
+    } elsif (ref $method eq TResolve) {
+        return $method->Invoke($this);
+    } elsif (ref $method eq 'CODE') {
+        return $method->($this,$action);
+    } else {
+        die InvalidOpException->new("Unsupported type of the method information", ref $method);
+    }
+}
+
+sub MakeParameter {
+    my ($this,$param,$action,$predefined) = @_;
+    
+    my $params = hashApply(
+	    {
+	    	id => $this->id,
+	    	action => $action,
+	    	query => $action->query
+	    },
+	    $predefined || {}
+	);
+    
+    
+    
+    if ($param) {
+        if (is $param, TTransform ) {
+            return $param->Transform($action->query);
+        } elsif ($param and not ref $param) {
+            return $params->{$param} || $action->query->param($param);
+        } else {
+            die InvalidOpException->new("Unsupported parameter mapping", $param);
+        }
+    } else {
+        return undef;
+    }
+}
+
 1;
 
 __END__
@@ -192,7 +262,10 @@
     			}
     		}
     	}
-    	list => 'search',
+    	index => {
+    		method => 'search',
+    		paremeters => [qw(filter page limit)]
+    	},
     	fetch => 'GetItemById'
     }   
 );
@@ -203,47 +276,41 @@
 
 Каждый ресурс представляет собой коллекцию и реализует методы C<HTTP> C<GET,POST,PUT,DELETE>.
 
-Ресурсы выстраиваются в иерархию, на основе пути. Поиск конечного реурса происходит последовательным
-вызовом метода GET с именем очередного ресурса. 
-  
+Вызов каждого из этих методов позволяет выполнить одну из операций над ресурсом, однако
+операций может быть больше, для этого создаются дочерние ресурсы (каждый из которых также
+может иметь четыре метода C<GET,POST,PUT,DELETE>), однако обращения к методам у дочерних
+ресурсов отображаются в вызовы методов у родительского ресурса.
+
+Такой подход позволяет расширить функциональность не изменяя стандарт C<HTTP>, а также обойти
+ограничения браузеров на методы C<PUT,DELETE>.
+
+Данный тип ресутсов расчитан на использование с конфигурацией, которую можно будет
+сохранить или прочитать, например, из файла. Для этого у ресурса есть ряд настроек,
+которые позволяют в простой форме задать отображения между C<HTTP> методами и методами
+объекта представленного данным ресурсом.
+
 
 =head2 HTTP METHODS
 
 =head3 C<GET>
 
-Возвращает коллекцию дочерних ресурсов.
-
-=head3 C<GET {id}>
+Возвращает данные из текущего ресурса. Обращение к данному методу не должно вносить
+изменений в ресурсы. 
 
-Возвращает дочерний объект с идентификатором C<id>
-
-=head3 C<GET {method}>
+=head3 C<PUT>
 
-Вызывает метод C<method> и возвращает его результаты. При публикации методов доступных
-через C<GET> данные методы не должны вносить изменений в предметную область.
-
-=head3 C<PUT {id}>
+Обновляет ресурс. Повторное обращение к данному методу должно приводить к одному и
+томуже результату.
 
-Обновляет дочерний ресурс с указанным идентификатором.
+=head3 C<DELETE>
 
-=head3 C<DELETE {id}>
-
-Удаляет дочерний ресурс с указанным идентификатором.
+Удаляет ресурс.
 
 =head3 C<POST>
 
-Добавляет новый дочерний ресурс в коллекцию.
-
-=head3 C<POST {method}>
-
-Вызывает метод C<method>.
-
-=head2 HTTP METHOD MAPPING 
-
-=head3 C<POST {method}>
-
-Вызывает метод C<method>, в отличии от C<GET> методы опубликованные через C<POST> могут вносить
-изменения в объекты. 
+Данный метод может вести себя как угодно, однако обычно он используется для добавления
+нового дочернего ресурса в коллекцию,также может использоваться для вызова метода, в случае
+если происходит публикация методов в качестве дочерних ресурсов.
 
 =head1 BROWSER COMPATIBILITY
 
@@ -251,29 +318,106 @@
 C<GET,POST>. Для решения данной проблемы используется режим совместимости C<enableForms>. В
 случае когда данный режим активен, автоматически публикуются дочерние ресурсы C<create,edit,delete>.
 
-=head2 C<GET create>
+Данные ресуры пбликуются как методы, что означает то, что обращения к ним будут превращены в
+выполнения соответсвующих методов на родительском объекте.
+
+=head2 C<create>
+
+По сути данные ресурс не является необходимостью, однако создается для целостности модели.
+
+=head3 C<GET>
+
+Передает управление методу C<get>
 
-Возвращает C<target>.
+=head3 C<POST>
+
+Передает управление методу C<post>
+
+=head2 C<edit>
 
-=head2 C<POST create>
+=head3 C<GET>
+
+Передает управление методу C<get>
+
+=head3 C<POST>
 
-Вызывает метод C<PostImpl> передавая ему свои параметры.
+Передает управление методу C<put>, как если бы он был выполнен непосредственно у
+родительского ресурса.
+
+=head2 C<delete>
+
+=head3 C<GET>
+
+Передает управление методу C<get>
+
+=head3 C<POST>
 
-=head2 C<GET edit>
+Передает управление методу C<delete>, , как если бы он был выполнен непосредственно у
+родительского ресурса.
+
+=head1 METHOD DEFINITIONS
 
-Возвращает C<target>.
+Все методы ресурсов данного типа задаются описаниями, хранящимися в соответствующих
+свойствах. Когда наступает необходимость вызова соответствующего метода, его описание
+бедется из свойства и передается методу C<InvokeMember>, который и производит вызов.
+
+=head2 C<HASH>
+
+Содержит в себе описание метода, который нужно вызвать, а также его параметры.
+
+=over
 
-=head2 C<POST edit>
+=item C<method>
+
+Имя метода который будет вызван.
+
+=item C<paremeters>
+
+Описание параметров метода, может быть либо массивом, либо хешем, либо простым
+значением.
+
+=over
+
+=item C<ARRAY>
 
-Вызывает метод C<$this->parent->PutImpl($this->id)> передавая ему свои параметры.
+Метод получает список параметров, каждый элемент данного массива будет превращен
+в параметр при помощи метода C<MakeParameter>
+
+=item C<HASH>
 
-=head2 C<GET delete>.
+Метод получает список параметров, который состоит пар ключ-значение, каждое значение 
+данного хеша будет превращено в зачение параметра метода при помощи метода C<MakeParameter>.
+Ключи хеша изменениям не подвергаются.
+
+=item Простое значение
+
+Метод получает одно значение, которое будет получено из текущего при помощи C<MakeParameter>.
 
-Возвращает C<target>.
+=back
+
+=back
+
+=head2 C<CODE>
+
+Если в описании метода находится ссылка на функцию, то эта функция будет вызвана с параметрами.
+Данный вариант полезен когда ресурсы создаются програмно обычного механизма описаний не достаточно
+для реализации требуемого функционала.
+
+=over
 
-=head2 C<POST delete>.
+=item C<$resource>
+
+Текущий ресурс у которого производится вызов метода.
+
+=item C<$action>
 
-Вызывает метод C<$this->parent->DeleteImpl($this->id)> передавая ему свои параметры.
+Текущий запрос C<IMPL::Web::Application::Action>.
+
+=back
+
+=head2 Простое значение
+
+Интерпретируется как имя метода у объекта данных текущего ресурса.
 
 =head1 MEMBERS
 
@@ -283,7 +427,8 @@
 
 =head2 C<[get]target>
 
-Объект (также может быть и класс), обеспечивающий функционал ресурса.
+Объект данных (может быть и класс, поскольку у него будут только вызываться
+методы), обеспечивающий функционал ресурса.
 
 =head2 C<[get]parent>
 
@@ -305,34 +450,30 @@
 отсутствует, то дочерние ресурсы не получится адресовать относительно данного.
 По умолчанию получает идентификатор дочернего ресурса первым параметром.  
 
-=head2 C<[get]list>
+=head2 C<[get]index>
 
 Описание метода для получения списка дочерних объектов. По умолчанию не
 получает параметров.
 
-=head2 C<[get]insert>
+=head2 C<[get]post>
 
 Описание метода для добавление дочернего ресурса. По умолчанию получает
 объект C<CGI> описывабщий текущий запрос первым параметром.
 
-=head2 C<[get]update>
+=head2 C<[get]put>
 
 Описание метода для обновления дочернего ресурса. По умолчанию получает
-идентификатор дочернего ресурса и объект C<CGI> текущего запроса.
+объект C<CGI> текущего запроса.
 
 =head2 C<[get]delete>
 
-Описание метода для удаления дочернего ресурса. По умолчанию получает
-идентификатор дочернего ресурса.
+Описание метода для удаления дочернего ресурса. По умолчанию не получает
+параметров.
 
-=head2 C<GetImpl($child,$action)>
+=head2 C<GetImpl($action)>
 
 =over
 
-=item C<$child>
-
-Идентификатор дочернего ресутсра
-
 =item C<$action>
 
 Текущий запрос C<IMPL::Web::Application::Action>.
@@ -342,14 +483,10 @@
 Переадресует запрос нужному методу внутреннего объекта C<target> при
 помощи C<InvokeMember>.
 
-=head2 C<PutImpl($child,$action)>
+=head2 C<PutImpl($action)>
 
 =over
 
-=item C<$child>
-
-Идентификатор дочернего ресутсра
-
 =item C<$action>
 
 Текущий запрос C<IMPL::Web::Application::Action>.
@@ -359,14 +496,10 @@
 Переадресует запрос нужному методу внутреннего объекта C<target> при
 помощи C<InvokeMember>.
 
-=head2 C<PostImpl($child,$action)>
+=head2 C<PostImpl($action)>
 
 =over
 
-=item C<$child>
-
-Идентификатор дочернего ресутсра
-
 =item C<$action>
 
 Текущий запрос C<IMPL::Web::Application::Action>.
@@ -376,14 +509,10 @@
 Переадресует запрос нужному методу внутреннего объекта C<target> при
 помощи C<InvokeMember>.
 
-=head2 C<DeleteImpl($child,$action)>
+=head2 C<DeleteImpl($action)>
 
 =over
 
-=item C<$child>
-
-Идентификатор дочернего ресутсра
-
 =item C<$action>
 
 Текущий запрос C<IMPL::Web::Application::Action>.
@@ -393,7 +522,7 @@
 Переадресует запрос нужному методу внутреннего объекта C<target> при
 помощи C<InvokeMember>.
 
-=head2 C<InvokeMember($memberInfo,$child,$action)>
+=head2 C<InvokeMember($memberInfo,$action,$params)>
 
 =over
 
@@ -401,20 +530,20 @@
 
 Описание члена внутреннего объекта C<target>, который нужно вызвать.
 
-=item C<$child>
-
-Идентификатор дочернего ресутсра
-
 =item C<$action>
 
 Текущий запрос C<IMPL::Web::Application::Action>.
 
+=item C<$params>
+
+Ссылка на хеш с предопределенными параметрами.
+
 =back
 
 Вызывает метод внутреннего объекта C<target>, предварительно подготовив
 параметры на основе описания C<$memberInfo> и при помощи С<MakeParameter()>.
 
-=head2 C<MakeParameter($paramDef,$child,$action)>
+=head2 C<MakeParameter($paramDef,$action)>
 
 =over
 
@@ -428,7 +557,7 @@
 
 =item C<id>
 
-Идентификатор дочернего ресурса
+Идентификатор ресурса
 
 =item C<query>
 
@@ -447,6 +576,10 @@
 Если описание параметра - объект C<IMPL::Transform>, то будет выполнено это преобразование над C<CGI>
 объектом текущего запроса C<< $paramDef->Transform($action->query) >>.
 
+=item C<$action>
+
+Текущий запрос
+
 =back
 
 =cut
\ No newline at end of file
--- a/Lib/IMPL/Web/Handler/RestController.pm	Tue Apr 24 19:52:07 2012 +0400
+++ b/Lib/IMPL/Web/Handler/RestController.pm	Wed Apr 25 02:49:23 2012 +0400
@@ -47,6 +47,7 @@
 	shift @segments;
 	
 	my ($obj,$view) = (pop(@segments) =~ m/(.*?)(?:\.(\w+))?$/);
+	push @segments, $obj;
 	
 	if ($this->types and my $type = $this->types->{$view}) {
         $action->response->contentType($type);		
@@ -56,12 +57,13 @@
 	
 	while(@segments) {
 		my $id = shift @segments;
-		$res = $this->contract->Transform( $res->InvokeHttpMethod('GET',$id,$action), { parent => $res, id => $id } );
+		
+		$res = $res->FetchChildResource($id,$action);
 		
 		die NotFoundException->new() unless $res;
 	}
 	
-	$res = $res->InvokeHttpMethod($method,$obj,$action);
+	$res = $res->InvokeHttpMethod($method,$action);
 }
 
 1;