Symfony 3
HTTP Fundamentals
HTTP 1.1
Request / Response
Request
GET / HTTP/1.1
Host: xkcd.com
Accept: text/html
User-Agent: Mozilla/5.0 (Macintosh)
Request
GET | Retrieve the resource from the server |
POST | Create a resource on the server |
PUT | Update the resource on the server |
DELETE | Delete the resource from the server |
GET /blog/15 HTTP/1.1
DELETE /blog/15 HTTP/1.1
POST /blog HTTP/1.1
Response
HTTP/1.1 200 OK
Date: Sat, 02 Apr 2011 21:05:05 GMT
Server: lighttpd/1.4.19
Content-Type: text/html
<html>
<!-- ... HTML for the xkcd comic -->
</html>
Req/Resp in PHP
$uri = $_SERVER['REQUEST_URI'];
$foo = $_GET['foo'];
header('Content-Type: text/html');
echo 'The URI requested is: '.$uri;
echo 'The value of the "foo" parameter is: '.$foo;
Req/Resp in PHP
HTTP/1.1 200 OK
Date: Sat, 03 Apr 2011 02:14:33 GMT
Server: Apache/2.2.17 (Unix)
Content-Type: text/html
The URI requested is: /testing?foo=symfony
The value of the "foo" parameter is: symfony
Req/Resp in Symfony
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
$request = Request::createFromGlobals();
$uri = $request->getPathInfo();
$foo = $request->get('foo');
$response = new Response();
$body = 'The URI requested is: '.$uri;
$body .= 'The value of the "foo" parameter is: '.$foo;
$response->setContent($body);
$response->setStatusCode(Response::HTTP_OK);
$response->headers->set('Content-Type', 'text/html');
$response->send();
Application Flow
Sample Controller
namespace AppBundle\Controller;
use Symfony\Component\HttpFoundation\Response;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
class SampleController
{
/**
* @Route("/sample")
*/
public function indexAction()
{
return new Response('<h1>Some beautiful message!</h1>');
}
}
Creating the Symfony Application
Creating App
$ symfony new my_project_name
Symfony Installer
$ composer create-project symfony/framework-standard-edition my_project_name
Composer
Running
$ cd my_project_name/
$ php bin/console server:run
localhost:8000
Controller
namespace AppBundle\Controller;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
use Symfony\Component\HttpFoundation\Response;
class LuckyController
{
/**
* @Route("/lucky/number")
*/
public function numberAction()
{
$number = rand(0, 100);
return new Response('Lucky number: '.$number);
}
}
localhost:8000/lucky/number
Playground
Twig
Template Engine
Syntax
Prints a variable or the result of an expression to the template.
{{ ... }}
A tag that controls the logic of the template; it is used to execute statements such as for-loops for example.
{% ... %}
{# ... #}
It's the equivalent of the PHP /* comment */ syntax.
Loop
<ul>
{% for item in navigation %}
<li><a href="{{ item.href }}">{{ item.caption }}</a></li>
{% endfor %}
</ul>
<ul>
{% for i in 0..10 %}
<li>Number {{ i }}</li>
{% endfor %}
</ul>
<ul>
{% for user in users if user.active %}
<li>{{ user.username }}</li>
{% else %}
<li>No users found</li>
{% endfor %}
</ul>
Conditional
{% if app.user %}
<div>
Hello {{ app.user }}
</div>
{% else %}
<div>
Please sign in
</div>
{% endif %}
Inheritance
{# app/Resources/views/base.html.twig #}
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Template inheritance</title>
</head>
<body>
<div id="content">
{% block body %}
No body defined!!!
{% endblock %}
</div>
</body>
</html>
Inheritance
{# app/Resources/views/blog/index.html.twig #}
{% extends 'base.html.twig' %}
{% block body %}
{% for entry in blog_entries %}
<h2>{{ entry.title }}</h2>
<p>{{ entry.body }}</p>
{% endfor %}
{% endblock %}
Filters
{{ myVar | upper }}
{{ myDate | date("m/d/Y") }}
{{ myObj | json_encode }}
{{ myArray | length }}
{{ myVar | raw }}
{{ numbers | reverse | join(',') }}
Controller template
/**
* @Route("/lucky/number/{count}")
*/
public function numberAction($count)
{
$number = rand(0, 100);
return $this->render('lucky/number.html.twig', [
'luckyNumber' => $number
]);
}
Controller template
{# app/Resources/views/lucky/number.html.twig #}
{% extends 'base.html.twig' %}
{% block body %}
<h1>Lucky Number: {{ luckyNumber }}</h1>
{% endblock %}
Playground
Routing
Name
/**
* @Route("/", name="blog_homepage")
*/
public function indexAction()
{
// ...
}
Placeholder
/**
* @Route("/blog/{slug}")
*/
public function showAction($slug)
{
// ...
}
Defaults
/**
* @Route("/blog/{page}", defaults={"page": 1})
*/
public function indexAction($page)
{
// ...
}
/**
* @Route("/blog/{page}")
*/
public function indexAction($page = 1)
{
// ...
}
Requirements
/**
* @Route("/blog/{page}", defaults={"page": 1}, requirements={
* "page": "\d+"
* })
*/
public function indexAction($page)
{
// ...
}
/**
* @Route("/{_locale}", defaults={"_locale": "en"}, requirements={
* "_locale": "en|fr"
* })
*/
public function homepageAction($_locale)
{
// ...
}
HTTP Method
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Method;
/**
* @Route("/api/posts/{id}")
* @Method({"GET","HEAD"})
*/
public function showAction($id)
{
// ...
}
/**
* @Route("/api/posts/{id}")
* @Method("PUT")
*/
public function editAction($id)
{
// ...
}
Advanced
/**
* @Route(
* "/articles/{_locale}/{year}/{title}.{_format}",
* defaults={"_format": "html"},
* requirements={
* "_locale": "en|fr",
* "_format": "html|rss",
* "year": "\d+"
* }
* )
* @Method("GET")
*/
public function showAction($_locale, $year, $title)
{
// ...
}
Special Parameters
As you've seen, this parameter is used to determine which controller is executed when the route is matched.
_format
_locale
_controller
Used to set the request format.
Used to set the locale on the request.
Debugging
$ bin/console debug:router
homepage ANY /
contact GET /contact
contact_process POST /contact
article_show ANY /articles/{_locale}/{year}/{title}.{_format}
blog ANY /blog/{page}
blog_show ANY /blog/{slug}
Playground
Controller
Controller
The goal of a controller is always the same: create and return a Response object
use Symfony\Component\HttpFoundation\Response;
public function helloAction()
{
return new Response('Hello world!');
}
Responses
public function helloAction()
{
return $this->render('hello.html.twig');
}
use Symfony/Component/HttpFoundation/JsonResponse;
public function helloAction()
{
return new JsonResponse([
'hello' => 'Hello human!!!'
]);
}
public function helloAction()
{
return new Response('<h1>Response content</h1>');
}
Responses
public function helloAction()
{
return $this->redirectToRoute('another_route_name');
}
public function helloAction()
{
return $this->forward('another_route_name');
}
public function helloAction()
{
return $this->redirect('http://symfony.com/doc');
}
Request Parameter
use Symfony\Component\HttpFoundation\Request;
public function indexAction(Request $request)
{
$request->isXmlHttpRequest(); // is it an Ajax request?
// request parameter
$request->get('page');
// retrieves an instance of UploadedFile identified by foo
$request->files->get('foo');
// retrieve an HTTP request header, with normalized, lowercase keys
$request->headers->get('host');
$request->headers->get('content_type');
}
Request Parameter
use Symfony\Component\HttpFoundation\Request;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
/**
* @Route('/{name}')
*
*/
public function indexAction(Request $request, $name)
{
return $this->render('index.html.twig', [ 'name' => $name ]);
}
Session
use Symfony\Component\HttpFoundation\Request;
public function indexAction(Request $request)
{
$session = $request->getSession();
// store an attribute for reuse during a later user request
$session->set('foo', 'bar');
// get the attribute set by another controller in another request
$foobar = $session->get('foobar');
// use a default value if the attribute doesn't exist
$filters = $session->get('filters', array());
}
Flash Messages
$this->addFlash(
'notice',
'Your changes were saved!'
);
{% for message in app.session.flashBag.get('notice') %}
<div class="alert alert-info">
{{ message }}
</div>
{% endfor %}
Playground
Database
& Doctrine
Connection info
# app/config/parameters.yml
parameters:
database_host: localhost
database_name: test_project
database_user: root
database_password: password
Generate Entity
$ bin/console doctrine:generate:entity
Entity Mapping
// src/AppBundle/Entity/Product.php
namespace AppBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Entity
* @ORM\Table(name="product")
*/
class Product
{
/**
* @ORM\Column(type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @ORM\Column(type="string", length=100)
*/
private $name;
}
Creating Tables
$ bin/console doctrine:schema:update --dump-sql
$ bin/console doctrine:schema:update --force
Fetching
// src/AppBundle/Controller/DefaultController.php
use AppBundle\Entity\Product;
public function showAction($productId)
{
$product = $this->getDoctrine()
->getRepository(Product::class)
->find($productId);
if (!$product) {
throw $this->createNotFoundException(
'No product found for id '.$productId
);
}
// ... do something, like pass the $product object into a template
}
Persisting
// src/AppBundle/Controller/DefaultController.php
use AppBundle\Entity\Product;
use Symfony\Component\HttpFoundation\Response;
public function createAction()
{
$product = new Product();
$product->setName('Keyboard');
$product->setPrice(19.99);
$product->setDescription('Ergonomic and stylish!');
$em = $this->getDoctrine()->getManager();
// tells Doctrine you want to (eventually) save the Product (no queries yet)
$em->persist($product);
// actually executes the queries (i.e. the INSERT query)
$em->flush();
return new Response('Saved new product with id '.$product->getId());
}
Updating
// src/AppBundle/Controller/DefaultController.php
use AppBundle\Entity\Product;
public function updateAction($productId)
{
$em = $this->getDoctrine()->getManager();
$product = $em->getRepository(Product::class)->find($productId);
if (!$product) {
throw $this->createNotFoundException(
'No product found for id '.$productId
);
}
$product->setName('New product name!');
$em->merge($product);
$em->flush();
return $this->redirectToRoute('homepage');
}
Deleting
// src/AppBundle/Controller/DefaultController.php
use AppBundle\Entity\Product;
public function deleteAction($productId)
{
$em = $this->getDoctrine()->getManager();
$product = $em->getRepository(Product::class)->find($productId);
if (!$product) {
throw $this->createNotFoundException(
'No product found for id '.$productId
);
}
$em->remove($product);
$em->flush();
return $this->redirectToRoute('homepage');
}
Querying
// src/AppBundle/Controller/DefaultController.php
use AppBundle\Entity\Product;
public function searchAction()
{
$em = $this->getDoctrine()->getManager();
$query = $em->createQuery(
'SELECT p
FROM AppBundle:Product p
WHERE p.price > :price
ORDER BY p.price ASC'
)->setParameter('price', 19.99);
$products = $query->getResult();
// ...
}
Querying
// src/AppBundle/Controller/DefaultController.php
use AppBundle\Entity\Product;
public function searchAction()
{
$em = $this->getDoctrine()->getManager();
$query = $em->createQueryBuilder()
->select('p')
->from(Product::class, 'p')
->where('p.price > :price')
->orderBy('p.price', 'ASC')
->setParameter('price', 19.99)
->getQuery();
$products = $query->getResult();
// ...
}
Relationship
// src/AppBundle/Entity/Product.php
// ...
class Product
{
// ...
/**
* @ORM\ManyToOne(targetEntity="Category", inversedBy="products")
* @ORM\JoinColumn(name="category_id", referencedColumnName="id")
*/
private $category;
}
Relationship
// src/AppBundle/Entity/Category.php
// ...
use Doctrine\Common\Collections\ArrayCollection;
class Category
{
// ...
/**
* @ORM\OneToMany(targetEntity="Product", mappedBy="category")
*/
private $products;
public function __construct()
{
$this->products = new ArrayCollection();
}
}
Playground
Forms
Form Builder
// controller class
$task = new Task();
$form = $this->createFormBuilder($task)
->add('task', TextType::class)
->add('dueDate', DateType::class)
->add('save', SubmitType::class, ['label' => 'Create Task'])
->getForm();
Rendering
{# form page #}
{{ form_start(form) }}
{{ form_widget(form) }}
{{ form_end(form) }}
Handling Form
// controller class
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid()) {
// ... perform some action, such as saving the task to the database
return $this->redirectToRoute('task_success');
}
Field Types
- TextType
- TextareaType
- EmailType
- IntegerType
- MoneyType
- NumberType
- PasswordType
- PercentType
- SearchType
- UrlType
- RangeType
Text fields
Field Types
- ChoiceType
- EntityType
- CountryType
- LanguageType
- LocaleType
- TimezoneType
- CurrencyType
Choice fields
Field Types
- DateType
- DateTimeType
- TimeType
- BirthdayType
Date and time fields
Custom render
{{ form_start(form) }}
{{ form_errors(form) }}
{{ form_row(form.task) }}
{{ form_row(form.dueDate) }}
{{ form_end(form) }}
Custom render
{{ form_start(form) }}
{{ form_errors(form) }}
<div>
{{ form_label(form.task) }}
{{ form_errors(form.task) }}
{{ form_widget(form.task) }}
</div>
<div>
{{ form_label(form.dueDate) }}
{{ form_errors(form.dueDate) }}
{{ form_widget(form.dueDate) }}
</div>
<div>
{{ form_widget(form.save) }}
</div>
{{ form_end(form) }}
Form class
// src/AppBundle/Form/TaskType.php
namespace AppBundle\Form;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\Form\Extension\Core\Type\SubmitType;
class TaskType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
->add('task')
->add('dueDate', null, array('widget' => 'single_text'))
->add('save', SubmitType::class)
;
}
}
Form class
// controller
use AppBundle\Form\TaskType;
public function newAction()
{
$task = ...;
$form = $this->createForm(TaskType::class, $task);
// ...
}
Form generator
$ bin/console doctrine:generate:form AppBundle:Product
Bootstrap theme
# app/config/config.yml
twig:
form_themes: ['bootstrap_3_layout.html.twig']
Validation
// src/AppBundle/Entity/Task.php
namespace AppBundle\Entity;
use Symfony\Component\Validator\Constraints as Assert;
class Task
{
/**
* @Assert\NotBlank()
*/
public $task;
/**
* @Assert\NotBlank()
* @Assert\Type("\DateTime")
*/
protected $dueDate;
}
Validation
// src/AppBundle/Form/TaskForm.php
use Symfony\Component\Validator\Constraints as Assert;
$builder
->add('task', null, [
'constraints' => [
new Assert\NotBlank(),
new Assert\Length([ 'min' => 3 ]),
],
])
->add('lastName', null, [
'constraints' => [
new Assert\NotBlank(),
new Assert\Type("\DateTime")
),
])
;
File upload
// src/AppBundle/Entity/Product.php
namespace AppBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Validator\Constraints as Assert;
class Product
{
// ...
/**
* @ORM\Column(type="string")
*
* @Assert\NotBlank(message="Please, upload the product brochure as a PDF file.")
* @Assert\File(mimeTypes={ "application/pdf" })
*/
private $brochure;
}
File upload
class ProductType extends AbstractType
{
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder
// ...
->add('brochure', FileType::class)
// ...
;
}
public function configureOptions(OptionsResolver $resolver)
{
$resolver->setDefaults(array(
'data_class' => Product::class,
));
}
}
File upload
$form = $this->createForm(ProductType::class, $product);
$form->handleRequest($request);
if ($form->isSubmitted() && $form->isValid()) {
/** @var Symfony\Component\HttpFoundation\File\UploadedFile $file */
$file = $product->getBrochure();
// Generate a unique name for the file before saving it
$fileName = md5(uniqid()).'.'.$file->guessExtension();
// Move the file to the directory where brochures are stored
$file->move('/upload_directory', $fileName);
// Update the 'brochure' property to store the PDF file name
// instead of its contents
$product->setBrochure($fileName)
}
Playground
Security
Authentication
# app/config/security.yml
security:
providers:
in_memory:
memory: ~
firewalls:
dev:
pattern: ^/(_(profiler|wdt)|css|images|js)/
security: false
default:
anonymous: ~
Basic authentication
# app/config/security.yml
security:
# ...
firewalls:
# ...
default:
anonymous: false
http_basic: ~
User provider
# app/config/security.yml
security:
providers:
in_memory:
memory:
users:
admin:
password: kitten
roles: 'ROLE_ADMIN'
rogerio:
password: 123456
roles: 'ROLE_USER'
# ...
No encoder has been configured for account "Symfony\Component\Security\Core\User\User"
User encoder
# app/config/security.yml
security:
# ...
encoders:
Symfony\Component\Security\Core\User\User: plaintext
# ...
User encoder
# app/config/security.yml
security:
# ...
encoders:
Symfony\Component\Security\Core\User\User:
algorithm: bcrypt
cost: 12
$ php bin/console security:encode-password
Changing password
// whatever *your* User object is
$user = new AppBundle\Entity\User();
$plainPassword = 'ryanpass';
$encoder = $this->container->get('security.password_encoder');
$encoded = $encoder->encodePassword($user, $plainPassword);
$user->setPassword($encoded);
Access control
# app/config/security.yml
security:
# ...
access_control:
- { path: ^/admin, roles: ROLE_ADMIN }
- { path: ^/admin/users, roles: ROLE_SUPER_ADMIN }
Securing controllers
// controller class
public function helloAction($name)
{
// The second parameter is used to specify on what object the role is tested.
$this->denyAccessUnlessGranted(
'ROLE_ADMIN',
null,
'Unable to access this page!'
);
// ...
}
Securing controllers
// controller class
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Security;
/**
* @Security("has_role('ROLE_ADMIN')")
*/
public function helloAction($name)
{
// ...
}
Securing controllers
// controller class
public function helloAction($name)
{
$checker = $this->get('security.authorization_checker');
if (!$checker->isGranted('IS_AUTHENTICATED_FULLY')) {
throw $this->createAccessDeniedException();
}
// ...
}
Twig access control
{# twig page #}
{% if is_granted('ROLE_ADMIN') %}
<a href="...">Delete</a>
{% endif %}
Special attributes
- IS_AUTHENTICATED_REMEMBERED: All logged in users have this, even if they are logged in because of a "remember me cookie".
- IS_AUTHENTICATED_FULLY: Users who are logged in.
- IS_AUTHENTICATED_ANONYMOUSLY: All users (even anonymous ones) have this
Retrieving user
// controller class
public function indexAction()
{
// via token storage
$user = $this->get('security.token_storage')->getToken()->getUser();
// shortcut
$user = $this->getUser();
}
Retrieving user
{# twig page #}
{% if is_granted('IS_AUTHENTICATED_FULLY') %}
<p>Username: {{ app.user.username }}</p>
{% endif %}
Hierarchical Roles
# app/config/security.yml
security:
# ...
role_hierarchy:
ROLE_ADMIN: ROLE_USER
ROLE_SUPER_ADMIN: [ROLE_ADMIN, ROLE_ALLOWED_TO_SWITCH]
Login form
# app/config/security.yml
security:
# ...
firewalls:
main:
anonymous: ~
form_login:
login_path: /login
check_path: /login_check
Security Controller
// src/AppBundle/Controller/SecurityController.php
/**
* @Route("/login", name="login")
*/
public function loginAction(Request $request)
{
$authenticationUtils = $this->get('security.authentication_utils');
// get the login error if there is one
$error = $authenticationUtils->getLastAuthenticationError();
// last username entered by the user
$lastUsername = $authenticationUtils->getLastUsername();
return $this->render('security/login.html.twig', [
// last username entered by the user
'last_username' => $lastUsername,
'error' => $error,
]);
}
Template
{# app/Resources/views/security/login.html.twig #}
{% if error %}
<div>{{ error.messageKey|trans(error.messageData, 'security') }}</div>
{% endif %}
<form action="{{ path('login_check') }}" method="post">
<label for="username">Username:</label>
<input type="text" id="username" name="_username"
value="{{ last_username }}" />
<label for="password">Password:</label>
<input type="password" id="password" name="_password" />
<button type="submit">login</button>
</form>
Update security.yml
# app/config/security.yml
security:
# ...
firewalls:
# order matters! This must be before the ^/ firewall
login_firewall:
pattern: ^/login$
anonymous: ~
main:
pattern: ^/
form_login:
login_path: /login
check_path: /login_check
access_control:
- { path: ^/login, roles: IS_AUTHENTICATED_ANONYMOUSLY }
- { path: ^/, roles: ROLE_ADMIN }
Logging out
# app/config/security.yml
security:
# ...
firewalls:
main:
# ...
logout:
path: /logout
target: /
Update routing.yml
# app/config/routing.yml
# ..
login_check:
path: /login_check
logout:
path: /logout
Custom User
// AppBundle\Entity\User;
use Symfony\Component\Security\Core\User\UserInterface;
class User implements UserInterface, \Serializable
{
// ...
public function serialize()
{
return serialize([
$this->id,
$this->username
]);
}
public function unserialize($serialized)
{
list(
$this->id,
$this->username
) = unserialize($serialized);
}
}
Update security.yml
security:
# ...
providers:
my_provider:
entity:
class: AppBundle\Entity\User
property: username
encoders:
AppBundle\Entity\User:
algorithm: bcrypt
cost: 12
firewalls:
# ...
main:
provider: my_provider
# ...
Playground
The end
Source: The Symfony Book
Symfony 3
By Rogério Alencar Lino Filho
Symfony 3
Material utilizado no treinamento
- 1,873