713 lines
		
	
	
	
		
			23 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			713 lines
		
	
	
	
		
			23 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| ---
 | ||
| title: Live Blogging From SymfonyLive London 2019
 | ||
| date: 2019-09-13
 | ||
| tags:
 | ||
|     - conference
 | ||
|     - php
 | ||
|     - symfony
 | ||
|     - symfonylive
 | ||
| ---
 | ||
| 
 | ||
| Inspired by [Matt Stauffer](https://twitter.com/stauffermatt)'s
 | ||
| [live blogging of the keynote](https://mattstauffer.com/blog/introducing-laravel-vapor)
 | ||
| at Laracon US, I’m going to do the same for the sessions that I’m attending at
 | ||
| [SymfonyLive London 2019](https://london2019.live.symfony.com)...
 | ||
| 
 | ||
| ## Keynote (Back to the basics)
 | ||
| 
 | ||
| **Embrace the Linux philosophy**
 | ||
| 
 | ||
| - How we grow the Symfony ecosystem. Built abstracts.
 | ||
| - HttpFoundation, HttpKernel
 | ||
| - Moved to infrastructure
 | ||
| - A few abstractions on top of PHP. Improved versions of PHP functions (`dump`
 | ||
|   and `var_dump`)
 | ||
| - Started a add higher level abstractions (e.g. Mailer), built on the lower
 | ||
|   ones.
 | ||
| - Recently worked on PHPUnit assertions. Mailer in Symony 4.4. Can test if an
 | ||
|   email is sent or queued
 | ||
| 
 | ||
| **Building flexible high-level abstractions on top of low-level ones**
 | ||
| 
 | ||
| ### What's next?
 | ||
| 
 | ||
| - Mailer announced in London last year. New component.
 | ||
| - System emails? e.g. new customer, new invoice.
 | ||
| - Symfony Mailer = Built-in responsive, flexible, and generic system emails
 | ||
|   - Twig with TwigExtraBundle
 | ||
|   - Twig `inky-extra` package (Twig 1.12+)
 | ||
|   - Zurb Foundation for Emails CSS stylesheet
 | ||
|   - Twig `cssinliner-extra` package (Twig 1.12+)
 | ||
|   - Optimised Twig layouts
 | ||
| - `SystemEmail` class extends templated email
 | ||
| - Can set importance,
 | ||
| - Customisable
 | ||
| - Always trying to keep flexible, so things can be overidden and customised
 | ||
| 
 | ||
| ### Sending SMS messages
 | ||
| 
 | ||
| - new `Texter` and `SmsMessage` class for sending SMS messages
 | ||
| - Same abstraction as emails, but for SMS messages
 | ||
| - Based on HttpClient + Symfony Messenger and third-party providers (Twilio and
 | ||
|   Nexmo) `twilio://` and `nemxo://`
 | ||
| - Can set via transport `$sms->setTransport('nexmo')`
 | ||
| - Extend the `SystemEmail` and do what you want
 | ||
| - Failover
 | ||
| 
 | ||
| ### Sending Messages
 | ||
| 
 | ||
| - Create `ChatMessage`
 | ||
| - Telegram and Slack
 | ||
| - `$message->setTransport('telegram')`, `$bus->dispatch($message)`
 | ||
| - Send to Slack **and** Telegram
 | ||
| - `SlackOptions` and `TelegramOptions` for adding emojis etc
 | ||
| - Common transport layer `TransportInterface`, `MessageInterface`
 | ||
| - Failover - e.g. if Twilio is down, send to Telegram
 | ||
| 
 | ||
| ### New component - SymfonyNotifier
 | ||
| 
 | ||
| - Channels - email, SMS, chat
 | ||
| - Transport, slack, telegram, twilio
 | ||
| - Create a notification, arguments are message and transports (array)
 | ||
| - Receiver
 | ||
| - Customise notifications, `InvoiceNotification` extends `Notification`.
 | ||
|   `getChannels`
 | ||
|   - Override default rendering
 | ||
|   - `ChatNotificationInterface` - `asChatMessage()`
 | ||
| - Semantic configuration
 | ||
|   - `composer req twilio-notifier telegram-notifier`
 | ||
| - Channels
 | ||
|   - Mailer
 | ||
|   - Chatter
 | ||
|   - Texter
 | ||
|   - Browser
 | ||
|   - Pusher (iOS, Android, Desktop native notifications)
 | ||
|   - Database (web notification centre)
 | ||
|   - **A unified way to notify Users via a unified Transport layer**
 | ||
| - Each integration is only 40 lines of code
 | ||
| 
 | ||
| ### What about a SystemNotification?
 | ||
| 
 | ||
| - Autoconfigured channels
 | ||
| - `new SystemNotification`, `Notifier::getSystemReceivers`
 | ||
| - Importance, automatically configures channels
 | ||
| - Different channels based on importance
 | ||
| - `ExceptionNotification` - get email with stack trace attached
 | ||
| 
 | ||
| Notifier
 | ||
| 
 | ||
| - send messages via a unified api
 | ||
| - send to one or many receivers
 | ||
| - Default configu or custom one
 | ||
| 
 | ||
| ### How can we leverage this new infrastructure?
 | ||
| 
 | ||
| - `Monolog NotifierHandler` - triggered on `Error` level logs
 | ||
| - Uses notified channel configuration
 | ||
| - Converts Error level logs to importance levels
 | ||
| - Configurablelike other Notifications
 | ||
| - 40 lines of code
 | ||
| - Failed Messages Listener - 10 lines of glue code
 | ||
| 
 | ||
| - **Experimental component in 5.0**
 | ||
| - Can't in in 4.4 as it's a LTS version
 | ||
| - First time an experimental component is added
 | ||
| - Stable in 5.1
 | ||
| 
 | ||
| ## Queues, busses and the Messenger component (Tobias Nyholm)
 | ||
| 
 | ||
| - Stack is top and buttom - Last-in, first-out
 | ||
| - Queue is back and front - last in, first out
 | ||
| 
 | ||
| ### 2013
 | ||
| 
 | ||
| - Using Symfony, used 40 or 50 bundles in a project - too much information!
 | ||
| - Used to copy and paste, duplicate a lot of code
 | ||
| - Testing your controllers - controllers as services?
 | ||
| - Controllers are 'comfortable'
 | ||
| - Tried adding `CurrentUserProvider` service to core, should be passed as an
 | ||
|   argument. Cannot test.
 | ||
| - 'Having Symfony all over the place wasn't the best thing' - when to framework
 | ||
|   (Matthias Noback)
 | ||
|   - Hexagonal architecture
 | ||
|   - Keep your kernel away from infrastructure. Let the framework handle the
 | ||
|     infrastructure.
 | ||
| - Controller -> Command -> Command Bus -> `CommandHandler`
 | ||
| 
 | ||
| #### What did we win?
 | ||
| 
 | ||
| - Can leverage Middleware with a command bus
 | ||
| - Queues as a service (RabbitMQ)
 | ||
| - Work queue - one producer, multiple consumers
 | ||
| - Queues should be durable - messages are also stored on disk, consumers should
 | ||
|   acknowledge a message once a message is handled
 | ||
| - Publish/subscribe
 | ||
|   - Producer -> Fanout/direct with routing (multiple queues) -> multiple
 | ||
|     consumers
 | ||
| - Topics - wildcards
 | ||
| 
 | ||
| ### 2016
 | ||
| 
 | ||
| - New intern. Understand everything, 'just PHP'. Plain PHP application, not
 | ||
|   'scary Symfony'
 | ||
| 
 | ||
| ### Symfony Messenger
 | ||
| 
 | ||
| - `composer req symfony/messager` - best MessageBus implementation
 | ||
| - Message -> Message bus -> Message handler
 | ||
| - Message is a plain PHP class
 | ||
| - Handler is a normal PHP class which is invokable
 | ||
| - `messenger:message_hander` tag in config
 | ||
| - Autowire with `MessageHandlerInterface`
 | ||
| - What if it takes 20 seconds to send a message? Use asynchronous.
 | ||
| - Transports as middleware (needs sender, receiver, configurable with DSN,
 | ||
|   encode/decode). `MESSENGER_DSN` added to `.env`
 | ||
| - Start consumer with `bin/console messager:consume-messages`. Time limit with
 | ||
|   `--time-limit 300`
 | ||
| - PHP Enqueue - production ready, battle-tested messaging solution for PHP
 | ||
| 
 | ||
| ### Issues
 | ||
| 
 | ||
| - Transformers, takes an object and transforms into an array -
 | ||
|   `FooTransformer implements TransformerInterface`.
 | ||
| - Don't break other apps by changing the payload.
 | ||
| 
 | ||
| #### Multiple buses
 | ||
| 
 | ||
| - Command bus, query bus, event bus
 | ||
| - Separate actions from reactions
 | ||
| 
 | ||
| #### Envelope
 | ||
| 
 | ||
| - Stamps for metadata - has the item been on the queue already?
 | ||
| 
 | ||
| #### Failures
 | ||
| 
 | ||
| - Requeue, different queue or same queue after a period of time
 | ||
| - Failed queue 1 every minute, failed queue 2 every hour - temporary glitches or
 | ||
|   a bug?
 | ||
| 
 | ||
| #### Creating entities
 | ||
| 
 | ||
| - What if two users registered at the same tiem? Use uuids rather than IDs.
 | ||
| - Symfony validation - can be used on messages, not just forms.
 | ||
| 
 | ||
| - Cache everything
 | ||
| 
 | ||
|   - Option 1: HTTP request -> Thin app (gets responses from Redis) -> POST to
 | ||
|     queue. Every GET request would warm cache
 | ||
|   - Option 2: HTTP request -> Thin app -> return 200 response -> pass to workers
 | ||
| 
 | ||
| - Tip: put Command and CommandHandlers in the same directory
 | ||
| 
 | ||
| ## HttpClient (Nicolas Grekas)
 | ||
| 
 | ||
| - new symfony component, released in may
 | ||
| - Httpclient contracts, separate package that contains interfaces
 | ||
|   - Symfony
 | ||
|   - PHP-FIG
 | ||
|   - Httplug
 | ||
| - `HttpClient::create()`. `$client->get()`
 | ||
| - JSON decoded with error handling
 | ||
| - Used on symfony.com website (#1391). Replaces Guzzle `Client` for
 | ||
|   `HttpClientInterface`
 | ||
| - Object is stateless, Guzzle is not. Doesn't handle cookies, cookies are state
 | ||
| - Remove boilerplate - use `toArray()`
 | ||
| - Options as third argument - array of headers, similar to Guzzle
 | ||
| 
 | ||
| ### What can we do with the Response?
 | ||
| 
 | ||
| - `getStatusCode(): int`
 | ||
| - `getHeaders(): array`
 | ||
| - `getContent(): string`
 | ||
| - `toArray(): array`
 | ||
| - `cancel(): void`
 | ||
| - `getInfo(): array` - metadata
 | ||
| - Everything is lazy!
 | ||
| - 80% of use-cases covered
 | ||
| 
 | ||
| ### What about PSR-18?
 | ||
| 
 | ||
| - Decorator/adapter to change to PSR compatible
 | ||
| - Same for Httplug
 | ||
| 
 | ||
| ### What about the remaining 20%?
 | ||
| 
 | ||
| - Options are part of the abstraction, not the implementation
 | ||
| 
 | ||
| #### Some of the options
 | ||
| 
 | ||
| - `timeout` - control inactivity periods
 | ||
| - `proxy` - get through a http proxy
 | ||
| - `on_progress` - display a progress bar / build a scoped client
 | ||
| - `base_url` - resolve relative URLS / build a scoped client
 | ||
| - `resolve` - protect webhooks against calls to internal endpoints
 | ||
| - `max_redirects` - disable or limit redirects
 | ||
| 
 | ||
| - Robust and failsafe by default
 | ||
| 
 | ||
| - Streamable uploads - `$mimeParts->toIterable()`.
 | ||
| - donwload a file
 | ||
| 
 | ||
|   ```php
 | ||
|   foreach ($client->stream($response) as $chunk) {
 | ||
|       // ...
 | ||
|   }
 | ||
|   ```
 | ||
| 
 | ||
| * Responses are lazy, requests are concurrent
 | ||
| * Asychronus requests. Reading in network order
 | ||
| 
 | ||
| ```
 | ||
| foreach ($client->stream($responses) as $response => $chunk) {
 | ||
|     if ($chunk->isLast()) {
 | ||
|         // a $response completed
 | ||
|     } else {
 | ||
|         // a $response's got network activity or timeout
 | ||
|     }
 | ||
| }
 | ||
| ```
 | ||
| 
 | ||
| - 379 request completed in 0.4s!
 | ||
| - `Stream` has second argument, max number of seconds to wait before yielding a
 | ||
|   timeout chunk
 | ||
| - `ResponseInterface::getInfo()` - get response headers, redirect count and URL,
 | ||
|   start time, HTTP method and code, user data and URL
 | ||
|   - `getInfo('debug')` - displays debug information
 | ||
| 
 | ||
| ### The components
 | ||
| 
 | ||
| - `NativeHttpClient` and `CurlHttpClient`
 | ||
|   - both provide
 | ||
|     - 100% contracts
 | ||
|     - secure directs
 | ||
|     - extended (time) info
 | ||
|     - transparent HTTP compression and (de)chunking
 | ||
|     - automatic HTTP proxy configuration via env vars
 | ||
| 
 | ||
| #### `NativeHttpClient`
 | ||
| 
 | ||
| - is most portable, works for everyone
 | ||
| - based on HTTP stream wrapper with fixed redirect logic
 | ||
| - blocking until response headers arrive
 | ||
| 
 | ||
| #### `CurlHttpClient`
 | ||
| 
 | ||
| - Requires ext-curl with fixed redirection logic
 | ||
| - Multiplexing response headers and bodies
 | ||
| - Leverages HTTP/2 and PUSH when available
 | ||
| - Keeps connections open also between synchronous requests, no DNS resolution so
 | ||
|   things are faster
 | ||
| 
 | ||
| #### Decorators
 | ||
| 
 | ||
| - ScopingHttpClient - auto-configure options based on request URL
 | ||
| - MockHttpClient - for testing, doesn't make actual HTTP requests
 | ||
| - CachingHttpClient - adds caching on a HTTP request
 | ||
| - Psr18Client
 | ||
| - HttplugClient
 | ||
| - TraceableHttpClient
 | ||
| 
 | ||
| ### Combining
 | ||
| 
 | ||
| #### FrameworkBundle/Autowiring
 | ||
| 
 | ||
| ```yml
 | ||
| framework:
 | ||
|   http_client:
 | ||
|     max_host_connections: 4
 | ||
|     deault_options:
 | ||
|       # ....
 | ||
|   scoped_client:
 | ||
|     # ...
 | ||
| ```
 | ||
| 
 | ||
| #### HttpBrowser
 | ||
| 
 | ||
| - HttpClient + DomCrawler + CssSelector + HttpKernel + BrowserKit
 | ||
| - RIP Goutte!
 | ||
| 
 | ||
| ### Coming in 4.4...
 | ||
| 
 | ||
| - `max_duration`
 | ||
| - `buffer` based on a callable
 | ||
| - `$chunk->isInformational()`
 | ||
| - `$response->toStream()`
 | ||
| - Async-compatible extensibility, when decoration is not enough
 | ||
| 
 | ||
| `composer req symfony/http-client`
 | ||
| 
 | ||
| ## Symfony Checker is coming (Valentine Boineau)
 | ||
| 
 | ||
| - Static analysis tool for Symfony
 | ||
|   - Does a method exist?
 | ||
|   - Is it deprecated?
 | ||
| - insight.symfony.com
 | ||
| - @symfonyinsight
 | ||
| - Released soon
 | ||
| 
 | ||
| ### Differences
 | ||
| 
 | ||
| - Specialise in Symfony - can see more relevant things
 | ||
| - Different interface to other services
 | ||
| 
 | ||
| ## Feeling unfulfilled by SPA promises? Go back to Twig (Dan Blows)
 | ||
| 
 | ||
| A way on the front-end JS, CSS, images at the beginning of the request, sends a
 | ||
| HTTP request (XHR/AJAX) to the back-end
 | ||
| 
 | ||
| ### Why SPAs?
 | ||
| 
 | ||
| - A way on the front-end JS, CSS, images at the beginning of the request, sends
 | ||
|   a HTTP request (XHR/AJAX) to the back-end
 | ||
| - no full page refresh
 | ||
| - Supposed to be much quicker
 | ||
| - 'Right tool for the job' - JS on the front-end, PHP on the back-end
 | ||
| - Division of responsibility == faster development
 | ||
| - Reusable API - Api -> Mobile App and SPA - easy to add another consumer
 | ||
| - Easier to debug?
 | ||
| 
 | ||
| ### Why not SPAs?
 | ||
| 
 | ||
| - Lots of HTTP requests (400 to load the initial page on one project) == slow
 | ||
|   front end
 | ||
| - Blurred responsibilities == tightly coupled teams
 | ||
| - harder to debug, bugs fall between systems and teams. Huge gap between
 | ||
|   front-end and back-end, passing responsibilites.
 | ||
| - You can fix these problems in SPAs, but is it worth it?
 | ||
|   - Examples of good SPAs - Trello, Flickr
 | ||
| 
 | ||
| ### Using Twig as an alternative to an SPA?
 | ||
| 
 | ||
| #### Faster UI - Try and figure out where the problem is.
 | ||
| 
 | ||
| If you're trying to speed things up, find out where the problem is.
 | ||
| 
 | ||
| - Browser tools
 | ||
| - Web Debug Toolbar
 | ||
| - Blackfire
 | ||
| - Optimise and monitor
 | ||
| 
 | ||
| #### Speed up Twig
 | ||
| 
 | ||
| - Speeding up Symfony
 | ||
| - ext/twig (PHP5 only, not PHP 7)
 | ||
| - Store compiled templates in Opcache, make sure it's enabled
 | ||
| - Render assets though the webserver (assetic not running all the time)
 | ||
| 
 | ||
| #### Edge side includes
 | ||
| 
 | ||
| - Component cached differently to the rest of the page
 | ||
| - Varnish/Nginx
 | ||
| - `render_esi`
 | ||
| - News block that caches frequently, rest of the page
 | ||
| 
 | ||
| #### HTTP/2 with Weblink
 | ||
| 
 | ||
| - slow finding CSS files to load - 'push' over CSS files, doesn't need to wait
 | ||
| - `preload()` - https://symfony.com/doc/current/web_link.html
 | ||
| 
 | ||
| #### Live updating pages
 | ||
| 
 | ||
| - Instantly update when sports results are updated, news articles are added
 | ||
| - Mercure - https://github.com/symfony/mercure
 | ||
| - LiveTwig - whole block or whole section, and live update `render_live`
 | ||
| - Turbolinks - replace whole body, keeps CSS and JS in memory. Merges new stuff
 | ||
|   in. `helthe/turbolinks`
 | ||
| - ReactPHP - shares kernel between requests
 | ||
| 
 | ||
| ### Writing better code with Twig
 | ||
| 
 | ||
| - Keep templates simple. Avoid spaghetti code, only about UI. HTML or small
 | ||
|   amounts of Twig.
 | ||
| - Avoid delimeter chains
 | ||
|   - Bad:`blog_post.authors.first.user_account.email_address`
 | ||
|   - Good `{{ blog_post.authors_email_address }}`
 | ||
|   - Less brittle, slow
 | ||
| 
 | ||
| * Filters
 | ||
|   - Use filters to be precise
 | ||
|   - Custom filters
 | ||
|   - Avoid chains. Can cause odd results. Create a new filter in PHP
 | ||
| * Functions
 | ||
|   - Write your own functions
 | ||
|   - Simpler templates
 | ||
|   - Get data, can use boolean statements
 | ||
| * Components
 | ||
|   - Break a page into components rather than one large page
 | ||
|   - `include()`
 | ||
|   - Use `only` to only pass that data. less tightenly coupled.
 | ||
|   * `render` calls the whole of Symfony, boots Kernel, can be expensive and slow
 | ||
|   * Loosely couple templates and controllers
 | ||
|     - Keep responses simple
 | ||
|     - What makes sense
 | ||
|     - if you need extra data in the template, get it in the template
 | ||
|   * View models
 | ||
|     - Mixed results
 | ||
|     - `BlogPostViewModel`
 | ||
|     - Can result in boilerplate code
 | ||
|     - Can be useful if the view model is different to the Entity
 | ||
|   * DRY
 | ||
|     - "Don't repeat yourself"
 | ||
| 
 | ||
| - Faster development
 | ||
|   - Separate UI tests from back-end tests. Different layers for different teams.
 | ||
|     People don't need to run everything if they are only changing certain
 | ||
|     things.
 | ||
| 
 | ||
| * Help your front end
 | ||
|   - Webpack - Encore
 | ||
|   - Type hinting in functions and filters, easier to debug
 | ||
|   - Logging
 | ||
|   - Friendly exceptions - help front-end devs by returning meaningful, readbale
 | ||
|     errors
 | ||
|   - Web Debug Toolbar and Profiler, provide training for toolbar and profilers
 | ||
|   - Twig-friendly development environment - Twig support in IDEs and text
 | ||
|     editors
 | ||
| 
 | ||
| SPAs are sometimes teh right solution. Why do they want to use it, can the same
 | ||
| benefits be added with Twig?
 | ||
| 
 | ||
| 3 most important points:
 | ||
| 
 | ||
| - Profile, identidy, optimise, monitor
 | ||
| - Loosely couple templates to your app code
 | ||
| - Help your front ends - put your front end developers first
 | ||
| - You don't need to use a SPA for single pages, use JavaScript for that one
 | ||
|   page. It doesn't need to be all or nothing.
 | ||
| 
 | ||
| ## BDD Your Symfony Application (Kamil Kokot)
 | ||
| 
 | ||
| - Applying BDD to Sylius
 | ||
| - 2 years since release of Sylius (Symfony 2 alpha)
 | ||
| - The business part is more important than the code part
 | ||
| 
 | ||
| ### What is BDD?
 | ||
| 
 | ||
| - Behaviour driven development. Combines TDD and DDD, into an agile methodology
 | ||
| - Encourages communication and creates shared understanding
 | ||
| - Living, executable documentation that non-programmers understand. Always
 | ||
|   correct.
 | ||
| - Feature file
 | ||
|   - Feature
 | ||
|   - Scenario - example of the behaviour for this feature. Simple, atomic. (e.g.
 | ||
|     I need a product in order to add it to a cart)
 | ||
|   - In order to...
 | ||
|   - Who gets the benefit?
 | ||
| 
 | ||
| ### BDD in practice
 | ||
| 
 | ||
| - Feature: booking flight tickets
 | ||
| - Scenario: booking flight ticket for one person
 | ||
|   - Given there are the following flights...
 | ||
|   - When I visit '/flight/LTN-WAW'
 | ||
|   - Then I should be on '/flight/LTN-WAW'
 | ||
|   - Add I should see "Your flight has been booked." in "#result"
 | ||
| - In the BDD way - what is the business logic? What is the value for this
 | ||
|   scenario? What is the reason 'why', and who benefits from this?
 | ||
|   - We just need to know that there are 5 seats left on a flight
 | ||
|   - Talk and communicate about how the feature is going to work - not just
 | ||
|     developers
 | ||
|   - BDD aids communication
 | ||
| - Questions we can ask
 | ||
|   - Can we get a different outcome when the context changes?
 | ||
|     - When there was only one seat available
 | ||
|     - When there were no available seats
 | ||
|   - Can we get the same outcome when the event changes? Can we change 'When' and
 | ||
|     'Then stays the same'
 | ||
|     - When it is booked for an adult and a child
 | ||
|     - When it is booked for an adult
 | ||
|   - Does anything else happen that is not mentioned?
 | ||
|     - Generate an invoice if a seat is booked
 | ||
|     - a pilot would like to get a notification that a seat was booked.
 | ||
|   * Figuring out the rules
 | ||
|     - Adults are 15+ years old
 | ||
|     - Children are 2-14 years old
 | ||
|     - Infants and children can only travel with an adult
 | ||
|     - We don't allow for overbooking
 | ||
|   - Translating rules into examples
 | ||
|     - Add a new scenario for each rule - e.g. don't allow over booking
 | ||
|       - "And the flight should be no longer available..."
 | ||
| 
 | ||
| ### Behat
 | ||
| 
 | ||
| - Used to automate and execute BDD tests, also SpecDDD
 | ||
| - maps steps to PHP code
 | ||
| - Given a context, when an event, then an outcome
 | ||
| - Domain Context, API context
 | ||
| - class implements `Context`, annotations for `@Given`, `@When`, `@Then`. allows
 | ||
|   for arguments and regular expressions
 | ||
| - Suites: change what code is executed, and what scenarios are executed. context
 | ||
|   and tags
 | ||
| - FriendsOfBehat SymfonyExtension - integrates Behat with Symfony
 | ||
|   - Contexts registered as Symfony services - inject dependencies, service as a
 | ||
|     context in Behat. Need to be 'public' for it to work
 | ||
|   - Reduces boilerplate code. Supports autowiring.
 | ||
|   - Zero configuration
 | ||
| 
 | ||
| ### Domain context
 | ||
| 
 | ||
| - `Given` verb matches `@Given` annotation. Same for `When` and `Then`.
 | ||
| - Transformers, type hint name string, return Client instance
 | ||
| 
 | ||
| ### API context
 | ||
| 
 | ||
| - inject `FlightBookingService` and `KernelBrowser`
 | ||
| - Use `$this->kernelBrowser->request()`
 | ||
| - Use `assert()` function wuthin `@Then`
 | ||
| 
 | ||
| ### Back to reality - how it's done with Sylius
 | ||
| 
 | ||
| - Business part applies to all context. Start talking about what needs to be
 | ||
|   done, start communicating
 | ||
| - Implement contexts for UI and API
 | ||
| - 12716 steps, 1175 scenarios, 8 min 8 sec, 2.4 scenarios /sec
 | ||
| - 12x faster than JS (17 min 48 sec, 0.19 scenario / sec)
 | ||
| - Treat test CI environment like production
 | ||
| 
 | ||
|   - Turn off debug settings, add caching
 | ||
|   - Enable OPcache
 | ||
| 
 | ||
| - Write features in a natural way
 | ||
| - Too many setup steps - merge steps. less visual debt. e.g. Create currency,
 | ||
|   zone and locale when creating a store
 | ||
| - Avoid scenarios that are too detailed. You should specify only what's
 | ||
|   important to this scenario.
 | ||
| 
 | ||
| ## Migrating to Symfony one route at a time (Steve Winter)
 | ||
| 
 | ||
| - New client with an old application, built in an old version of another
 | ||
|   framework with unusual dependency management, no tests, no version control and
 | ||
|   deploying via FTP. Done over a ~3 month period.
 | ||
| 
 | ||
| - Subscription based index of suppliers
 | ||
| - New requirements to implement by the client
 | ||
| - Our requirements: Needed a deployment process, make it testable, fix the build
 | ||
|   chain
 | ||
| - Solution attempt 1: Migrate to a new version of the current framework
 | ||
|   - Minor template and design changes were fine
 | ||
|   - Modifiy features, add new dependencies.
 | ||
| - Solution attempt 2: Upgrade to the latest version - same outcome due to
 | ||
|   multiple BC breaks (no semver), lots of manual steps
 | ||
| - Solution attempt 3: Symfony!
 | ||
|   - Semver! Backwards compatibility promise
 | ||
|   - Symfony app to run in parallel, Apache proxy rules and minor changes to the
 | ||
|     legacy app, added data transfer mechanisms
 | ||
|   - Anything new done in Symfony
 | ||
|   - Installed on the same server with it's own vhost but not publicly accessible
 | ||
|   - Deployed independently of legacy app
 | ||
| 
 | ||
| ### Apache proxy rules
 | ||
| 
 | ||
| Proxy `/public` to symfony app
 | ||
| 
 | ||
| ### Legacy app
 | ||
| 
 | ||
| - Shared cookie for single login between apps - user account details (name etc),
 | ||
|   session details (login time)
 | ||
| 
 | ||
| ### Added functionality
 | ||
| 
 | ||
| - Built in Symfony
 | ||
| - new proxy rules for new routes
 | ||
| - Add menu links to legacy app menu
 | ||
| - How do we show how many reminders are active?
 | ||
|   - Symfony based API called from the front-end
 | ||
| 
 | ||
| ### Migrating routes
 | ||
| 
 | ||
| - Rebuilt or extend in Symfony app
 | ||
| - Test and deploy, then update the apache config to add new proxy rules
 | ||
| 
 | ||
| ### A gotcha
 | ||
| 
 | ||
| - Legacy app uses CSRF
 | ||
| - Needed to track the token, added to shared cookie and pass through to the
 | ||
|   Symfony side
 | ||
| 
 | ||
| ### Storing data
 | ||
| 
 | ||
| - Both apps using the same data with different credentials
 | ||
| - Some shared tables, some tables are specific to each app
 | ||
| 
 | ||
| ### Remaining challenges
 | ||
| 
 | ||
| - User session management, still handled by legacy app
 | ||
| - Templating/CSS - two versions of everything
 | ||
|   - Next step: move all CSS to Symfony
 | ||
| 
 | ||
| ### Summary
 | ||
| 
 | ||
| - Add Symfony app, Apache proxy rules for routes
 | ||
| - User transfer mechanisms
 | ||
| - New functionality added in Symfony
 | ||
| 
 | ||
| ### Is this right for you?
 | ||
| 
 | ||
| It depends. Fine for a 'modest' size. Use a real proxy for larger scale apps,
 | ||
| use different servers with database replication.
 | ||
| 
 | ||
| ## Closing Keynote: The fabulous World of Emojis and other Unicode symbols (Nicolas Grekas)
 | ||
| 
 | ||
| - ASCII. Still used today. Map between the first 128 numbers to characters. OK
 | ||
|   for UK and US.
 | ||
| - 256 numbers in Windows-1252 (character sets). Each country had their own set.
 | ||
| - It's legacy. 0.2% for Windows-1252. 88.8% for UTF-8 (Feb 2017)
 | ||
| - Unicode: 130k characters, 135 scripts (alphabets)
 | ||
| - Validation errors using native alphabet - e.g. invalid last name when
 | ||
|   submitting a form
 | ||
| - 17 plans, each square is 255 code points
 | ||
| - Emojis are characters, not images
 | ||
| - Gliph is a visual representation of a character
 | ||
| - From code points to bytes
 | ||
|   - UTF-8: 1,2,3 or 4 bytes
 | ||
|   - UTF16: 2 or 4 bytes
 | ||
|   - UTF-32: 4 bytes
 | ||
| - UTF-8 is compatible with ASCII
 | ||
| - Case sensitivity - 1k characters are concerned. One uppercase letter, two
 | ||
|   lower case variants. Turkish exception (similar looking letters that are
 | ||
|   different letters with different meanings). Full case folding.
 | ||
| - Collations - ordering is depends on the language. 'ch' in Spanish is a single
 | ||
|   character.
 | ||
| - Single number in unicode to represent accents. Combining characters.
 | ||
| - Composed (NFC) and decomposed (NFD) forms - normalisation for comparison
 | ||
| - Grapheme clusters - multiple characters, but one letter as you write it
 | ||
|   (separate characters for letters and accent)
 | ||
| - Emjois - combining characters. e.g. Combine face with colour. Different codes
 | ||
|   and character names. Also applies to ligatures. A way to combine several
 | ||
|   images together into one single visual representation.
 | ||
| 
 | ||
| ### unicode fundamentals
 | ||
| 
 | ||
| - uppercase, lowercase, folding
 | ||
| - compositions, ligatures
 | ||
| - comparistions - normalisations and collations
 | ||
| - segmentation: characters, words, sentences and hyphens
 | ||
| - locales: cultural conventions, translitterations
 | ||
| - identifiers & security, confusables
 | ||
| - display: direction, width
 | ||
| 
 | ||
| ### unicode in practice
 | ||
| 
 | ||
| - MySQL - `utf*_*`. `SET NAMES utf8mb4` for security and storing emojis. Cannot
 | ||
|   store emojis with `utf8`
 | ||
| 
 | ||
| ### in php
 | ||
| 
 | ||
| - `mb_*()`
 | ||
| - `iconv_*()`
 | ||
| - `preg_*()`
 | ||
| - `grapheme_*()` `normalizer_*()`
 | ||
| - `symfony/polyfill-*` - pure PHP implementation
 | ||
| - Made a component - **symfony/string** -
 | ||
|   https://github.com/symfony/symfony/pull/33553
 | ||
| - Object orientated api for strings. Immutable value objects
 | ||
| - `AbstractString`
 | ||
|   - `GraphemeString`
 | ||
|   - `Utf8String`
 | ||
|   - `BinaryString`
 | ||
| 
 | ||
| * AbstractString - Methods to serialize, get length, to binary or grapheme or
 | ||
|   utf8
 | ||
|   - Methods for starts with, ends with, is empty, join, prepend, split, trim,
 | ||
|     title etc
 |