Robert LemkeGetting intoFLOW3
Robert Lemkeproject founder of FLOW3 and TYPO3 “Phoenix”co-founder of the TYPO3 Associationcoach, coder, consultant36 year...
StartersInstallationKickstart & Hello World!Controller, Actions, Arguments & HTTPDomain-Driven DesignDoctrineForms, Valida...
Main DishesResources, Image UploadSession HandlingUser, Account, AuthenticationAuthorization
Deserts Caching       Testing Logging       Deployment Signal-Slot   I18n Routing       Espresso
?
At a GlanceFLOW3 is a web application platform • holistic concept for your apps • modular, extensible, package based • ped...
Foundation for the Next Generation CMSTYPO3 “Phoenix” is the all-newEnterprise CMS • content repository, workspaces,   ver...
FLOW3 Website and Download#
Git Clone$ git clone --recursive git://git.typo3.org/FLOW3/Distributions/Base.git .Cloning into ....remote: Counting objec...
Set File Permissions  $ sudo ./flow3 core:setfilepermissions robert _www _www  FLOW3 File Permission Script  Checking perm...
Set Up Database ConnectionConfiguration/Settings.yaml #                                                       # # Global Se...
Set Up Virtual HostApache Virtual Host  <VirtualHost *:80>       DocumentRoot /opt/local/apache2/htdocs/Talks/FLOW3/Web/  ...
Final Check
C lo ne t h e Be a s t 5         1                1   1        Ro bert Lem ke         D.P. F l u x t r        time ();
Biggest Book Store: Amazon
Biggest River: Amazon River                              © Google
Smallest River: Roe River                            © Google
Smallest River: Roe River                            © Google
Smallest River: Roe River                            © Google
Smallest River: Roe River
Smallest Book Store: Roe Books
Sketchy Model
Jus t C o de a Sh o p 5        2                1   1       Ro bert Lem ke        D.P. F l u x t r       time ();
Tackling the Heart of Software Development                                         /**Domain-Driven Design                ...
Domain-Driven DesignDomain activity or business of the userDomain-Driven Design is about • focussing on the domain and dom...
Domain-Driven DesignUbiquitous Language • important prerequisite for successful   collaboration • use the same words for  ...
Domain-Driven DesignBuilding Blocks • Entity: An object that is not defined by its attributes, but rather by a   thread of ...
Domain-Driven DesignBuilding Blocks • Service: When an operation does not conceptually belong to any   object. Following t...
Domain-Driven Design
Object ManagementFLOW3s take on Dependency Injection • one of the first PHP implementations   (started in 2006, improved ev...
Constructor Injectionnamespace AcmeDemoController;use TYPO3FLOW3MvcControllerActionController;use AcmeDemoServiceGreeterSe...
Setter Injectionnamespace AcmeDemoController;use TYPO3FLOW3MVCControllerActionController;use AcmeDemoServiceGreeterService...
Property Injectionnamespace TYPO3DemoController;use TYPO3FLOW3Annotations as FLOW3;use TYPO3FLOW3MVCControllerActionContro...
Objects.yaml   TYPO3FLOW3SecurityCryptographyRsaWalletServiceInterface:     className: TYPO3FLOW3SecurityCryptographyRsaWa...
Object Managementclass Customer {	   /**	    * @FLOW3Inject	    * @var AcmeCustomerNumberGenerator	    */	   protected $cu...
Object Management                                     <?php                                     declare(ENCODING = u      ...
Basic Object Persistence	   	 // Create a new customer and persist it:	   $customer = new Customer("Robert");	   $this->cu...
Validation and Doctrine Annotationsnamespace TYPO3BlogDomainModel;/** * A Blog object * * @Entity */class Blog {    /**   ...
Persistence-related Annotations@Entity        Declares a class as "entity"@Column        Controls the database column rela...
Persistence-related Annotations@var           Defines the type of a property, collections can be               typed using ...
Custom Queries using the Query Object Model/** * A PostRepository */class PostRepository extends TYPO3FLOW3PersistenceRepo...
Schema ManagementRunning Migrations• needed after installation or upgrade:$ ./flow3 doctrine:migrate
Schema ManagementManual database updates• for simple situations this can be good enough:$ ./flow3 doctrine:create$ ./flow3...
Schema ManagementGenerating migrations$ ./flow3 doctrine:migrationgenerateGenerated new migration class to "…/Version20110...
ValidationValidation in FLOW3• you do not want to code checks into your controllers• FLOW3 separates validation from your ...
ValidationValidation Models• BaseProperties  rules defining the minimum requirements on individual properties of a  model• ...
ValidationBase Properties• Validation rules defined directly at the properties  	   /**  	    * @var string  	    * @valida...
ValidationValidators• validators provided by FLOW3 can be used through their short name • Count, Float, NotEmpty, RegularE...
Property Mapper	   $articleArray = array(	   	 headline => Hello World!,	   	 story => Just a demo ...	   );	   $article =...
Resource Management Image Upload Resources are handled like other properties in a form:	 <f:form method="blog" action="upd...
Property Mapper    Allow nested object structures    For security reasons the creation of nested structure through the    ...
FluidExample for assigning a string to a Fluid variable:	   	 // in the action controller:	   $this->view->assign(title, W...
FluidVariables can also be objects:	   	 // in the action controller:	   $this->view->assign(conference, $conference);	   ...
Fluidif-then-else:	   	 // in the action controller:	   $this->view->assign(post, $blogPost);	   <!-- in the Fluid templat...
Fluidfor-each:	   	 // in the action controller:	   $this->view->assign(ages, array("Karsten" => 34, "Robert" => 35));	   ...
Fluidfor-each:	   	 // in the action controller:	   $this->view->assign(post, $blogPost);	   <!-- in the Fluid template: -...
FluidView helpers – in this case the link.action view helper:	   <!-- in the Fluid template: -->	   {namespace f=TYPO3Flui...
Security Policy
SecurityCross-Site Request Forgery • enables an attacker to execute privileged operations without being   authenticated • ...
SecurityAvoiding Cross-Site Request Forgery • add a (truly!) random string token to each link or form • make sure this tok...
SecurityCSRF Protection in FLOW3 • you must not forget to add that token to any link • FLOW3 automatically adds the CSRF t...
Roadmaphttp://forge.typo3.org/projects/flow3-distribution-base/roadmap
Conference Appgit://git.typo3.org/TYPO3v5/Distributions/Conference.git
Blog Appgit://git.typo3.org/FLOW3/Applications/Blog.git
Rossmann• second biggest drug store  in Germany• 5,13 billion € turnover• 31,000 employees
Customer Database• custom persistence with CouchDB• SOAP support• continuous delivery• cluster setup                      ...
Amadeus• world’s biggest  e-ticket provider• 217 markets• 948 million billable  transactions / year• 2,7 billion € revenue
Social Media Suite• central hub for social media activities  for potentially thousands of travel  agencies• advanced form ...
“Our senior developers areextremely happy with FLOW3 –it is definitely the mostcapable PHP framework we       Fabian Pfütze...
?
Thanks for having me!Slides:     http://slideshare.net/robertlemkeExamples:   http://github.com/robertlemkeBlog:       htt...
Getting Into FLOW3 (DPC12)
Upcoming SlideShare
Loading in …5
×

Getting Into FLOW3 (DPC12)

1,847
-1

Published on

FLOW3 is a web application platform which uses Domain-Driven Design as its major underlying concept. This approach makes FLOW3 easy to learn and at the same time clean and flexible for complex projects. It features namespaces, has an emphasis on clean, object-oriented code and provides a seemless Doctrine 2 integration.

FLOW3 incorporates Dependency Injection in a way which lets you truly enjoy creating a stable and easy-to-test application architecture (no configuration necessary). Being the only Aspect-Oriented Programming capable PHP framework, FLOW3 allows you to cleanly separate cross-cutting concerns like security from your main application logic.

This tutorial takes you through an imaginary project from scratch. During the journey we’ll visit all important areas of the framework.

Published in: Technology
0 Comments
2 Likes
Statistics
Notes
  • Be the first to comment

No Downloads
Views
Total Views
1,847
On Slideshare
0
From Embeds
0
Number of Embeds
3
Actions
Shares
0
Downloads
13
Comments
0
Likes
2
Embeds 0
No embeds

No notes for slide

Getting Into FLOW3 (DPC12)

  1. 1. Robert LemkeGetting intoFLOW3
  2. 2. Robert Lemkeproject founder of FLOW3 and TYPO3 “Phoenix”co-founder of the TYPO3 Associationcoach, coder, consultant36 years oldlives in Lübeck, Germany1 wife, 2 daughters, 1 espresso machinelikes drumming
  3. 3. StartersInstallationKickstart & Hello World!Controller, Actions, Arguments & HTTPDomain-Driven DesignDoctrineForms, Validation
  4. 4. Main DishesResources, Image UploadSession HandlingUser, Account, AuthenticationAuthorization
  5. 5. Deserts Caching Testing Logging Deployment Signal-Slot I18n Routing Espresso
  6. 6. ?
  7. 7. At a GlanceFLOW3 is a web application platform • holistic concept for your apps • modular, extensible, package based • pedantically clean with focus on quality • puts a smile on developer’s faces • free & Open Source (LGPL v3) • backed by one of the largest Open Source projects
  8. 8. Foundation for the Next Generation CMSTYPO3 “Phoenix” is the all-newEnterprise CMS • content repository, workspaces, versions, i18n, modular UI ... • powered by FLOW3 • compatible code base • use TYPO3 features in FLOW3 standalone apps as you like
  9. 9. FLOW3 Website and Download#
  10. 10. Git Clone$ git clone --recursive git://git.typo3.org/FLOW3/Distributions/Base.git .Cloning into ....remote: Counting objects: 3837, done.remote: Compressing objects: 100% (2023/2023), done.remote: Total 3837 (delta 2007), reused 2721 (delta 1465)Receiving objects: 100% (3837/3837), 3.49 MiB | 28 KiB/s, done.Resolving deltas: 100% (2007/2007), done.
  11. 11. Set File Permissions $ sudo ./flow3 core:setfilepermissions robert _www _www FLOW3 File Permission Script Checking permissions from here upwards. Making sure Data and Web/_Resources exist. Setting file permissions, trying to set ACLs via chmod ... Done.Linux: $ sudo usermod -a -G www-data robertMac OS X: $ sudo dscl . -append /Groups/_www GroupMembership robert
  12. 12. Set Up Database ConnectionConfiguration/Settings.yaml # # # Global Settings # # # TYPO3: FLOW3: persistence: backendOptions: dbname: demo user: demo password: password host: 127.0.0.1 # only on Windows: core: phpBinaryPathAndFilename: C:/path/to/php.exe
  13. 13. Set Up Virtual HostApache Virtual Host <VirtualHost *:80> DocumentRoot /opt/local/apache2/htdocs/Talks/FLOW3/Web/ ServerName dev.flow3.rob SetEnv FLOW3_CONTEXT Development </VirtualHost> <VirtualHost *:80> DocumentRoot /opt/local/apache2/htdocs/Talks/FLOW3/Web/ ServerName flow3.rob SetEnv FLOW3_CONTEXT Production </VirtualHost>
  14. 14. Final Check
  15. 15. C lo ne t h e Be a s t 5 1 1 1 Ro bert Lem ke D.P. F l u x t r time ();
  16. 16. Biggest Book Store: Amazon
  17. 17. Biggest River: Amazon River © Google
  18. 18. Smallest River: Roe River © Google
  19. 19. Smallest River: Roe River © Google
  20. 20. Smallest River: Roe River © Google
  21. 21. Smallest River: Roe River
  22. 22. Smallest Book Store: Roe Books
  23. 23. Sketchy Model
  24. 24. Jus t C o de a Sh o p 5 2 1 1 Ro bert Lem ke D.P. F l u x t r time ();
  25. 25. Tackling the Heart of Software Development /**Domain-Driven Design * A Book * * @FLOW3Scope(“protot ype”) * @FLOW3EntityA methodology which ... */ class Book { • results in rich domain models /** * @var string */ • provides a common language protected $title; across the project team /** * @var string */ • simplify the design of complex protected $isbn; applications /** * @var string */ protected $description ;FLOW3 is the first PHP framework /**tailored to Domain-Driven Design * @var integer */ protected $price;
  26. 26. Domain-Driven DesignDomain activity or business of the userDomain-Driven Design is about • focussing on the domain and domain logic • accurately mapping the concepts to software • forming a ubiquitous language among the project members
  27. 27. Domain-Driven DesignUbiquitous Language • important prerequisite for successful collaboration • use the same words for • discussion • modeling • development • documentation
  28. 28. Domain-Driven DesignBuilding Blocks • Entity: An object that is not defined by its attributes, but rather by a thread of continuity and its identity. • Value Object: An object that contains attributes but has no conceptual identity. They should be treated as immutable. • Aggregate: A collection of objects that are bound together by a root entity, otherwise known as an aggregate root. The aggregate root guarantees the consistency of changes being made within the aggregate by forbidding external objects from holding references to its members.
  29. 29. Domain-Driven DesignBuilding Blocks • Service: When an operation does not conceptually belong to any object. Following the natural contours of the problem, you can implement these operations in services. • Repository: methods for retrieving domain objects should delegate to a specialized Repository object such that alternative storage implementations may be easily interchanged.
  30. 30. Domain-Driven Design
  31. 31. Object ManagementFLOW3s take on Dependency Injection • one of the first PHP implementations (started in 2006, improved ever since) • object management for the whole lifecycle of all objects • no unnecessary configuration if information can be gatered automatically (autowiring) • intuitive use and no bad magical surprises • fast! (like hardcoded or faster)
  32. 32. Constructor Injectionnamespace AcmeDemoController;use TYPO3FLOW3MvcControllerActionController;use AcmeDemoServiceGreeterService;class DemoController extends ActionController { /** * @var AcmeDemoServiceGreeterService */ protected $greeterService; /** * @param AcmeDemoServiceGreeterService */ public function __construct(GreeterService $greeterService) { $this->greeterService = $greeterService; } /** * @param string $name */ public function helloAction($name) { return $this->greeterService->greet($name); }}
  33. 33. Setter Injectionnamespace AcmeDemoController;use TYPO3FLOW3MVCControllerActionController;use AcmeDemoServiceGreeterService;class DemoController extends ActionController { /** * @var AcmeDemoServiceGreeterService */ protected $greeterService; /** * @param AcmeDemoServiceGreeterService */ public function injectGreeterService(GreeterService $greeterService) { $this->greeterService = $greeterService; } /** * @param string $name */ public function helloAction($name) { return $this->greeterService->greet($name); }}
  34. 34. Property Injectionnamespace TYPO3DemoController;use TYPO3FLOW3Annotations as FLOW3;use TYPO3FLOW3MVCControllerActionController;use AcmeDemoServiceGreeterService;class DemoController extends ActionController { /** * @var TYPO3DemoServiceGreeterService * @FLOW3Inject */ protected $greeterService; /** * @param string $name */ public function helloAction($name) { return $this->greeterService->greet($name); }}
  35. 35. Objects.yaml TYPO3FLOW3SecurityCryptographyRsaWalletServiceInterface: className: TYPO3FLOW3SecurityCryptographyRsaWalletServicePhp scope: singleton properties: keystoreCache: object: factoryObjectName: TYPO3FLOW3CacheCacheManager factoryMethodName: getCache arguments: 1: value: FLOW3_Security_Cryptography_RSAWallet
  36. 36. Object Managementclass Customer { /** * @FLOW3Inject * @var AcmeCustomerNumberGenerator */ protected $customerNumberGenerator; ...}$customer = new Customer();$customer->getCustomerNumber();
  37. 37. Object Management <?php declare(ENCODING = u tf-8); namespace TYPO3Conf erenceDomainModel use TYPO3FLOW3Anno Conference; tations as FLOW3;FLOW3 creates proxy classes /** * Autogenerated Prox y Classfor realizing DI and AOP magic * @FLOW3Scope(“prot otype”) * @FLOW3Entity */ • new operator is supported class Paper extends TYPO3FLOW3Persist Paper_Original implem ents TYPO3FLOW3Obj ect enceAspectPersiste nceMagicInterface { /** • proxy classes are created * @var string on the fly * @ORMId * @ORMColumn(length ="40") * introduced by TYPO 3FLOW3Persistence */ AspectPersistenceMa • in production context all protected $FLOW3_Per sistence_Identifier = NULL; code is static private $FLOW3_AOP_P roxy_targetMethodsAn dGroupedAdvices = ar ra private $FLOW3_AOP_P roxy_groupedAdviceCh ains = array(); private $FLOW3_AOP_P roxy_methodIsInAdvic eMode = array(); /** * Autogenerated Prox y Method */ public function __co nstruct()
  38. 38. Basic Object Persistence // Create a new customer and persist it: $customer = new Customer("Robert"); $this->customerRepository->add($customer); // Find an existing customer: $otherCustomer = $this->customerRepository->findByFirstName("Karsten"); // and delete it: $this->customerRepository->remove($otherCustomer);
  39. 39. Validation and Doctrine Annotationsnamespace TYPO3BlogDomainModel;/** * A Blog object * * @Entity */class Blog { /** * @var string * @validate Text, StringLength(minimum = 1, maximum = 80) * @Column(length="80") */ protected $title; /** * @var DoctrineCommonCollectionsCollection<TYPO3BlogDomainModelPost> * @OneToMany(mappedBy="blog") * @OrderBy({"date" = "DESC"}) */ protected $posts; ...}
  40. 40. Persistence-related Annotations@Entity Declares a class as "entity"@Column Controls the database column related to the class property. Very useful for longer text content (type="text" !)@ManyToOne Defines relations to other entities. Unlike with@OneToMany vanilla Doctrine targetEntity does not have to be@ManyToMany given but will be reused from the @var@OneToOne annotation. cascade can be used to cascade operation to related objects.
  41. 41. Persistence-related Annotations@var Defines the type of a property, collections can be typed using angle brackets: DoctrineCommonCollectionsCollection<TYPO3ConferenceDomainModelComment>@transient The property will be ignored, it will neither be persisted nor reconstituted@identity Marks the property as part of an objects identity
  42. 42. Custom Queries using the Query Object Model/** * A PostRepository */class PostRepository extends TYPO3FLOW3PersistenceRepository { /** * Finds posts by the specified tag and blog * * @param TYPO3BlogDomainModelTag $tag * @param TYPO3BlogDomainModelBlog $blog The blog the post must refer to * @return TYPO3FLOW3PersistenceQueryResultInterface The posts */ public function findByTagAndBlog(TYPO3BlogDomainModelTag $tag, TYPO3BlogDomainModelBlog $blog) { $query = $this->createQuery(); return $query->matching( $query->logicalAnd( $query->equals(blog, $blog), $query->contains(tags, $tag) ) ) ->setOrderings(array( date => TYPO3FLOW3PersistenceQueryInterface::ORDER_DESCENDING) ) ->execute(); }}
  43. 43. Schema ManagementRunning Migrations• needed after installation or upgrade:$ ./flow3 doctrine:migrate
  44. 44. Schema ManagementManual database updates• for simple situations this can be good enough:$ ./flow3 doctrine:create$ ./flow3 doctrine:update• useful when • you are just starting a project and have never released
  45. 45. Schema ManagementGenerating migrations$ ./flow3 doctrine:migrationgenerateGenerated new migration class to "…/Version20110608074324.php"from schema differences.$• Generated migrations can contain errors and should be checked and adjusted as needed• Migrations need to be moved to their “owning” package manually
  46. 46. ValidationValidation in FLOW3• you do not want to code checks into your controllers• FLOW3 separates validation from your controller’s concerns • no PHP code needed for validation • declared through annotations
  47. 47. ValidationValidation Models• BaseProperties rules defining the minimum requirements on individual properties of a model• BaseModel rules or custom validators enforcing the minimum requirements on the combination of properties of a model• Supplemental rules defining additional requirements on a model for a specific situation (e.g. a certain action method)
  48. 48. ValidationBase Properties• Validation rules defined directly at the properties /** * @var string * @validate StringLength(minimum = 10, maximum = 100) */ protected $title; /** * @var string * @validate StringLength(minimum = 1, maximum = 50) */ protected $author;
  49. 49. ValidationValidators• validators provided by FLOW3 can be used through their short name • Count, Float, NotEmpty, RegularExpression, Uuid, DateTime, NumberRange, StringLength, Alphanumeric, Integer, Number, String, EmailAddress, Label, Raw, Text• custom validators need to implement the ValidatorInterface• use them by specifying the fully qualified class name /** * @var DambekalnsStuffDomainModelStuff * @validate DambekalnsStuffDomainValidatorStuffValidator */ protected $stuff;
  50. 50. Property Mapper $articleArray = array( headline => Hello World!, story => Just a demo ... ); $article = $mapper->convert($sourceArray, Acme.DemoDomainModelArticle);
  51. 51. Resource Management Image Upload Resources are handled like other properties in a form: <f:form method="blog" action="update" object="{blog}" name="blog"enctype="multipart/form-data"> <f:if condition="{blog.authorPicture}"> <img src="{f:uri.resource(resource: blog.authorPicture)}" /> </f:if> <label for="authorPicture">Author picture</label> <f:form.upload property="authorPicture" id="authorPicture" /> <f:form.submit value="Update"/> </f:form>
  52. 52. Property Mapper Allow nested object structures For security reasons the creation of nested structure through the property mapper is disabled by default /** * @return void */ public function initializeUpdateAction() { $this->arguments[article]->getPropertyMappingConfiguration() ->allowCreationForSubProperty(picture); $this->arguments[article]->getPropertyMappingConfiguration() ->allowModificationForSubProperty(picture); }
  53. 53. FluidExample for assigning a string to a Fluid variable: // in the action controller: $this->view->assign(title, Welcome to Fluid); <!-- in the Fluid template: --> <head> <title>{title}</title> </head>
  54. 54. FluidVariables can also be objects: // in the action controller: $this->view->assign(conference, $conference); <!-- in the Fluid template: --> <div class="venue"> <p>Venue Street: {conference.venue.street}</p> </div>
  55. 55. Fluidif-then-else: // in the action controller: $this->view->assign(post, $blogPost); <!-- in the Fluid template: --> <f:if condition="{post.comments}"> <f:then>There are some comments.</f:then> <f:else>There are no comments.</f:else> </f:if>
  56. 56. Fluidfor-each: // in the action controller: $this->view->assign(ages, array("Karsten" => 34, "Robert" => 35)); <!-- in the Fluid template: --> <ul> <f:for each="{ages}" as="age" key="name"> <li>{name} is {age} year old.</li> </f:for> </ul>
  57. 57. Fluidfor-each: // in the action controller: $this->view->assign(post, $blogPost); <!-- in the Fluid template: --> <f:if condition="{post.comments}"> <ul> <f:for each="{post.comments}" as="comment" > <li>{post.title}</li> </f:for> </ul> </f:if>
  58. 58. FluidView helpers – in this case the link.action view helper: <!-- in the Fluid template: --> {namespace f=TYPO3FluidViewHelpers} <f:link.action action="delete" arguments="{post: post, really: yes}"> Delete this post </f:link.action>
  59. 59. Security Policy
  60. 60. SecurityCross-Site Request Forgery • enables an attacker to execute privileged operations without being authenticated • the risk lies in using malicious links or forms while still being authenticated • imagine a link coming in through an URL shortener...
  61. 61. SecurityAvoiding Cross-Site Request Forgery • add a (truly!) random string token to each link or form • make sure this token is correct before executing anything • change the token as often as possible to make it impossible to send you a working malicious link while you’re logged in • in most cases, we can assume that it should be enough to generate one token when you log in – that’s the default
  62. 62. SecurityCSRF Protection in FLOW3 • you must not forget to add that token to any link • FLOW3 automatically adds the CSRF token to each • link you generate • each form you create with Fluid • and checks it for every call to a protected action • the protection can be disabled using @skipCsrfProtection on an action
  63. 63. Roadmaphttp://forge.typo3.org/projects/flow3-distribution-base/roadmap
  64. 64. Conference Appgit://git.typo3.org/TYPO3v5/Distributions/Conference.git
  65. 65. Blog Appgit://git.typo3.org/FLOW3/Applications/Blog.git
  66. 66. Rossmann• second biggest drug store in Germany• 5,13 billion € turnover• 31,000 employees
  67. 67. Customer Database• custom persistence with CouchDB• SOAP support• continuous delivery• cluster setup by networkteam, Kiel
  68. 68. Amadeus• world’s biggest e-ticket provider• 217 markets• 948 million billable transactions / year• 2,7 billion € revenue
  69. 69. Social Media Suite• central hub for social media activities for potentially thousands of travel agencies• advanced form engine• various detail improvements by AKOM360, Munich• uses an early version of TYPO3 Phoenix
  70. 70. “Our senior developers areextremely happy with FLOW3 –it is definitely the mostcapable PHP framework we Fabian Pfütze Project Leadhave come acrossso far.”
  71. 71. ?
  72. 72. Thanks for having me!Slides: http://slideshare.net/robertlemkeExamples: http://github.com/robertlemkeBlog: http://robertlemke.comTwitter: @robertlemkeFeedback: robert@typo3.orgFLOW3: http://flow3.typo3.org

×