You've already forked joplin
							
							
				mirror of
				https://github.com/laurent22/joplin.git
				synced 2025-10-31 00:07:48 +02:00 
			
		
		
		
	First commit
This commit is contained in:
		
							
								
								
									
										17
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| /app/config/parameters.yml | ||||
| /build/ | ||||
| /phpunit.xml | ||||
| /var/* | ||||
| !/var/cache | ||||
| /var/cache/* | ||||
| !var/cache/.gitkeep | ||||
| !/var/logs | ||||
| /var/logs/* | ||||
| !var/logs/.gitkeep | ||||
| !/var/sessions | ||||
| /var/sessions/* | ||||
| !var/sessions/.gitkeep | ||||
| !var/SymfonyRequirements.php | ||||
| /vendor/ | ||||
| /web/bundles/ | ||||
| *.sublime-workspace | ||||
							
								
								
									
										4
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| notes | ||||
| ===== | ||||
|  | ||||
| A Symfony project created on October 5, 2016, 4:15 pm. | ||||
							
								
								
									
										7
									
								
								app/.htaccess
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								app/.htaccess
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| <IfModule mod_authz_core.c> | ||||
|     Require all denied | ||||
| </IfModule> | ||||
| <IfModule !mod_authz_core.c> | ||||
|     Order deny,allow | ||||
|     Deny from all | ||||
| </IfModule> | ||||
							
								
								
									
										7
									
								
								app/AppCache.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								app/AppCache.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| <?php | ||||
|  | ||||
| use Symfony\Bundle\FrameworkBundle\HttpCache\HttpCache; | ||||
|  | ||||
| class AppCache extends HttpCache | ||||
| { | ||||
| } | ||||
							
								
								
									
										50
									
								
								app/AppKernel.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								app/AppKernel.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| <?php | ||||
|  | ||||
| use Symfony\Component\HttpKernel\Kernel; | ||||
| use Symfony\Component\Config\Loader\LoaderInterface; | ||||
|  | ||||
| class AppKernel extends Kernel | ||||
| { | ||||
|     public function registerBundles() | ||||
|     { | ||||
|         $bundles = [ | ||||
|             new Symfony\Bundle\FrameworkBundle\FrameworkBundle(), | ||||
|             new Symfony\Bundle\SecurityBundle\SecurityBundle(), | ||||
|             new Symfony\Bundle\TwigBundle\TwigBundle(), | ||||
|             new Symfony\Bundle\MonologBundle\MonologBundle(), | ||||
|             new Symfony\Bundle\SwiftmailerBundle\SwiftmailerBundle(), | ||||
|             new Doctrine\Bundle\DoctrineBundle\DoctrineBundle(), | ||||
|             new Sensio\Bundle\FrameworkExtraBundle\SensioFrameworkExtraBundle(), | ||||
|             new AppBundle\AppBundle(), | ||||
|         ]; | ||||
|  | ||||
|         if (in_array($this->getEnvironment(), ['dev', 'test'], true)) { | ||||
|             $bundles[] = new Symfony\Bundle\DebugBundle\DebugBundle(); | ||||
|             $bundles[] = new Symfony\Bundle\WebProfilerBundle\WebProfilerBundle(); | ||||
|             $bundles[] = new Sensio\Bundle\DistributionBundle\SensioDistributionBundle(); | ||||
|             $bundles[] = new Sensio\Bundle\GeneratorBundle\SensioGeneratorBundle(); | ||||
|         } | ||||
|  | ||||
|         return $bundles; | ||||
|     } | ||||
|  | ||||
|     public function getRootDir() | ||||
|     { | ||||
|         return __DIR__; | ||||
|     } | ||||
|  | ||||
|     public function getCacheDir() | ||||
|     { | ||||
|         return dirname(__DIR__).'/var/cache/'.$this->getEnvironment(); | ||||
|     } | ||||
|  | ||||
|     public function getLogDir() | ||||
|     { | ||||
|         return dirname(__DIR__).'/var/logs'; | ||||
|     } | ||||
|  | ||||
|     public function registerContainerConfiguration(LoaderInterface $loader) | ||||
|     { | ||||
|         $loader->load($this->getRootDir().'/config/config_'.$this->getEnvironment().'.yml'); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										13
									
								
								app/Resources/views/base.html.twig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								app/Resources/views/base.html.twig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
|     <head> | ||||
|         <meta charset="UTF-8" /> | ||||
|         <title>{% block title %}Welcome!{% endblock %}</title> | ||||
|         {% block stylesheets %}{% endblock %} | ||||
|         <link rel="icon" type="image/x-icon" href="{{ asset('favicon.ico') }}" /> | ||||
|     </head> | ||||
|     <body> | ||||
|         {% block body %}{% endblock %} | ||||
|         {% block javascripts %}{% endblock %} | ||||
|     </body> | ||||
| </html> | ||||
							
								
								
									
										76
									
								
								app/Resources/views/default/index.html.twig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								app/Resources/views/default/index.html.twig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| {% extends 'base.html.twig' %} | ||||
|  | ||||
| {% block body %} | ||||
|     <div id="wrapper"> | ||||
|         <div id="container"> | ||||
|             <div id="welcome"> | ||||
|                 <h1><span>Welcome to</span> Symfony {{ constant('Symfony\\Component\\HttpKernel\\Kernel::VERSION') }}</h1> | ||||
|             </div> | ||||
|  | ||||
|             <div id="status"> | ||||
|                 <p> | ||||
|                     <svg id="icon-status" width="1792" height="1792" viewBox="0 0 1792 1792" xmlns="http://www.w3.org/2000/svg"><path d="M1671 566q0 40-28 68l-724 724-136 136q-28 28-68 28t-68-28l-136-136-362-362q-28-28-28-68t28-68l136-136q28-28 68-28t68 28l294 295 656-657q28-28 68-28t68 28l136 136q28 28 28 68z" fill="#759E1A"/></svg> | ||||
|  | ||||
|                     Your application is now ready. You can start working on it at: | ||||
|                     <code>{{ base_dir }}</code> | ||||
|                 </p> | ||||
|             </div> | ||||
|  | ||||
|             <div id="next"> | ||||
|                 <h2>What's next?</h2> | ||||
|                 <p> | ||||
|                     <svg id="icon-book" version="1.1" xmlns="http://www.w3.org/2000/svg" x="0px" y="0px" viewBox="-12.5 9 64 64" enable-background="new -12.5 9 64 64" xml:space="preserve"> | ||||
|                         <path fill="#AAA" d="M6.8,40.8c2.4,0.8,4.5-0.7,4.9-2.5c0.2-1.2-0.3-2.1-1.3-3.2l-0.8-0.8c-0.4-0.5-0.6-1.3-0.2-1.9 | ||||
|                             c0.4-0.5,0.9-0.8,1.8-0.5c1.3,0.4,1.9,1.3,2.9,2.2c-0.4,1.4-0.7,2.9-0.9,4.2l-0.2,1c-0.7,4-1.3,6.2-2.7,7.5 | ||||
|                             c-0.3,0.3-0.7,0.5-1.3,0.6c-0.3,0-0.4-0.3-0.4-0.3c0-0.3,0.2-0.3,0.3-0.4c0.2-0.1,0.5-0.3,0.4-0.8c0-0.7-0.6-1.3-1.3-1.3 | ||||
|                             c-0.6,0-1.4,0.6-1.4,1.7s1,1.9,2.4,1.8c0.8,0,2.5-0.3,4.2-2.5c2-2.5,2.5-5.4,2.9-7.4l0.5-2.8c0.3,0,0.5,0.1,0.8,0.1 | ||||
|                             c2.4,0.1,3.7-1.3,3.7-2.3c0-0.6-0.3-1.2-0.9-1.2c-0.4,0-0.8,0.3-1,0.8c-0.1,0.6,0.8,1.1,0.1,1.5c-0.5,0.3-1.4,0.6-2.7,0.4l0.3-1.3 | ||||
|                             c0.5-2.6,1-5.7,3.2-5.8c0.2,0,0.8,0,0.8,0.4c0,0.2,0,0.2-0.2,0.5c-0.2,0.3-0.3,0.4-0.2,0.7c0,0.7,0.5,1.1,1.2,1.1 | ||||
|                             c0.9,0,1.2-1,1.2-1.4c0-1.2-1.2-1.8-2.6-1.8c-1.5,0.1-2.8,0.9-3.7,2.1c-1.1,1.3-1.8,2.9-2.3,4.5c-0.9-0.8-1.6-1.8-3.1-2.3 | ||||
|                             c-1.1-0.7-2.3-0.5-3.4,0.3c-0.5,0.4-0.8,1-1,1.6c-0.4,1.5,0.4,2.9,0.8,3.4l0.9,1c0.2,0.2,0.6,0.8,0.4,1.5c-0.3,0.8-1.2,1.3-2.1,1 | ||||
|                             c-0.4-0.2-1-0.5-0.9-0.9c0.1-0.2,0.2-0.3,0.3-0.5s0.1-0.3,0.1-0.3c0.2-0.6-0.1-1.4-0.7-1.6c-0.6-0.2-1.2,0-1.3,0.8 | ||||
|                             C4.3,38.4,4.7,40,6.8,40.8z M46.1,20.9c0-4.2-3.2-7.5-7.1-7.5h-3.8C34.8,10.8,32.7,9,30.2,9L-2.3,9.1c-2.8,0.1-4.9,2.4-4.9,5.4 | ||||
|                             L-7,58.6c0,4.8,8.1,13.9,11.6,14.1l34.7-0.1c3.9,0,7-3.4,7-7.6L46.1,20.9z M-0.3,36.4c0-8.6,6.5-15.6,14.5-15.6 | ||||
|                             c8,0,14.5,7,14.5,15.6S22.1,52,14.2,52C6.1,52-0.3,45-0.3,36.4z M42.1,65.1c0,1.8-1.5,3.1-3.1,3.1H4.6c-0.7,0-3-1.8-4.5-4.4h30.4 | ||||
|                             c2.8,0,5-2.4,5-5.4V17.9h3.7c1.6,0,2.9,1.4,2.9,3.1V65.1L42.1,65.1z"/> | ||||
|                     </svg> | ||||
|  | ||||
|                     Read the documentation to learn | ||||
|                     <a href="http://symfony.com/doc/{{ constant('Symfony\\Component\\HttpKernel\\Kernel::VERSION')[:3] }}/book/page_creation.html"> | ||||
|                         How to create your first page in Symfony | ||||
|                     </a> | ||||
|                 </p> | ||||
|             </div> | ||||
|  | ||||
|         </div> | ||||
|     </div> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block stylesheets %} | ||||
| <style> | ||||
|     body { background: #F5F5F5; font: 18px/1.5 sans-serif; } | ||||
|     h1, h2 { line-height: 1.2; margin: 0 0 .5em; } | ||||
|     h1 { font-size: 36px; } | ||||
|     h2 { font-size: 21px; margin-bottom: 1em; } | ||||
|     p { margin: 0 0 1em 0; } | ||||
|     a { color: #0000F0; } | ||||
|     a:hover { text-decoration: none; } | ||||
|     code { background: #F5F5F5; max-width: 100px; padding: 2px 6px; word-wrap: break-word; } | ||||
|     #wrapper { background: #FFF; margin: 1em auto; max-width: 800px; width: 95%; } | ||||
|     #container { padding: 2em; } | ||||
|     #welcome, #status { margin-bottom: 2em; } | ||||
|     #welcome h1 span { display: block; font-size: 75%; } | ||||
|     #icon-status, #icon-book { float: left; height: 64px; margin-right: 1em; margin-top: -4px; width: 64px; } | ||||
|     #icon-book { display: none; } | ||||
|  | ||||
|     @media (min-width: 768px) { | ||||
|         #wrapper { width: 80%; margin: 2em auto; } | ||||
|         #icon-book { display: inline-block; } | ||||
|         #status a, #next a { display: block; } | ||||
|  | ||||
|         @-webkit-keyframes fade-in { 0% { opacity: 0; } 100% { opacity: 1; } } | ||||
|         @keyframes fade-in { 0% { opacity: 0; } 100% { opacity: 1; } } | ||||
|         .sf-toolbar { opacity: 0; -webkit-animation: fade-in 1s .2s forwards; animation: fade-in 1s .2s forwards;} | ||||
|     } | ||||
| </style> | ||||
| {% endblock %} | ||||
							
								
								
									
										11
									
								
								app/autoload.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								app/autoload.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| <?php | ||||
|  | ||||
| use Doctrine\Common\Annotations\AnnotationRegistry; | ||||
| use Composer\Autoload\ClassLoader; | ||||
|  | ||||
| /** @var ClassLoader $loader */ | ||||
| $loader = require __DIR__.'/../vendor/autoload.php'; | ||||
|  | ||||
| AnnotationRegistry::registerLoader([$loader, 'loadClass']); | ||||
|  | ||||
| return $loader; | ||||
							
								
								
									
										68
									
								
								app/config/config.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								app/config/config.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| imports: | ||||
|     - { resource: parameters.yml } | ||||
|     - { resource: security.yml } | ||||
|     - { resource: services.yml } | ||||
|  | ||||
| # Put parameters here that don't need to change on each machine where the app is deployed | ||||
| # http://symfony.com/doc/current/best_practices/configuration.html#application-related-configuration | ||||
| parameters: | ||||
|     locale: en | ||||
|  | ||||
| framework: | ||||
|     #esi:             ~ | ||||
|     #translator:      { fallbacks: ["%locale%"] } | ||||
|     secret:          "%secret%" | ||||
|     router: | ||||
|         resource: "%kernel.root_dir%/config/routing.yml" | ||||
|         strict_requirements: ~ | ||||
|     form:            ~ | ||||
|     csrf_protection: ~ | ||||
|     validation:      { enable_annotations: true } | ||||
|     #serializer:      { enable_annotations: true } | ||||
|     templating: | ||||
|         engines: ['twig'] | ||||
|     default_locale:  "%locale%" | ||||
|     trusted_hosts:   ~ | ||||
|     trusted_proxies: ~ | ||||
|     session: | ||||
|         # http://symfony.com/doc/current/reference/configuration/framework.html#handler-id | ||||
|         handler_id:  session.handler.native_file | ||||
|         save_path:   "%kernel.root_dir%/../var/sessions/%kernel.environment%" | ||||
|     fragments:       ~ | ||||
|     http_method_override: true | ||||
|     assets: ~ | ||||
|  | ||||
| # Twig Configuration | ||||
| twig: | ||||
|     debug:            "%kernel.debug%" | ||||
|     strict_variables: "%kernel.debug%" | ||||
|  | ||||
| # Doctrine Configuration | ||||
| doctrine: | ||||
|     dbal: | ||||
|         driver:   pdo_mysql | ||||
|         host:     "%database_host%" | ||||
|         port:     "%database_port%" | ||||
|         dbname:   "%database_name%" | ||||
|         user:     "%database_user%" | ||||
|         password: "%database_password%" | ||||
|         charset:  UTF8 | ||||
|         # if using pdo_sqlite as your database driver: | ||||
|         #   1. add the path in parameters.yml | ||||
|         #     e.g. database_path: "%kernel.root_dir%/data/data.db3" | ||||
|         #   2. Uncomment database_path in parameters.yml.dist | ||||
|         #   3. Uncomment next line: | ||||
|         #     path:     "%database_path%" | ||||
|  | ||||
|     orm: | ||||
|         auto_generate_proxy_classes: "%kernel.debug%" | ||||
|         naming_strategy: doctrine.orm.naming_strategy.underscore | ||||
|         auto_mapping: true | ||||
|  | ||||
| # Swiftmailer Configuration | ||||
| swiftmailer: | ||||
|     transport: "%mailer_transport%" | ||||
|     host:      "%mailer_host%" | ||||
|     username:  "%mailer_user%" | ||||
|     password:  "%mailer_password%" | ||||
|     spool:     { type: memory } | ||||
							
								
								
									
										34
									
								
								app/config/config_dev.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								app/config/config_dev.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| imports: | ||||
|     - { resource: config.yml } | ||||
|  | ||||
| framework: | ||||
|     router: | ||||
|         resource: "%kernel.root_dir%/config/routing_dev.yml" | ||||
|         strict_requirements: true | ||||
|     profiler: { only_exceptions: false } | ||||
|  | ||||
| web_profiler: | ||||
|     toolbar: true | ||||
|     intercept_redirects: false | ||||
|  | ||||
| monolog: | ||||
|     handlers: | ||||
|         main: | ||||
|             type: stream | ||||
|             path: "%kernel.logs_dir%/%kernel.environment%.log" | ||||
|             level: debug | ||||
|             channels: [!event] | ||||
|         console: | ||||
|             type:   console | ||||
|             channels: [!event, !doctrine] | ||||
|         # uncomment to get logging in your browser | ||||
|         # you may have to allow bigger header sizes in your Web server configuration | ||||
|         #firephp: | ||||
|         #    type:   firephp | ||||
|         #    level:  info | ||||
|         #chromephp: | ||||
|         #    type:   chromephp | ||||
|         #    level:  info | ||||
|  | ||||
| #swiftmailer: | ||||
| #    delivery_address: me@example.com | ||||
							
								
								
									
										21
									
								
								app/config/config_prod.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								app/config/config_prod.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| imports: | ||||
|     - { resource: config.yml } | ||||
|  | ||||
| #doctrine: | ||||
| #    orm: | ||||
| #        metadata_cache_driver: apc | ||||
| #        result_cache_driver: apc | ||||
| #        query_cache_driver: apc | ||||
|  | ||||
| monolog: | ||||
|     handlers: | ||||
|         main: | ||||
|             type:         fingers_crossed | ||||
|             action_level: error | ||||
|             handler:      nested | ||||
|         nested: | ||||
|             type:  stream | ||||
|             path:  "%kernel.logs_dir%/%kernel.environment%.log" | ||||
|             level: debug | ||||
|         console: | ||||
|             type:  console | ||||
							
								
								
									
										16
									
								
								app/config/config_test.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								app/config/config_test.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| imports: | ||||
|     - { resource: config_dev.yml } | ||||
|  | ||||
| framework: | ||||
|     test: ~ | ||||
|     session: | ||||
|         storage_id: session.storage.mock_file | ||||
|     profiler: | ||||
|         collect: false | ||||
|  | ||||
| web_profiler: | ||||
|     toolbar: false | ||||
|     intercept_redirects: false | ||||
|  | ||||
| swiftmailer: | ||||
|     disable_delivery: true | ||||
							
								
								
									
										19
									
								
								app/config/parameters.yml.dist
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								app/config/parameters.yml.dist
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| # This file is a "template" of what your parameters.yml file should look like | ||||
| # Set parameters here that may be different on each deployment target of the app, e.g. development, staging, production. | ||||
| # http://symfony.com/doc/current/best_practices/configuration.html#infrastructure-related-configuration | ||||
| parameters: | ||||
|     database_host:     127.0.0.1 | ||||
|     database_port:     ~ | ||||
|     database_name:     symfony | ||||
|     database_user:     root | ||||
|     database_password: ~ | ||||
|     # You should uncomment this if you want use pdo_sqlite | ||||
|     # database_path: "%kernel.root_dir%/data.db3" | ||||
|  | ||||
|     mailer_transport:  smtp | ||||
|     mailer_host:       127.0.0.1 | ||||
|     mailer_user:       ~ | ||||
|     mailer_password:   ~ | ||||
|  | ||||
|     # A secret key that's used to generate certain security-related tokens | ||||
|     secret:            ThisTokenIsNotSoSecretChangeIt | ||||
							
								
								
									
										3
									
								
								app/config/routing.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								app/config/routing.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| app: | ||||
|     resource: "@AppBundle/Controller/" | ||||
|     type:     annotation | ||||
							
								
								
									
										14
									
								
								app/config/routing_dev.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								app/config/routing_dev.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| _wdt: | ||||
|     resource: "@WebProfilerBundle/Resources/config/routing/wdt.xml" | ||||
|     prefix:   /_wdt | ||||
|  | ||||
| _profiler: | ||||
|     resource: "@WebProfilerBundle/Resources/config/routing/profiler.xml" | ||||
|     prefix:   /_profiler | ||||
|  | ||||
| _errors: | ||||
|     resource: "@TwigBundle/Resources/config/routing/errors.xml" | ||||
|     prefix:   /_error | ||||
|  | ||||
| _main: | ||||
|     resource: routing.yml | ||||
							
								
								
									
										24
									
								
								app/config/security.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								app/config/security.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| # To get started with security, check out the documentation: | ||||
| # http://symfony.com/doc/current/book/security.html | ||||
| security: | ||||
|  | ||||
|     # http://symfony.com/doc/current/book/security.html#where-do-users-come-from-user-providers | ||||
|     providers: | ||||
|         in_memory: | ||||
|             memory: ~ | ||||
|  | ||||
|     firewalls: | ||||
|         # disables authentication for assets and the profiler, adapt it according to your needs | ||||
|         dev: | ||||
|             pattern: ^/(_(profiler|wdt)|css|images|js)/ | ||||
|             security: false | ||||
|  | ||||
|         main: | ||||
|             anonymous: ~ | ||||
|             # activate different ways to authenticate | ||||
|  | ||||
|             # http_basic: ~ | ||||
|             # http://symfony.com/doc/current/book/security.html#a-configuring-how-your-users-will-authenticate | ||||
|  | ||||
|             # form_login: ~ | ||||
|             # http://symfony.com/doc/current/cookbook/security/form_login_setup.html | ||||
							
								
								
									
										17
									
								
								app/config/services.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								app/config/services.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| # Learn more about services, parameters and containers at | ||||
| # http://symfony.com/doc/current/book/service_container.html | ||||
| parameters: | ||||
| #    parameter_name: value | ||||
|  | ||||
| services: | ||||
|  | ||||
|     app.eloquent: | ||||
|         class: AppBundle\Eloquent | ||||
|         arguments: [] | ||||
|  | ||||
|     # app.fine_diff: | ||||
|     #     class: AppBundle\FineDiff | ||||
|     #     arguments: [] | ||||
|  | ||||
|     twig.exception_listener: | ||||
|       class: stdObject | ||||
							
								
								
									
										27
									
								
								bin/console
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										27
									
								
								bin/console
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| #!/usr/bin/env php | ||||
| <?php | ||||
|  | ||||
| use Symfony\Bundle\FrameworkBundle\Console\Application; | ||||
| use Symfony\Component\Console\Input\ArgvInput; | ||||
| use Symfony\Component\Debug\Debug; | ||||
|  | ||||
| // if you don't want to setup permissions the proper way, just uncomment the following PHP line | ||||
| // read http://symfony.com/doc/current/book/installation.html#configuration-and-setup for more information | ||||
| //umask(0000); | ||||
|  | ||||
| set_time_limit(0); | ||||
|  | ||||
| /** @var Composer\Autoload\ClassLoader $loader */ | ||||
| $loader = require __DIR__.'/../app/autoload.php'; | ||||
|  | ||||
| $input = new ArgvInput(); | ||||
| $env = $input->getParameterOption(['--env', '-e'], getenv('SYMFONY_ENV') ?: 'dev'); | ||||
| $debug = getenv('SYMFONY_DEBUG') !== '0' && !$input->hasParameterOption(['--no-debug', '']) && $env !== 'prod'; | ||||
|  | ||||
| if ($debug) { | ||||
|     Debug::enable(); | ||||
| } | ||||
|  | ||||
| $kernel = new AppKernel($env, $debug); | ||||
| $application = new Application($kernel); | ||||
| $application->run($input); | ||||
							
								
								
									
										146
									
								
								bin/symfony_requirements
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										146
									
								
								bin/symfony_requirements
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,146 @@ | ||||
| #!/usr/bin/env php | ||||
| <?php | ||||
|  | ||||
| require_once dirname(__FILE__).'/../var/SymfonyRequirements.php'; | ||||
|  | ||||
| $lineSize = 70; | ||||
| $symfonyRequirements = new SymfonyRequirements(); | ||||
| $iniPath = $symfonyRequirements->getPhpIniConfigPath(); | ||||
|  | ||||
| echo_title('Symfony Requirements Checker'); | ||||
|  | ||||
| echo '> PHP is using the following php.ini file:'.PHP_EOL; | ||||
| if ($iniPath) { | ||||
|     echo_style('green', '  '.$iniPath); | ||||
| } else { | ||||
|     echo_style('yellow', '  WARNING: No configuration file (php.ini) used by PHP!'); | ||||
| } | ||||
|  | ||||
| echo PHP_EOL.PHP_EOL; | ||||
|  | ||||
| echo '> Checking Symfony requirements:'.PHP_EOL.'  '; | ||||
|  | ||||
| $messages = array(); | ||||
| foreach ($symfonyRequirements->getRequirements() as $req) { | ||||
|     if ($helpText = get_error_message($req, $lineSize)) { | ||||
|         echo_style('red', 'E'); | ||||
|         $messages['error'][] = $helpText; | ||||
|     } else { | ||||
|         echo_style('green', '.'); | ||||
|     } | ||||
| } | ||||
|  | ||||
| $checkPassed = empty($messages['error']); | ||||
|  | ||||
| foreach ($symfonyRequirements->getRecommendations() as $req) { | ||||
|     if ($helpText = get_error_message($req, $lineSize)) { | ||||
|         echo_style('yellow', 'W'); | ||||
|         $messages['warning'][] = $helpText; | ||||
|     } else { | ||||
|         echo_style('green', '.'); | ||||
|     } | ||||
| } | ||||
|  | ||||
| if ($checkPassed) { | ||||
|     echo_block('success', 'OK', 'Your system is ready to run Symfony projects'); | ||||
| } else { | ||||
|     echo_block('error', 'ERROR', 'Your system is not ready to run Symfony projects'); | ||||
|  | ||||
|     echo_title('Fix the following mandatory requirements', 'red'); | ||||
|  | ||||
|     foreach ($messages['error'] as $helpText) { | ||||
|         echo ' * '.$helpText.PHP_EOL; | ||||
|     } | ||||
| } | ||||
|  | ||||
| if (!empty($messages['warning'])) { | ||||
|     echo_title('Optional recommendations to improve your setup', 'yellow'); | ||||
|  | ||||
|     foreach ($messages['warning'] as $helpText) { | ||||
|         echo ' * '.$helpText.PHP_EOL; | ||||
|     } | ||||
| } | ||||
|  | ||||
| echo PHP_EOL; | ||||
| echo_style('title', 'Note'); | ||||
| echo '  The command console could use a different php.ini file'.PHP_EOL; | ||||
| echo_style('title', '~~~~'); | ||||
| echo '  than the one used with your web server. To be on the'.PHP_EOL; | ||||
| echo '      safe side, please check the requirements from your web'.PHP_EOL; | ||||
| echo '      server using the '; | ||||
| echo_style('yellow', 'web/config.php'); | ||||
| echo ' script.'.PHP_EOL; | ||||
| echo PHP_EOL; | ||||
|  | ||||
| exit($checkPassed ? 0 : 1); | ||||
|  | ||||
| function get_error_message(Requirement $requirement, $lineSize) | ||||
| { | ||||
|     if ($requirement->isFulfilled()) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     $errorMessage = wordwrap($requirement->getTestMessage(), $lineSize - 3, PHP_EOL.'   ').PHP_EOL; | ||||
|     $errorMessage .= '   > '.wordwrap($requirement->getHelpText(), $lineSize - 5, PHP_EOL.'   > ').PHP_EOL; | ||||
|  | ||||
|     return $errorMessage; | ||||
| } | ||||
|  | ||||
| function echo_title($title, $style = null) | ||||
| { | ||||
|     $style = $style ?: 'title'; | ||||
|  | ||||
|     echo PHP_EOL; | ||||
|     echo_style($style, $title.PHP_EOL); | ||||
|     echo_style($style, str_repeat('~', strlen($title)).PHP_EOL); | ||||
|     echo PHP_EOL; | ||||
| } | ||||
|  | ||||
| function echo_style($style, $message) | ||||
| { | ||||
|     // ANSI color codes | ||||
|     $styles = array( | ||||
|         'reset' => "\033[0m", | ||||
|         'red' => "\033[31m", | ||||
|         'green' => "\033[32m", | ||||
|         'yellow' => "\033[33m", | ||||
|         'error' => "\033[37;41m", | ||||
|         'success' => "\033[37;42m", | ||||
|         'title' => "\033[34m", | ||||
|     ); | ||||
|     $supports = has_color_support(); | ||||
|  | ||||
|     echo($supports ? $styles[$style] : '').$message.($supports ? $styles['reset'] : ''); | ||||
| } | ||||
|  | ||||
| function echo_block($style, $title, $message) | ||||
| { | ||||
|     $message = ' '.trim($message).' '; | ||||
|     $width = strlen($message); | ||||
|  | ||||
|     echo PHP_EOL.PHP_EOL; | ||||
|  | ||||
|     echo_style($style, str_repeat(' ', $width)); | ||||
|     echo PHP_EOL; | ||||
|     echo_style($style, str_pad(' ['.$title.']', $width, ' ', STR_PAD_RIGHT)); | ||||
|     echo PHP_EOL; | ||||
|     echo_style($style, $message); | ||||
|     echo PHP_EOL; | ||||
|     echo_style($style, str_repeat(' ', $width)); | ||||
|     echo PHP_EOL; | ||||
| } | ||||
|  | ||||
| function has_color_support() | ||||
| { | ||||
|     static $support; | ||||
|  | ||||
|     if (null === $support) { | ||||
|         if (DIRECTORY_SEPARATOR == '\\') { | ||||
|             $support = false !== getenv('ANSICON') || 'ON' === getenv('ConEmuANSI'); | ||||
|         } else { | ||||
|             $support = function_exists('posix_isatty') && @posix_isatty(STDOUT); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return $support; | ||||
| } | ||||
							
								
								
									
										1
									
								
								cli-client/.config1/config.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								cli-client/.config1/config.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| {"last_sync_id":81,"file_map":[],"test":"abcd","client_id":"11111111111111111111111111111111","last_sync_time":1476289357,"session_id":null,"folder_items":[]} | ||||
							
								
								
									
										1
									
								
								cli-client/.config2/config.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								cli-client/.config2/config.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| {"last_sync_id":80,"file_map":[],"client_id":"22222222222222222222222222222222","last_sync_time":1476289618,"folder_items":[],"session_id":null} | ||||
							
								
								
									
										2
									
								
								cli-client/client1.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										2
									
								
								cli-client/client1.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| #!/bin/bash | ||||
| php main.php --config ~/src/notes/cli-client/.config1 "$@" | ||||
							
								
								
									
										2
									
								
								cli-client/client2.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										2
									
								
								cli-client/client2.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| #!/bin/bash | ||||
| php main.php --config ~/src/notes/cli-client/.config2 "$@" | ||||
							
								
								
									
										174
									
								
								cli-client/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								cli-client/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	//"bufio" | ||||
| 	//"errors" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	// "os/exec" | ||||
| 	// "os/user" | ||||
| 	"path/filepath" | ||||
| 	"path" | ||||
| 	"crypto/md5" | ||||
| 	// "runtime" | ||||
| 	// "strconv" | ||||
| 	"strings" | ||||
| 	// "time" | ||||
| 	"net/url" | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/jessevdk/go-flags" | ||||
| ) | ||||
|  | ||||
| const VERSION = "1.0.0" | ||||
|  | ||||
| type SyncCommandOptions struct { | ||||
| 	// FfmpegPath        string `long:"ffmpeg" description:"Path to ffmpeg." default:"ffmpeg"` | ||||
| 	// FrameDirPath      string `short:"d" long:"frame-dir" description:"Path to directory that will contain the captured frames. (default: <PictureDirectory>/pmcctv)"` | ||||
| 	// RemoteDir         string `short:"r" long:"remote-dir" description:"Remote location where frames will be saved to. Must contain a path compatible with scp (eg. user@someip:~/pmcctv)."` | ||||
| 	// RemotePort        string `short:"p" long:"remote-port" description:"Port of remote location where frames will be saved to. If not set, whatever is the default scp port will be used (should be 22)."` | ||||
| 	// BurstModeDuration int    `short:"b" long:"burst-mode-duration" description:"Duration of burst mode, in seconds. Set to 0 to disable burst mode altogether." default:"10"` | ||||
| 	// BurstModeFormat   string `short:"f" long:"burst-mode-format" description:"Format of burst mode captured files, either \"image\" or \"video\"." default:"video"` | ||||
| 	// FramesTtl         int    `short:"t" long:"time-to-live" description:"For how long captured frames should be kept, in days." default:"7"` | ||||
| 	// InputDevice       string `short:"i" long:"input-device" description:"Name of capture input device. (default: auto-detect)"` | ||||
| } | ||||
|  | ||||
| type AppCommandOptions struct { | ||||
| 	Version bool `long:"version" description:"Display version information"` | ||||
| } | ||||
|  | ||||
| type CommandOptions struct { | ||||
| 	App AppCommandOptions | ||||
| 	Sync SyncCommandOptions | ||||
| } | ||||
|  | ||||
| func printHelp(flagParser *flags.Parser) { | ||||
| 	flagParser.WriteHelp(os.Stdout) | ||||
| 	fmt.Printf("\n") | ||||
| 	fmt.Printf("For help with a particular command, type \"%s <command> --help\"\n", path.Base(os.Args[0]))	 | ||||
| } | ||||
|  | ||||
| func createFlagParser() (CommandOptions, *flags.Parser) { | ||||
| 	var opts CommandOptions | ||||
| 	flagParser := flags.NewParser(&opts.App, flags.HelpFlag|flags.PassDoubleDash) | ||||
|  | ||||
| 	flagParser.AddCommand( | ||||
| 		"sync", | ||||
| 		"Synchronize notes", | ||||
| 		"Synchronize local notes with the server.", | ||||
| 		&opts.Sync, | ||||
| 	) | ||||
|  | ||||
| 	return opts, flagParser | ||||
| } | ||||
|  | ||||
| func createId(path string) string { | ||||
| 	h := md5.New() | ||||
| 	io.WriteString(h, "31208854954776365651") | ||||
| 	io.WriteString(h, path) | ||||
| 	return fmt.Sprintf("%x", h.Sum(nil)) | ||||
| } | ||||
|  | ||||
| func readFile(path string) (string, error) { | ||||
| 	content, err := ioutil.ReadFile(path) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	return string(content), nil | ||||
| } | ||||
|  | ||||
| func makeApiCall(method string, path string, data url.Values) (error, []byte) { | ||||
| 	baseUrl := "http://127.0.0.1:8000" | ||||
| 	fullUrl := baseUrl + "/" + path | ||||
|  | ||||
| 	client := http.Client{} | ||||
| 	request, err := http.NewRequest(method, fullUrl, strings.NewReader(data.Encode())) | ||||
| 	if err != nil { | ||||
| 		return err, []byte{} | ||||
| 	} | ||||
|  | ||||
| 	request.Header.Add("Content-Type", "application/x-www-form-urlencoded") | ||||
|  | ||||
| 	response, err := client.Do(request) | ||||
| 	body, err := ioutil.ReadAll(response.Body) | ||||
| 	if err != nil { | ||||
| 		return err, []byte{} | ||||
| 	} | ||||
|  | ||||
| 	return nil, body | ||||
| } | ||||
|  | ||||
| // type Note struct { | ||||
| //     Id string | ||||
| //     Title string | ||||
| //     Body string | ||||
| // } | ||||
|  | ||||
| func main() { | ||||
| 	var err error | ||||
|  | ||||
| 	//err, body := makeApiCall("GET", "users/aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", url.Values{}) | ||||
|  | ||||
| 	note := url.Values{} | ||||
| 	note.Add("title", "from go") | ||||
| 	note.Add("body", "body from go") | ||||
|  | ||||
| 	err, body := makeApiCall("POST", "notes", note) | ||||
| 	fmt.Println(err) | ||||
| 	fmt.Println(string(body)) | ||||
| 	os.Exit(0) | ||||
|  | ||||
|  | ||||
|  | ||||
| 	opts, flagParser := createFlagParser() | ||||
|  | ||||
| 	args, err := flagParser.Parse() | ||||
|  | ||||
| 	if err != nil { | ||||
| 		t := err.(*flags.Error).Type | ||||
| 		if t == flags.ErrHelp { | ||||
| 			printHelp(flagParser) | ||||
| 			os.Exit(0) | ||||
| 		} else if t == flags.ErrCommandRequired { | ||||
| 			// Here handle default flags (which are not associated with any command) | ||||
| 			if opts.App.Version { | ||||
| 				fmt.Println(VERSION) | ||||
| 				os.Exit(0) | ||||
| 			} | ||||
| 			printHelp(flagParser) | ||||
| 			os.Exit(0) | ||||
| 		} else { | ||||
| 			fmt.Printf("Error: %s\n", err) | ||||
| 			fmt.Printf("Type '%s --help' for more information.\n", path.Base(os.Args[0])) | ||||
| 			os.Exit(1) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	_ = args | ||||
|  | ||||
| 	fullPath := "/home/laurent/src/notes/cli-client/test" | ||||
|  | ||||
| 	walkPath := func (path string, info os.FileInfo, err error) error { | ||||
| 		if len(path) - len(fullPath) <= 0 { | ||||
| 			return nil | ||||
| 		} | ||||
| 		p := path[len(fullPath)+1:]; | ||||
| 		fmt.Println(p) | ||||
| 		fmt.Println(createId(p)) | ||||
|  | ||||
| 		if !info.IsDir() { | ||||
| 			content, err := readFile(path) | ||||
| 			if err != nil { | ||||
| 				fmt.Println(err) | ||||
| 				return err | ||||
| 			} | ||||
| 			fmt.Println(content) | ||||
| 		} | ||||
|  | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	filepath.Walk(fullPath, walkPath) | ||||
| } | ||||
							
								
								
									
										349
									
								
								cli-client/main.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										349
									
								
								cli-client/main.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,349 @@ | ||||
| <?php | ||||
|  | ||||
| function escapePathElement($element) { | ||||
| 	$valid = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789ÀàÁáÂâÃãÄäÇçÈèÉéÊêËëÌìÍíÎîÏïÑnÒòÓóÔôÕõÖöŠšÚùÛúÜûÙüÝyŸÿŽz_- ().,'; | ||||
| 	$output = ''; | ||||
| 	$chars = preg_split('//u', $element, -1, PREG_SPLIT_NO_EMPTY); // Split a UTF-8 string into characters | ||||
| 	foreach ($chars as $c) { | ||||
| 		if (strpos($valid, $c) !== false) { | ||||
| 			$output .= $c; | ||||
| 		} else { | ||||
| 			$output .= rawurlencode($c); | ||||
| 		} | ||||
| 	} | ||||
| 	return $output; | ||||
| } | ||||
|  | ||||
| function escapePath($path) { | ||||
| 	$output = ''; | ||||
| 	$elements = preg_split('/[\\\\\/]/', $path); | ||||
| 	for ($i = 0; $i < count($elements); $i++) { | ||||
| 		$e = $elements[$i]; | ||||
| 		if ($i > 0) $output .= '/'; | ||||
| 		$output .= escapePathElement($e); | ||||
| 	} | ||||
| 	return $output; | ||||
| } | ||||
|  | ||||
| class Api { | ||||
|  | ||||
| 	private $sessionId = null; | ||||
| 	private $baseUrl = null; | ||||
|  | ||||
| 	public function __construct($baseUrl) { | ||||
| 		$this->baseUrl = $baseUrl; | ||||
| 	} | ||||
|  | ||||
| 	static public function createId($string) { | ||||
| 		// TODO: This needs to be unique per user | ||||
| 		return md5('gKcr0 ^L3UL^fJV%1IW~~/Q`.,WRAr</8@$.k|uyK-w^d:k|{h!%(};|)OY9^lu=' . $string); | ||||
| 	} | ||||
|  | ||||
| 	public function setSessionId($sessionId) { | ||||
| 		$this->sessionId = $sessionId; | ||||
| 	} | ||||
|  | ||||
| 	public function toCurlCmd($method, $url, $data = null) { | ||||
| 		$cmd = 'curl'; | ||||
|  | ||||
| 		$addMethod = true; | ||||
| 		if ($method == 'GET') $addMethod = false; | ||||
| 		if ($method == 'POST' && count($data)) $addMethod = false; | ||||
| 		if ($addMethod) $cmd .= ' -X ' . $method; | ||||
|  | ||||
| 		if ($data) { | ||||
| 			foreach ($data as $k => $v) { | ||||
| 				$cmd .= ' -F "' . $k . '=' . rawurlencode($v) . '"'; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		$cmd .= ' ' . "'" . $url . "'"; | ||||
| 		return $cmd; | ||||
| 	} | ||||
|  | ||||
| 	public function exec($method, $path, $query = null, $data = null) { | ||||
| 		$url = $this->baseUrl . '/' . $path; | ||||
|  | ||||
| 		if (!$query) $query = array(); | ||||
| 		if ($this->sessionId) $query['session'] = $this->sessionId; | ||||
| 		if (count($query)) $url .= '?' . http_build_query($query); | ||||
|  | ||||
| 		$ch = curl_init($url); | ||||
| 		curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); | ||||
| 		if ($data) curl_setopt($ch, CURLOPT_POSTFIELDS, $data); | ||||
| 		if ($method == 'PATCH' || $method == 'PUT' || $method == 'DELETE') { | ||||
| 			curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method); | ||||
| 		} | ||||
|  | ||||
| 		$cmd = $this->toCurlCmd($method, $url, $data); | ||||
| 		echo $cmd . "\n"; | ||||
|  | ||||
| 		$content = curl_exec($ch); | ||||
| 		curl_close($ch); | ||||
|  | ||||
| 		$output = json_decode($content, true); | ||||
| 		if ($output === null) throw new Exception('Invalid response: ' . $content . "\n\nCommand: " . $cmd . "\n"); | ||||
| 		if (isset($output['error'])) throw new Exception('API error: ' . $content . "\n\nCommand: " . $cmd . "\n"); | ||||
|  | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	public function login($email, $password, $clientId) { | ||||
| 		$method = 'POST'; | ||||
| 		$path = 'sessions'; | ||||
| 		$data = array( | ||||
| 			'email' => $email, | ||||
| 			'password' => $password, | ||||
| 			'client_id' => $clientId, | ||||
| 		); | ||||
|  | ||||
| 		return $this->exec($method, $path, null, $data); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| class Config { | ||||
|  | ||||
| 	protected $dirPath = null; | ||||
|  | ||||
| 	public function __construct($dirPath) { | ||||
| 		$this->dirPath = $dirPath; | ||||
| 	} | ||||
|  | ||||
| 	protected function load() { | ||||
| 		$c = @file_get_contents($this->dirPath . '/config.json'); | ||||
| 		$c = json_decode($c, true); | ||||
| 		if ($c === null) $c = array(); | ||||
| 		if (!isset($c['last_sync_id'])) $c['last_sync_id'] = 0; | ||||
| 		if (!isset($c['last_sync_time'])) $c['last_sync_time'] = 0; | ||||
| 		if (!isset($c['folder_items'])) $c['folder_items'] = array(); | ||||
| 		if (!isset($c['client_id'])) $c['client_id'] = null; | ||||
| 		if (!isset($c['session_id'])) $c['session_id'] = null; | ||||
| 		return $c; | ||||
| 	} | ||||
|  | ||||
| 	protected function save($c) { | ||||
| 		file_put_contents($this->dirPath . '/config.json', json_encode($c)); | ||||
| 	} | ||||
|  | ||||
| 	public function get($name) { | ||||
| 		$c = $this->load(); | ||||
| 		if (!isset($c[$name])) throw new Exception('Invalid key name: ' . $name); | ||||
| 		return $c[$name]; | ||||
| 	} | ||||
|  | ||||
| 	public function set($name, $value) { | ||||
| 		$c = $this->load(); | ||||
| 		$c[$name] = $value; | ||||
| 		$this->save($c); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| class FolderItem { | ||||
|  | ||||
| 	private $title = ''; | ||||
| 	private $body = ''; | ||||
| 	private $id; | ||||
| 	private $parentId; | ||||
| 	private $isFolder; | ||||
| 	private $modTime; | ||||
|  | ||||
| 	public function setTitle($v) { $this->title = $v; } | ||||
| 	public function setBody($v) { $this->body = $v; } | ||||
| 	public function setId($v) { $this->id = $v; } | ||||
| 	public function setParentId($v) { $this->parentId = $v; } | ||||
| 	public function setIsFolder($v) { $this->isFolder = $v; } | ||||
| 	public function setModTime($v) { $this->modTime = $v; } | ||||
|  | ||||
| 	public function title() { return $this->title; } | ||||
| 	public function body() { return $this->body; } | ||||
| 	public function id() { return $this->id; } | ||||
| 	public function parentId() { return $this->parentId; } | ||||
| 	public function isFolder() { return $this->isFolder; } | ||||
| 	public function isNote() { return !$this->isFolder(); } | ||||
| 	public function modTime() { return $this->modTime; } | ||||
|  | ||||
| 	public function toApiArray() { | ||||
| 		$output = array( | ||||
| 			'title' => $this->title(), | ||||
| 			'parent_id' => $this->parentId(), | ||||
| 		); | ||||
| 		if ($this->isNote()) $output['body'] = $this->body(); | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	public function fromApiArray($type, $array) { | ||||
| 		$this->setTitle($array['title']); | ||||
| 		if ($type == 'note') $this->setBody($array['body']); | ||||
| 		$this->setId($array['id']); | ||||
| 		$this->setParentId($array['parent_id']); | ||||
| 		$this->setIsFolder($type == 'folder'); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| class FolderItems { | ||||
|  | ||||
| 	private $items = array(); | ||||
|  | ||||
| 	private function getFolderItems($dir, $parentId, &$output) { | ||||
| 		$paths = glob($dir . '/*'); | ||||
| 		foreach ($paths as $path) { | ||||
| 			$isFolder = is_dir($path); | ||||
| 			$modTime = filemtime($path); | ||||
|  | ||||
| 			$o = new FolderItem(); | ||||
| 			$o->setTitle(basename($path)); | ||||
| 			$o->setId(Api::createId($parentId . '_' . $o->title())); | ||||
| 			$o->setParentId($parentId); | ||||
| 			$o->setIsFolder($isFolder); | ||||
| 			$o->setModTime($modTime); | ||||
|  | ||||
| 			if (!$isFolder) $o->setBody(file_get_contents($path)); | ||||
| 			$output[] = $o; | ||||
| 			if ($isFolder) $this->getFolderItems($path, $o->id(), $output); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public function fromPath($path) { | ||||
| 		$this->items = array(); | ||||
| 		$this->getFolderItems($path, null, $this->items); | ||||
| 	} | ||||
|  | ||||
| 	public function all() { | ||||
| 		return $this->items; | ||||
| 	} | ||||
|  | ||||
| 	public function add($item) { | ||||
| 		$this->items[] = $item; | ||||
| 	} | ||||
|  | ||||
| 	public function setById($id, $item) { | ||||
| 		$found = false; | ||||
| 		for ($i = 0; $i < count($this->items); $i++) { | ||||
| 			$it = $this->items[$i]; | ||||
| 			if ($it->id() == $id) { | ||||
| 				$found = true; | ||||
| 				$this->items[$i] = $item; | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (!$found) { | ||||
| 			$this->items[] = $item; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public function byId($id) { | ||||
| 		foreach ($this->all() as $item) { | ||||
| 			if ($item->id() == $id) return $item; | ||||
| 		} | ||||
| 		return null; | ||||
| 	} | ||||
|  | ||||
| 	public function itemFullPath($item) { | ||||
| 		if (!$item->parentId()) return $item->title(); | ||||
| 		$parent = $this->byId($item->parentId()); | ||||
| 		if (!$parent) throw new Exception('Cannot find parent with ID ' . $item->parentId()); | ||||
| 		return escapePath($this->itemFullPath($parent) . '/' . $item->title()); | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| $shortopts = ""; | ||||
| $longopts = array( | ||||
|     "config:", | ||||
|     "sync", | ||||
| ); | ||||
|  | ||||
| $flags = getopt($shortopts, $longopts); | ||||
|  | ||||
| if (!isset($flags['config'])) $flags['config'] = '/home/laurent/src/notes/cli-client/.config'; | ||||
|  | ||||
| $config = new Config($flags['config']); | ||||
|  | ||||
| $dataPath = '/home/laurent/src/notes/cli-client/test_' . $config->get('client_id'); | ||||
|  | ||||
| $api = new Api('http://127.0.0.1:8000'); | ||||
| $session = $api->login('test@example.com', '12345678', $config->get('client_id')); | ||||
| $api->setSessionId($session['id']); | ||||
|  | ||||
| if (array_key_exists('sync', $flags)) { | ||||
| 	$syncStartTime = time(); | ||||
| 	$lastSyncTime = $config->get('last_sync_time'); | ||||
| 	$folderItems = new FolderItems(); | ||||
| 	$folderItems->fromPath($dataPath); | ||||
|  | ||||
| 	// ------------------------------------------------------------------------------------------ | ||||
| 	// Get latest changes from API | ||||
| 	// ------------------------------------------------------------------------------------------ | ||||
|  | ||||
| 	$response = $api->exec('GET', 'synchronizer', array('last_id' => $config->get('last_sync_id'))); | ||||
| 	// $response = $api->exec('GET', 'synchronizer', array('last_id' => 80)); | ||||
|  | ||||
| 	$pathMap = array(); | ||||
| 	$folders = array(); | ||||
| 	$notes = array(); | ||||
| 	$maxId = null; | ||||
| 	foreach ($response['items'] as $item) { | ||||
| 		$folderItem = new FolderItem(); | ||||
|  | ||||
| 		switch ($item['type']) { | ||||
|  | ||||
| 			case 'create': | ||||
| 			case 'update': | ||||
|  | ||||
| 				$resource = $api->exec('GET', $item['item_type'] . 's/' . $item['item_id']); | ||||
| 				$folderItem->fromApiArray($item['item_type'], $resource); | ||||
| 				break; | ||||
|  | ||||
| 			default: | ||||
|  | ||||
| 				throw new Exception('Unsupported action type: ' . $item['type']); | ||||
|  | ||||
| 		} | ||||
|  | ||||
| 		$folderItems->setById($folderItem->id(), $folderItem); | ||||
|  | ||||
| 		$maxId = max($item['id'], $maxId); | ||||
| 	} | ||||
|  | ||||
| 	foreach ($folderItems->all() as $item) { | ||||
| 		$relativePath = $folderItems->itemFullPath($item); | ||||
| 		$path = $dataPath . '/' . $relativePath; | ||||
|  | ||||
| 		foreach (array('folder', 'note') as $itemType) { | ||||
| 			if ($item->isFolder() && $itemType == 'folder') { | ||||
| 				@mkdir($path, 0755, true); // Ignore "File exists" warning | ||||
| 				if (!is_dir($path)) throw new Exception('Could not create folder at ' . $path); | ||||
| 			} | ||||
|  | ||||
| 			if ($item->isNote() && $itemType == 'note') { | ||||
| 				if ($item->body() !== file_get_contents($path)) { | ||||
| 					file_put_contents($path, $item->body()); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		$pathMap[$item->id()] = $relativePath; | ||||
| 	} | ||||
|  | ||||
| 	// ------------------------------------------------------------------------------------------ | ||||
| 	// Send changed notes and folders to API | ||||
| 	// ------------------------------------------------------------------------------------------ | ||||
|  | ||||
| 	foreach ($folderItems->all() as $item) { | ||||
| 		if ($item->modTime() < $lastSyncTime) continue; | ||||
|  | ||||
| 		if ($item->isFolder()) { | ||||
| 			$api->exec('PUT', 'folders/' . $item->id(), null, $item->toApiArray()); | ||||
| 		} else { | ||||
| 			$api->exec('PUT', 'notes/' . $item->id(), null, $item->toApiArray()); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	$config->set('last_sync_time', $syncStartTime); | ||||
| 	$config->set('folder_items', json_encode($pathMap)); | ||||
| 	if ($maxId !== null) $config->set('last_sync_id', $maxId); | ||||
| } | ||||
| @@ -0,0 +1,3 @@ | ||||
| # FOLDER2/note1 | ||||
|  | ||||
| Modified on client 2 | ||||
| @@ -0,0 +1 @@ | ||||
| # NOTE 1 | ||||
| @@ -0,0 +1 @@ | ||||
| # NOTE 2 | ||||
| @@ -0,0 +1 @@ | ||||
| test subnode | ||||
| @@ -0,0 +1,3 @@ | ||||
| # FOLDER2/note1 | ||||
|  | ||||
| Modified on client 2 | ||||
| @@ -0,0 +1 @@ | ||||
| # NOTE 1 | ||||
| @@ -0,0 +1 @@ | ||||
| # NOTE 2 | ||||
| @@ -0,0 +1 @@ | ||||
| test subnode | ||||
							
								
								
									
										66
									
								
								composer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								composer.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| { | ||||
|     "name": "laurent/notes", | ||||
|     "license": "proprietary", | ||||
|     "type": "project", | ||||
|     "autoload": { | ||||
|         "psr-4": { | ||||
|             "": "src/" | ||||
|         }, | ||||
|         "classmap": [ | ||||
|             "app/AppKernel.php", | ||||
|             "app/AppCache.php" | ||||
|         ] | ||||
|     }, | ||||
|     "autoload-dev": { | ||||
|         "psr-4": { | ||||
|             "Tests\\": "tests/" | ||||
|         } | ||||
|     }, | ||||
|     "require": { | ||||
|         "php": ">=5.5.9", | ||||
|         "symfony/symfony": "3.1.*", | ||||
|         "doctrine/orm": "^2.5", | ||||
|         "doctrine/doctrine-bundle": "^1.6", | ||||
|         "doctrine/doctrine-cache-bundle": "^1.2", | ||||
|         "symfony/swiftmailer-bundle": "^2.3", | ||||
|         "symfony/monolog-bundle": "^2.8", | ||||
|         "symfony/polyfill-apcu": "^1.0", | ||||
|         "sensio/distribution-bundle": "^5.0", | ||||
|         "sensio/framework-extra-bundle": "^3.0.2", | ||||
|         "incenteev/composer-parameter-handler": "^2.0", | ||||
|  | ||||
|         "illuminate/database": "*", | ||||
|         "yetanotherape/diff-match-patch": "*" | ||||
|     }, | ||||
|     "require-dev": { | ||||
|         "sensio/generator-bundle": "^3.0", | ||||
|         "symfony/phpunit-bridge": "^3.0" | ||||
|     }, | ||||
|     "scripts": { | ||||
|         "symfony-scripts": [ | ||||
|             "Incenteev\\ParameterHandler\\ScriptHandler::buildParameters", | ||||
|             "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::buildBootstrap", | ||||
|             "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::clearCache", | ||||
|             "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installAssets", | ||||
|             "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::installRequirementsFile", | ||||
|             "Sensio\\Bundle\\DistributionBundle\\Composer\\ScriptHandler::prepareDeploymentTarget" | ||||
|         ], | ||||
|         "post-install-cmd": [ | ||||
|             "@symfony-scripts" | ||||
|         ], | ||||
|         "post-update-cmd": [ | ||||
|             "@symfony-scripts" | ||||
|         ] | ||||
|     }, | ||||
|     "extra": { | ||||
|         "symfony-app-dir": "app", | ||||
|         "symfony-bin-dir": "bin", | ||||
|         "symfony-var-dir": "var", | ||||
|         "symfony-web-dir": "web", | ||||
|         "symfony-tests-dir": "tests", | ||||
|         "symfony-assets-install": "relative", | ||||
|         "incenteev-parameters": { | ||||
|             "file": "app/config/parameters.yml" | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										2386
									
								
								composer.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										2386
									
								
								composer.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										11
									
								
								notes.sublime-project
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								notes.sublime-project
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| { | ||||
| 	"folders": | ||||
| 	[ | ||||
| 		{ | ||||
| 			"path": ".", | ||||
| 			"folder_exclude_patterns": [ | ||||
| 				"var" | ||||
| 			] | ||||
| 		} | ||||
| 	] | ||||
| } | ||||
							
								
								
									
										31
									
								
								phpunit.xml.dist
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								phpunit.xml.dist
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
|  | ||||
| <!-- https://phpunit.de/manual/current/en/appendixes.configuration.html --> | ||||
| <phpunit xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:noNamespaceSchemaLocation="http://schema.phpunit.de/4.8/phpunit.xsd" | ||||
|          backupGlobals="false" | ||||
|          colors="true" | ||||
|          bootstrap="app/autoload.php" | ||||
| > | ||||
|     <php> | ||||
|         <ini name="error_reporting" value="-1" /> | ||||
|         <server name="KERNEL_DIR" value="app/" /> | ||||
|     </php> | ||||
|  | ||||
|     <testsuites> | ||||
|         <testsuite name="Project Test Suite"> | ||||
|             <directory>tests</directory> | ||||
|         </testsuite> | ||||
|     </testsuites> | ||||
|  | ||||
|     <filter> | ||||
|         <whitelist> | ||||
|             <directory>src</directory> | ||||
|             <exclude> | ||||
|                 <directory>src/*Bundle/Resources</directory> | ||||
|                 <directory>src/*/*Bundle/Resources</directory> | ||||
|                 <directory>src/*/Bundle/*Bundle/Resources</directory> | ||||
|             </exclude> | ||||
|         </whitelist> | ||||
|     </filter> | ||||
| </phpunit> | ||||
							
								
								
									
										2
									
								
								run_tests.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										2
									
								
								run_tests.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| #!/bin/bash | ||||
| phpunit --bootstrap vendor/autoload.php tests/Model/ | ||||
							
								
								
									
										7
									
								
								src/.htaccess
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/.htaccess
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| <IfModule mod_authz_core.c> | ||||
|     Require all denied | ||||
| </IfModule> | ||||
| <IfModule !mod_authz_core.c> | ||||
|     Order deny,allow | ||||
|     Deny from all | ||||
| </IfModule> | ||||
							
								
								
									
										9
									
								
								src/AppBundle/AppBundle.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/AppBundle/AppBundle.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle; | ||||
|  | ||||
| use Symfony\Component\HttpKernel\Bundle\Bundle; | ||||
|  | ||||
| class AppBundle extends Bundle | ||||
| { | ||||
| } | ||||
							
								
								
									
										197
									
								
								src/AppBundle/Controller/ApiController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										197
									
								
								src/AppBundle/Controller/ApiController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,197 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Controller; | ||||
|  | ||||
| use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\Controller; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use Symfony\Component\HttpFoundation\JsonResponse; | ||||
| use AppBundle\Model\BaseModel; | ||||
| use AppBundle\Model\Session; | ||||
| use AppBundle\Model\User; | ||||
| use AppBundle\Exception\ForbiddenException; | ||||
| use AppBundle\Exception\UnauthorizedException; | ||||
| use Illuminate\Database\Eloquent\Collection; | ||||
| use AppBundle\Exception\BaseException; | ||||
|  | ||||
| abstract class ApiController extends Controller { | ||||
|  | ||||
| 	protected $db = null; | ||||
| 	protected $session = null; | ||||
| 	protected $user = null; | ||||
|  | ||||
| 	private $useTestUserAndSession = true; | ||||
| 	private $testClientNum = 1; | ||||
|  | ||||
| 	public function setContainer(\Symfony\Component\DependencyInjection\ContainerInterface $container = null) { | ||||
| 		parent::setContainer($container); | ||||
|  | ||||
| 		set_exception_handler(function($e) { | ||||
| 			if ($e instanceof BaseException) { | ||||
| 				$r = $e->toJsonResponse(); | ||||
| 				$r->send(); | ||||
| 				echo "\n"; | ||||
| 			} else { | ||||
| 				$msg = array(); | ||||
| 				$msg[] = 'Exception: ' . $e->getMessage() . ' at ' . $e->getFile() . ':' . $e->getLine(); | ||||
| 				$msg[] = ''; | ||||
| 				$msg[] = $e->getTraceAsString(); | ||||
| 				echo implode("\n", $msg); | ||||
| 				echo "\n"; | ||||
| 			} | ||||
| 		}); | ||||
|  | ||||
| 		// 1. client 1 : bla bla bla | ||||
| 		// 2. client 2 : bla bla bla fromclient2 | ||||
| 		// 3. client 1 : client1bla bla bla | ||||
|  | ||||
| 		// RESULT: client1bla bla bla | ||||
|  | ||||
| 		// Because diff for 3 is done between 2 and 3 | ||||
| 		// Need to introduce revID so that Change class knows between which versions the diff should be made | ||||
|  | ||||
| 		// HACK: get connection once here so that it's initialized and can  | ||||
| 		// be accessed from models. | ||||
| 		$this->db = $this->get('app.eloquent')->connection(); | ||||
|  | ||||
| 		$s = $this->session(); | ||||
|  | ||||
| 		// TODO: to keep it simple, only respond to logged in users, but in theory some data | ||||
| 		// could be public. | ||||
| 		if (!$s || !$this->user()) throw new UnauthorizedException('A session and user are required'); | ||||
|  | ||||
| 		BaseModel::setClientId($s ? $s->client_id : 0); | ||||
| 	} | ||||
|  | ||||
| 	protected function session() { | ||||
| 		if ($this->useTestUserAndSession) { | ||||
| 			$session = Session::find(Session::unhex('BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB')); | ||||
| 			if ($session) $session->delete(); | ||||
| 			$session = new Session(); | ||||
| 			$session->id = Session::unhex('BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB'); | ||||
| 			$session->owner_id = Session::unhex('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); | ||||
| 			$session->client_id = Session::unhex($this->testClientNum == 1 ? 'C1C1C1C1C1C1C1C1C1C1C1C1C1C1C1C1' : 'C2C2C2C2C2C2C2C2C2C2C2C2C2C2C2C2'); | ||||
| 			$session->save(); | ||||
| 			return $session; | ||||
| 		} | ||||
|  | ||||
|  | ||||
| 		if ($this->session) return $this->session; | ||||
| 		$request = $this->container->get('request_stack')->getCurrentRequest(); | ||||
| 		$this->session = Session::find(BaseModel::unhex($request->query->get('session'))); | ||||
| 		return $this->session; | ||||
| 	} | ||||
|  | ||||
| 	protected function user() { | ||||
| 		if ($this->useTestUserAndSession) { | ||||
| 			$user = User::find(User::unhex('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA')); | ||||
| 			if (!$user) { | ||||
| 				$user = new User(); | ||||
| 				$user->id = User::unhex('AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA'); | ||||
| 				$user->owner_id = $user->id; | ||||
| 				$user->email = 'test@example.com'; | ||||
| 				$user->password = Session::hashPassword('12345678'); | ||||
| 				$user->save(); | ||||
| 			} | ||||
| 			return $user; | ||||
| 		} | ||||
|  | ||||
|  | ||||
| 		if ($this->user) return $this->user; | ||||
| 		$s = $this->session(); | ||||
| 		$this->user = $s ? $s->owner() : null; | ||||
| 		return $this->user; | ||||
| 	} | ||||
|  | ||||
| 	protected function aclCheck($resource) { | ||||
| 		if (!is_array($resource)) $resource = array($resource); | ||||
| 		$user = $this->user(); | ||||
| 		if (!$user) throw new ForbiddenException(); | ||||
| 		foreach ($resource as $r) { | ||||
| 			if (!isset($r->owner_id)) continue; | ||||
| 			if ($r->owner_id != $user->id) throw new ForbiddenException(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	static private function serializeResponse($data) { | ||||
| 		$output = $data; | ||||
|  | ||||
| 		if ($output instanceof Collection) $output = $output->all(); | ||||
|  | ||||
| 		if ($output instanceof BaseModel) { | ||||
| 			$output = $output->toPublicArray(); | ||||
| 		} else if (is_array($output)) { | ||||
| 			foreach ($output as $k => $v) { | ||||
| 				$output[$k] = self::serializeResponse($v); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	static protected function successResponse($data = null) { | ||||
| 		$output = self::serializeResponse($data); | ||||
| 		return new JsonResponse($output); | ||||
| 	} | ||||
|  | ||||
| 	static protected function errorResponse($message, $errorCode = 0, $httpCode = 400) { | ||||
| 		$o = array('error' => $message, 'code' => $errorCode); | ||||
| 		$response = new JsonResponse($o); | ||||
| 		$response->setStatusCode($httpCode); | ||||
| 		return $response; | ||||
| 	} | ||||
|  | ||||
| 	protected function multipleValues($v) { | ||||
| 		if ($v === null || $v === false) return array(); | ||||
| 		if (strpos((string)$v, ';') === false) return array($v); | ||||
| 		return explode(';', $v); | ||||
| 	} | ||||
|  | ||||
| 	// PHP doesn't parse PATCH and PUT requests automatically, so it needs | ||||
| 	// to be done manually. | ||||
| 	// http://stackoverflow.com/a/5488449/561309 | ||||
| 	protected function patchParameters() { | ||||
| 		$output = array(); | ||||
| 		$input = file_get_contents('php://input'); | ||||
| 		preg_match('/boundary=(.*)$/', $_SERVER['CONTENT_TYPE'], $matches); | ||||
| 		$boundary = $matches[1]; | ||||
| 		$blocks = preg_split("/-+$boundary/", $input); | ||||
| 		array_pop($blocks); | ||||
| 		foreach ($blocks as $id => $block) { | ||||
| 			if (empty($block)) continue; | ||||
|  | ||||
| 			// you'll have to var_dump $block to understand this and maybe replace \n or \r with a visibile char | ||||
|  | ||||
| 			// parse uploaded files | ||||
| 			if (strpos($block, 'application/octet-stream') !== FALSE) { | ||||
| 				// match "name", then everything after "stream" (optional) except for prepending newlines  | ||||
| 				preg_match("/name=\"([^\"]*)\".*stream[\n|\r]+([^\n\r].*)?$/s", $block, $matches); | ||||
| 			} else { | ||||
| 				// match "name" and optional value in between newline sequences | ||||
| 				preg_match('/name=\"([^\"]*)\"[\n|\r]+([^\n\r].*)?\r$/s', $block, $matches); | ||||
| 			} | ||||
| 			if (!isset($matches[2])) { | ||||
| 				// Regex above will not find anything if the parameter has not value. For example | ||||
| 				// "parent_id" below: | ||||
|  | ||||
| 				// Content-Disposition: form-data; name="parent_id" | ||||
| 				// | ||||
| 				// | ||||
| 				// Content-Disposition: form-data; name="id" | ||||
| 				// | ||||
| 				// 54ad197be333c98778c7d6f49506efcb | ||||
|  | ||||
| 				$output[$matches[1]] = ''; | ||||
| 			} else { | ||||
| 				$output[$matches[1]] = $matches[2]; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	protected function putParameters() { | ||||
| 		return $this->patchParameters(); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										81
									
								
								src/AppBundle/Controller/FoldersController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								src/AppBundle/Controller/FoldersController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Controller; | ||||
|  | ||||
| use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\Controller; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use AppBundle\Controller\ApiController; | ||||
| use AppBundle\Model\Folder; | ||||
|  | ||||
| class FoldersController extends ApiController { | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/folders") | ||||
| 	 */ | ||||
| 	public function allAction(Request $request) { | ||||
| 		if ($request->isMethod('POST')) { | ||||
| 			$folder = new Folder(); | ||||
| 			$folder->fromPublicArray($request->request->all()); | ||||
| 			$folder->save(); | ||||
| 			return static::successResponse($folder); | ||||
| 		} | ||||
|  | ||||
| 		return static::errorResponse('Invalid method'); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/folders/{id}") | ||||
| 	 */ | ||||
| 	public function oneAction($id, Request $request) { | ||||
| 		$folder = Folder::find(Folder::unhex($id)); | ||||
| 		if (!$folder && !$request->isMethod('PUT')) return static::errorResponse('Not found', 0, 404); | ||||
|  | ||||
| 		if ($request->isMethod('GET')) { | ||||
| 			return static::successResponse($folder); | ||||
| 		} | ||||
|  | ||||
| 		if ($request->isMethod('PUT')) { | ||||
| 			// TODO: call fromPublicArray() - handles unhex conversion | ||||
|  | ||||
| 			$data = $this->putParameters(); | ||||
| 			$isNew = !$folder; | ||||
| 			if ($isNew) $folder = new Folder(); | ||||
| 			foreach ($data as $n => $v) { | ||||
| 				if ($n == 'parent_id') $v = Folder::unhex($v); | ||||
| 				$folder->{$n} = $v; | ||||
| 			} | ||||
| 			$folder->owner_id = $this->user()->id; | ||||
| 			$folder->id = Folder::unhex($id); | ||||
| 			$folder->setIsNew($isNew); | ||||
| 			$folder->save(); | ||||
| 			return static::successResponse($folder); | ||||
| 		} | ||||
|  | ||||
| 		return static::successResponse($folder); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/folders/{id}/notes") | ||||
| 	 */ | ||||
| 	public function linkAction($id, Request $request) { | ||||
| 		$folder = Folder::find(Folder::unhex($id)); | ||||
| 		if (!$folder) return static::errorResponse('Not found', 0, 404); | ||||
|  | ||||
| 		if ($request->isMethod('GET')) { | ||||
| 			return static::successResponse($folder->notes()); | ||||
| 		} | ||||
|  | ||||
| 		if ($request->isMethod('POST')) { | ||||
| 			$ids = $this->multipleValues($request->request->get('id')); | ||||
| 			if (!count($ids)) static::errorResponse('id parameter is missing'); | ||||
| 			$ids = Folder::unhex($ids); | ||||
| 			$folder->add($ids); | ||||
|  | ||||
| 			return static::successResponse(); | ||||
| 		} | ||||
|  | ||||
| 		return static::errorResponse('Invalid method'); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										71
									
								
								src/AppBundle/Controller/NotesController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/AppBundle/Controller/NotesController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Controller; | ||||
|  | ||||
| use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\Controller; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use AppBundle\Controller\ApiController; | ||||
| use AppBundle\Model\Note; | ||||
|  | ||||
| class NotesController extends ApiController { | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/notes") | ||||
| 	 */ | ||||
| 	public function allAction(Request $request) { | ||||
| 		if ($request->isMethod('POST')) { | ||||
| 			$note = new Note(); | ||||
| 			$note->fromPublicArray($request->request->all()); | ||||
| 			$note->owner_id = $this->user()->id; | ||||
| 			$note->save(); | ||||
| 			return static::successResponse($note->toPublicArray()); | ||||
| 		} | ||||
|  | ||||
| 		return static::errorResponse('Invalid method'); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/notes/{id}") | ||||
| 	 */ | ||||
| 	public function oneAction($id, Request $request) { | ||||
| 		$note = Note::find(Note::unhex($id)); | ||||
| 		if (!$note && !$request->isMethod('PUT')) return static::errorResponse('Not found', 0, 404);	 | ||||
|  | ||||
| 		if ($request->isMethod('GET')) { | ||||
| 			return static::successResponse($note); | ||||
| 		} | ||||
|  | ||||
| 		if ($request->isMethod('PUT')) { | ||||
| 			$data = $this->putParameters(); | ||||
| 			$isNew = !$note; | ||||
| 			if ($isNew) $note = new Note(); | ||||
| 			foreach ($data as $n => $v) { | ||||
| 				if ($n == 'parent_id') $v = Note::unhex($v); | ||||
| 				$note->{$n} = $v; | ||||
| 			} | ||||
| 			$note->id = Note::unhex($id); | ||||
| 			$note->owner_id = $this->user()->id; | ||||
| 			$note->setIsNew($isNew); | ||||
| 			$note->save(); | ||||
| 			return static::successResponse($note); | ||||
| 		} | ||||
|  | ||||
| 		if ($request->isMethod('PATCH')) { | ||||
| 			$data = $this->patchParameters(); | ||||
| 			foreach ($data as $n => $v) { | ||||
| 				$note->{$n} = $v; | ||||
| 			} | ||||
| 			$note->save(); | ||||
| 			return static::successResponse($note); | ||||
| 		} | ||||
|  | ||||
| 		if ($request->isMethod('DELETE')) { | ||||
| 			$note->delete(); | ||||
| 			return static::successResponse(); | ||||
| 		} | ||||
|  | ||||
| 		return static::errorResponse('Invalid method'); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										42
									
								
								src/AppBundle/Controller/SessionsController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								src/AppBundle/Controller/SessionsController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Controller; | ||||
|  | ||||
| use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\Controller; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use AppBundle\Controller\ApiController; | ||||
| use AppBundle\Model\User; | ||||
| use AppBundle\Model\Session; | ||||
| use AppBundle\Exception\NotFoundException; | ||||
| use AppBundle\Exception\MethodNotAllowedException; | ||||
|  | ||||
|  | ||||
| use AppBundle\Model\Action; | ||||
|  | ||||
| class SessionsController extends ApiController { | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/sessions") | ||||
| 	 */ | ||||
| 	public function allAction(Request $request) { | ||||
| 		if ($request->isMethod('POST')) { | ||||
| 			$data = $request->request->all(); | ||||
| 			// Note: the login method will throw an exception in case of failure | ||||
| 			$session = Session::login($data['email'], $data['password'], Session::unhex($data['client_id'])); | ||||
| 			return static::successResponse($session); | ||||
| 		} | ||||
|  | ||||
| 		throw new MethodNotAllowedException(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/sessions/{id}", name="one_session") | ||||
| 	 */ | ||||
| 	public function oneAction($id, Request $request) { | ||||
| 		$session = Session::find(Session::unhex($id)); | ||||
| 		if (!$session) throw new NotFoundException(); | ||||
| 		return static::successResponse($session); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										114
									
								
								src/AppBundle/Controller/SynchronizerController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								src/AppBundle/Controller/SynchronizerController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Controller; | ||||
|  | ||||
| use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\Controller; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use AppBundle\Controller\ApiController; | ||||
| use AppBundle\Model\Action; | ||||
| use AppBundle\Exception\UnauthorizedException; | ||||
|  | ||||
| /* | ||||
|  | ||||
| JS | ||||
|  | ||||
|  | ||||
| class Session | ||||
| 	::login() | ||||
|  | ||||
| class Synchronizer | ||||
| 	::fromId(id) | ||||
|  | ||||
| class Note | ||||
|  | ||||
| class User | ||||
|  | ||||
|  | ||||
| HISTORY | ||||
|  | ||||
| client_id, create, type, id => get full object from api | ||||
| update, type, id => get full object from api | ||||
| delete, type, id => remove object | ||||
|  | ||||
| Revisions | ||||
| ---------------- | ||||
| id | ||||
| client_id | ||||
| (user_id ?) | ||||
| action (create, update, delete) | ||||
| item_type (note, folder) | ||||
| item_field (title, body, completed...) | ||||
| item_id | ||||
|  | ||||
| if current client ID = revision.client_id - skip | ||||
|  | ||||
| Current client id = 123 | ||||
|  | ||||
| Client ID | Action | Item ID | ||||
| ------------------------------------ | ||||
| 456         delete   777 | ||||
| 123         update   777 - conflict - move to folder - note that deleted by X and then modified by Y | ||||
|  | ||||
|  | ||||
| Client ID | Action | Item ID         | Rev ID | ||||
| ------------------------------------------------------ | ||||
| 456         update   777             | 2 | ||||
| 123         update   777 - conflict  | 3 | ||||
|  | ||||
| Find rev 1 and do three way merge with 2 and 3 - means there's a need to store history of all versions of body/title | ||||
|  | ||||
|  | ||||
|  | ||||
| // Each item should have a revision ID. | ||||
| // When processing revisions => if item revision = revision.id - skip | ||||
|  | ||||
|  | ||||
| API CALL | ||||
|  | ||||
| /synchronizer/?last_id=<last_id_that_caller_synched_to> | ||||
|  | ||||
| SIMPLE IMPLEMENTATION: | ||||
|  | ||||
| loop through changes | ||||
| create list of actions: | ||||
| 	create one action per item ID / field (by merging all into one) | ||||
|  | ||||
| loop through actions | ||||
| skip current client id = action.client_id | ||||
|  | ||||
| send back list: | ||||
|  | ||||
| { | ||||
| 	sync_id: 132456, | ||||
| 	notes: [ | ||||
| 		{ | ||||
| 			action: 'update', | ||||
| 			field: 'body', | ||||
| 			body: 'blabla' | ||||
| 		} | ||||
| 	], | ||||
| 	has_more: true | ||||
| } | ||||
|  | ||||
| If has_more, resume with last sync_id | ||||
|  | ||||
|  | ||||
|  | ||||
| */ | ||||
|  | ||||
| class SynchronizerController extends ApiController { | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/synchronizer") | ||||
| 	 */ | ||||
| 	public function allAction(Request $request) { | ||||
| 		$id = (int)$request->query->get('last_id'); | ||||
|  | ||||
| 		if (!$this->user() || !$this->session()) throw new UnauthorizedException(); | ||||
|  | ||||
| 		$actions = Action::actionsDoneAfterId($this->user()->id, $this->session()->client_id, $id); | ||||
| 		return static::successResponse($actions); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										209
									
								
								src/AppBundle/Controller/UsersController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										209
									
								
								src/AppBundle/Controller/UsersController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,209 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Controller; | ||||
|  | ||||
| use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route; | ||||
| use Symfony\Bundle\FrameworkBundle\Controller\Controller; | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| use AppBundle\Controller\ApiController; | ||||
| use AppBundle\Model\User; | ||||
| use AppBundle\Model\Session; | ||||
| use AppBundle\Model\Change; | ||||
| use AppBundle\Model\FolderItem; | ||||
| use AppBundle\Exception\ValidationException; | ||||
|  | ||||
|  | ||||
| use AppBundle\Diff; | ||||
|  | ||||
| use DiffMatchPatch\DiffMatchPatch; | ||||
|  | ||||
|  | ||||
|  | ||||
| class UsersController extends ApiController { | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/users") | ||||
| 	 */ | ||||
| 	public function allAction(Request $request) { | ||||
|  | ||||
|  | ||||
|  | ||||
| 		$source = "This is the first line.\n\nThis is the second line."; | ||||
| 		$target1 = "This is the first line XXX.\n\nThis is the second line."; | ||||
| 		$target2 = "This is the first line.\n\nThis is the second line YYY."; | ||||
|  | ||||
| 		$r = Diff::merge3($source, $target1, $target2); | ||||
| 		var_dump($r);die(); | ||||
|  | ||||
|  | ||||
| 		// $dmp = new DiffMatchPatch(); | ||||
| 		// $patches = $dmp->patch_make($source, $target1); | ||||
| 		// // @@ -1,11 +1,12 @@ | ||||
| 		// //  Th | ||||
| 		// // -e | ||||
| 		// // +at | ||||
| 		// //   quick b | ||||
| 		// // @@ -22,18 +22,17 @@ | ||||
| 		// //  jump | ||||
| 		// // -s | ||||
| 		// // +ed | ||||
| 		// //   over | ||||
| 		// // -the | ||||
| 		// // +a | ||||
| 		// //   laz | ||||
| 		// $result = $dmp->patch_apply($patches, $target2); | ||||
| 		// var_dump($result); | ||||
| 		// die(); | ||||
|  | ||||
| 		// $dmp = new DiffMatchPatch(); | ||||
|  | ||||
| 		// $source = "This is the first line.\n\nThis is the second line."; | ||||
| 		// $target1 = "This is the first line XXX.\n\nThis is the second line."; | ||||
| 		// $target2 = "edsùfrklq lkzerlmk zemlkrmzlkerm lze."; | ||||
|  | ||||
|  | ||||
| 		// $diff1 = $dmp->patch_make($source, $target1); | ||||
| 		// $diff2 = $dmp->patch_make($source, $target2); | ||||
|  | ||||
| 		// //var_dump($dmp->patch_toText($diff1)); | ||||
| 		// // //var_dump($diff1[0]->patch_toText()); | ||||
|  | ||||
| 		// $r = $dmp->patch_apply($diff1, $source); | ||||
| 		// $r = $dmp->patch_apply($diff1, $target2); | ||||
| 		// var_dump($r);die(); | ||||
|  | ||||
| 		// $r = $dmp->patch_apply($diff2, $r[0]); | ||||
|  | ||||
| 		// var_dump($r); | ||||
|  | ||||
|  | ||||
| 		// $dmp = new DiffMatchPatch(); | ||||
| 		// $patches = $dmp->patch_make($source, $target1); | ||||
| 		// // @@ -1,11 +1,12 @@ | ||||
| 		// //  Th | ||||
| 		// // -e | ||||
| 		// // +at | ||||
| 		// //   quick b | ||||
| 		// // @@ -22,18 +22,17 @@ | ||||
| 		// //  jump | ||||
| 		// // -s | ||||
| 		// // +ed | ||||
| 		// //   over | ||||
| 		// // -the | ||||
| 		// // +a | ||||
| 		// //   laz | ||||
| 		// $result = $dmp->patch_apply($patches, $target2); | ||||
| 		// var_dump($result); | ||||
|  | ||||
| 		// die(); | ||||
|  | ||||
| 		// $r = Diff::merge($source, $target1, $target2); | ||||
| 		// var_dump($r);die(); | ||||
|  | ||||
| 		// $diff1 = xdiff_string_diff($source, $target1); | ||||
| 		// $diff2 = xdiff_string_diff($source, $target2); | ||||
|  | ||||
| 		// $errors = array(); | ||||
| 		// $t = xdiff_string_merge3($source , $target1, $target2, $errors); | ||||
| 		// var_dump($errors); | ||||
| 		// var_dump($t);die(); | ||||
|  | ||||
| 		// var_dump($diff1); | ||||
| 		// var_dump($diff2); | ||||
|  | ||||
| 		// $errors = array(); | ||||
| 		// $t = xdiff_string_patch($source, $diff1, XDIFF_PATCH_NORMAL, $errors); | ||||
| 		// var_dump($t); | ||||
| 		// var_dump($errors); | ||||
|  | ||||
| 		// $errors = array(); | ||||
| 		// $t = xdiff_string_patch($t, $diff2, XDIFF_PATCH_NORMAL, $errors); | ||||
| 		// var_dump($t); | ||||
| 		// var_dump($errors); | ||||
|  | ||||
|  | ||||
|  | ||||
| 		// var_dump($diff1); | ||||
| 		// var_dump($diff2); | ||||
|  | ||||
| 		// $change = new Change(); | ||||
| 		// $change->user_id = FolderItem::unhex('204705F2E2E698036034FDC709840B80'); | ||||
| 		// $change->client_id = FolderItem::unhex('11111111111111111111111111111111'); | ||||
| 		// $change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		// $change->item_field = FolderItem::enumId('field', 'title'); | ||||
| 		// $change->item_id = FolderItem::unhex('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); | ||||
| 		// $change->delta = 'salut ca va'; | ||||
| 		// $change->save(); | ||||
|  | ||||
|  | ||||
| 		// $change = new Change(); | ||||
| 		// $change->user_id = FolderItem::unhex('204705F2E2E698036034FDC709840B80'); | ||||
| 		// $change->client_id = FolderItem::unhex('11111111111111111111111111111111'); | ||||
| 		// $change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		// $change->item_field = FolderItem::enumId('field', 'title'); | ||||
| 		// $change->item_id = FolderItem::unhex('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); | ||||
| 		// $change->createDelta('salut, ça va ? oui très bien'); | ||||
| 		// $change->save(); | ||||
|  | ||||
| 		// $change = new Change(); | ||||
| 		// $change->user_id = FolderItem::unhex('204705F2E2E698036034FDC709840B80'); | ||||
| 		// $change->client_id = FolderItem::unhex('11111111111111111111111111111111'); | ||||
| 		// $change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		// $change->item_field = FolderItem::enumId('field', 'title'); | ||||
| 		// $change->item_id = FolderItem::unhex('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); | ||||
| 		// $change->createDelta('salut - oui très bien'); | ||||
| 		// $change->save(); | ||||
|  | ||||
| 		// $change = new Change(); | ||||
| 		// $change->user_id = FolderItem::unhex('204705F2E2E698036034FDC709840B80'); | ||||
| 		// $change->client_id = FolderItem::unhex('11111111111111111111111111111111'); | ||||
| 		// $change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		// $change->item_field = FolderItem::enumId('field', 'title'); | ||||
| 		// $change->item_id = FolderItem::unhex('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'); | ||||
| 		// $change->createDelta('salut, ça va ? oui bien'); | ||||
| 		// $change->save(); | ||||
|  | ||||
|  | ||||
|  | ||||
| 		$d = Change::fullFieldText(FolderItem::unhex('DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD'), FolderItem::enumId('field', 'title')); | ||||
| 		var_dump($d);die(); | ||||
|  | ||||
|  | ||||
|  | ||||
| 		die(); | ||||
|  | ||||
|  | ||||
| 		// $fineDiff = $this->get('app.fine_diff'); | ||||
| 		// $opcodes = $fineDiff->getDiffOpcodes('salut ca va', 'salut va?'); | ||||
| 		// var_dump($opcodes); | ||||
| 		// $merged = $fineDiff->renderToTextFromOpcodes('salut ca va', $opcodes); | ||||
| 		// var_dump($merged); | ||||
| 		// die(); | ||||
|  | ||||
| 		if ($request->isMethod('POST')) { | ||||
| 			$user = new User(); | ||||
| 			$data = $request->request->all(); | ||||
|  | ||||
| 			$errors = User::validate($data); | ||||
| 			if (count($errors)) throw ValidationException::fromErrors($errors); | ||||
|  | ||||
| 			$data['password'] = Session::hashPassword($data['password']); | ||||
| 			$user->fromPublicArray($data); | ||||
| 			$user->save(); | ||||
| 			return static::successResponse($user->toPublicArray()); | ||||
| 		} | ||||
|  | ||||
| 		return static::errorResponse('Invalid method'); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * @Route("/users/{id}") | ||||
| 	 */ | ||||
| 	public function oneAction($id, Request $request) { | ||||
| 		$user = User::find(User::unhex($id)); | ||||
| 		if (!$user) return static::errorResponse('Not found', 0, 404); | ||||
| 		$this->aclCheck($user); | ||||
| 		return static::successResponse($user); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										33
									
								
								src/AppBundle/Diff.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								src/AppBundle/Diff.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle; | ||||
|  | ||||
| use DiffMatchPatch\DiffMatchPatch; | ||||
|  | ||||
| class Diff { | ||||
|  | ||||
| 	static private $dmp_ = null; | ||||
|  | ||||
| 	static private function dmp() { | ||||
| 		if (self::$dmp_) return self::$dmp_; | ||||
| 		self::$dmp_ = new DiffMatchPatch(); | ||||
| 		return self::$dmp_; | ||||
| 	} | ||||
|  | ||||
| 	static public function diff($from, $to) { | ||||
| 		return self::dmp()->patch_toText(self::dmp()->patch_make($from, $to)); | ||||
| 	} | ||||
|  | ||||
| 	static public function patch($from, $diff) { | ||||
| 		return self::dmp()->patch_apply(self::dmp()->patch_fromText($diff), $from); | ||||
| 	} | ||||
|  | ||||
| 	// Basically applies diff(orginal, mod1) to mod2. Note sure if this is really | ||||
| 	// equivalent to a three-way merge. | ||||
| 	static public function merge3($original, $modified1, $modified2) { | ||||
| 		$patches = self::dmp()->patch_make($original, $modified1); | ||||
| 		return self::dmp()->patch_apply($patches, $modified2); | ||||
| 	} | ||||
|  | ||||
|  | ||||
| } | ||||
							
								
								
									
										30
									
								
								src/AppBundle/Eloquent.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/AppBundle/Eloquent.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle; | ||||
|  | ||||
| class Eloquent { | ||||
|  | ||||
| 	private $capsule_ = null; | ||||
|  | ||||
| 	public function __construct() { | ||||
| 		$this->capsule_ = new \Illuminate\Database\Capsule\Manager(); | ||||
|  | ||||
| 		$this->capsule_->addConnection([ | ||||
| 			'driver'    => 'mysql', | ||||
| 			'host'      => 'localhost', | ||||
| 			'database'  => 'notes', | ||||
| 			'username'  => 'root', | ||||
| 			'password'  => 'pass', | ||||
| 			'charset'   => 'utf8', | ||||
| 			'collation' => 'utf8_unicode_ci', | ||||
| 			'prefix'    => '', | ||||
| 		]); | ||||
|  | ||||
| 		$this->capsule_->bootEloquent(); | ||||
| 	} | ||||
|  | ||||
| 	public function connection() { | ||||
| 		return $this->capsule_->getConnection('default'); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										9
									
								
								src/AppBundle/Exception/AuthException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/AppBundle/Exception/AuthException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| class AuthException extends BaseException { | ||||
|  | ||||
| 	protected $message = 'invalid authentication'; | ||||
|  | ||||
| } | ||||
							
								
								
									
										36
									
								
								src/AppBundle/Exception/BaseException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								src/AppBundle/Exception/BaseException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| use Symfony\Component\HttpFoundation\JsonResponse; | ||||
|  | ||||
| class BaseException extends \Exception { | ||||
|  | ||||
| 	protected $httpStatusCode = 400; | ||||
|  | ||||
| 	public function getType() { | ||||
| 		$t = str_replace('Exception', '', get_called_class()); | ||||
| 		$t = str_replace("AppBundle\\\\", '', $t); | ||||
| 		return $t; | ||||
| 	} | ||||
|  | ||||
| 	public function getHttpStatusCode() { | ||||
| 		return $this->httpStatusCode; | ||||
| 	} | ||||
|  | ||||
| 	public function toErrorArray() { | ||||
| 		$o = array(); | ||||
| 		$o['error'] = $this->getMessage(); | ||||
| 		if ($this->getCode()) $o['code'] = $this->getCode(); | ||||
| 		$o['type'] = $this->getType(); | ||||
| 		return $o; | ||||
| 	} | ||||
|  | ||||
| 	public function toJsonResponse($errorObject = null) { | ||||
| 		if (!$errorObject) $errorObject = $this->toErrorArray(); | ||||
| 		$response = new JsonResponse($errorObject); | ||||
| 		$response->setStatusCode($this->getHttpStatusCode()); | ||||
| 		return $response; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										10
									
								
								src/AppBundle/Exception/ForbiddenException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/AppBundle/Exception/ForbiddenException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| class ForbiddenException extends BaseException { | ||||
|  | ||||
| 	protected $message = 'forbidden'; | ||||
| 	protected $httpStatusCode = 403; | ||||
|  | ||||
| } | ||||
							
								
								
									
										10
									
								
								src/AppBundle/Exception/MethodNotAllowedException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/AppBundle/Exception/MethodNotAllowedException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| class MethodNotAllowedException extends BaseException { | ||||
|  | ||||
| 	protected $message = 'method not allowed'; | ||||
| 	protected $httpStatusCode = 405; | ||||
|  | ||||
| } | ||||
							
								
								
									
										10
									
								
								src/AppBundle/Exception/NotFoundException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/AppBundle/Exception/NotFoundException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| class NotFoundException extends BaseException { | ||||
|  | ||||
| 	protected $message = 'not found'; | ||||
| 	protected $httpStatusCode = 400; | ||||
|  | ||||
| } | ||||
							
								
								
									
										10
									
								
								src/AppBundle/Exception/UnauthorizedException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/AppBundle/Exception/UnauthorizedException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| class UnauthorizedException extends BaseException { | ||||
|  | ||||
| 	protected $message = 'unauthorized'; | ||||
| 	protected $httpStatusCode = 401; | ||||
|  | ||||
| } | ||||
							
								
								
									
										23
									
								
								src/AppBundle/Exception/ValidationException.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/AppBundle/Exception/ValidationException.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Exception; | ||||
|  | ||||
| class ValidationException extends BaseException { | ||||
|  | ||||
| 	protected $message = 'validation error'; | ||||
| 	public $validationErrors = array(); | ||||
|  | ||||
| 	static public function fromErrors($errors) { | ||||
| 		if (!count($errors)) return new ValidationException(); | ||||
| 		$e = new ValidationException($errors[0]['message']); | ||||
| 		$e->validationErrors = $errors; | ||||
| 		return $e; | ||||
| 	} | ||||
|  | ||||
| 	public function toErrorArray() { | ||||
| 		$o = parent::toErrorArray(); | ||||
| 		$o['validation_errors'] = $this->validationErrors; | ||||
| 		return $o; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										28
									
								
								src/AppBundle/Model/Action.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/AppBundle/Model/Action.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| class Action extends BaseModel { | ||||
|  | ||||
| 	static protected $enums = array( | ||||
| 		'type' => array('create', 'update', 'delete'), | ||||
| 	); | ||||
|  | ||||
| 	static public function actionsDoneAfterId($userId, $clientId, $actionId) { | ||||
| 		$limit = 100; | ||||
| 		$items = self::where('id', '>', $actionId) | ||||
| 		             ->where('user_id', '=', $userId) | ||||
| 		             ->where('client_id', '!=', $clientId) | ||||
| 		             ->orderBy('id') | ||||
| 		             ->limit($limit + 1) | ||||
| 		             ->get(); | ||||
| 		$hasMore = $limit < count($items); | ||||
| 		if ($hasMore) array_pop($items); | ||||
|  | ||||
| 		return array( | ||||
| 			'has_more' => $hasMore, | ||||
| 			'items' => $items, | ||||
| 		); | ||||
| 	} | ||||
| 	 | ||||
| } | ||||
							
								
								
									
										286
									
								
								src/AppBundle/Model/BaseModel.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										286
									
								
								src/AppBundle/Model/BaseModel.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,286 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| use \Illuminate\Database\Eloquent\Model; | ||||
| use \Illuminate\Support\Facades\DB; | ||||
|  | ||||
| class BaseModel extends \Illuminate\Database\Eloquent\Model { | ||||
|  | ||||
| 	public $timestamps = false; | ||||
| 	public $useUuid = false; | ||||
| 	protected $changedVersionedFieldValues = array(); | ||||
| 	protected $versionedFields = array(); | ||||
| 	private $isNew = null; | ||||
| 	private $revId = 0; | ||||
|  | ||||
| 	static private $clientId = null; | ||||
| 	static protected $enums = array( | ||||
| 		'field' => array('title', 'body', 'completed'), | ||||
| 	); | ||||
| 	static protected $defaultValidationRules = array(); | ||||
| 	static protected $defaultValidationMessages = array( | ||||
| 		'required' => '{key} is required', | ||||
| 		'notEmpty' => '{key} cannot be empty', | ||||
| 		'minLength' => '{key} must be at least {arg0} characters long', | ||||
| 		'maxLength' => '{key} must not be longer than {arg0} characters', | ||||
| 		'function' => '{key} is invalid', | ||||
| 	); | ||||
|  | ||||
| 	public function __construct($attributes = array()) { | ||||
| 		parent::__construct($attributes); | ||||
| 	} | ||||
|  | ||||
| 	static public function setClientId($clientId) { | ||||
| 		self::$clientId = $clientId; | ||||
| 	} | ||||
|  | ||||
| 	static public function clientId() { | ||||
| 		return self::$clientId; | ||||
| 	} | ||||
|  | ||||
| 	public function fromPublicArray($array) { | ||||
| 		foreach ($array as $k => $v) { | ||||
| 			if ($k == 'rev_id') { | ||||
| 				$this->revId = $v; | ||||
| 			} else if (in_array($k, $this->versionedFields)) { | ||||
| 				$this->changedVersionedFieldValues[$k] = $v; | ||||
| 			} else { | ||||
| 				$this->{$k} = $v; | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public function toPublicArray() { | ||||
| 		$output = $this->toArray(); | ||||
| 		if ($this->useUuid) { | ||||
| 			$output['id'] = self::hex($output['id']); | ||||
| 		} | ||||
| 		 | ||||
| 		if (!empty($output['parent_id'])) $output['parent_id'] = self::hex($output['parent_id']); | ||||
| 		if (!empty($output['owner_id'])) $output['owner_id'] = self::hex($output['owner_id']); | ||||
| 		if (!empty($output['client_id'])) $output['client_id'] = self::hex($output['client_id']); | ||||
| 		if (!empty($output['item_id'])) $output['item_id'] = self::hex($output['item_id']); | ||||
| 		if (!empty($output['user_id'])) $output['user_id'] = self::hex($output['user_id']); | ||||
|  | ||||
| 		foreach ($output as $k => $v) { | ||||
| 			if (isset(static::$enums[$k])) { | ||||
| 				$output[$k] = static::enumName($k, $v); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (isset($output['item_type'])) { | ||||
| 			$output['item_type'] = FolderItem::enumName('type', $output['item_type'], true); | ||||
| 		} | ||||
|  | ||||
| 		if (isset($output['item_field'])) { | ||||
| 			$output['item_field'] = BaseModel::enumName('field', $output['item_field'], true); | ||||
| 		} | ||||
|  | ||||
| 		$maxRevId = 0; | ||||
| 		foreach ($this->versionedFields as $field) { | ||||
| 			$r = $this->versionedFieldValue($field, true); | ||||
| 			$output[$field] = $r['text']; | ||||
| 			$maxRevId = max($maxRevId, $r['revId']); | ||||
| 		} | ||||
|  | ||||
| 		$output['rev_id'] = $maxRevId; | ||||
|  | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	public function versionedFieldValue($fieldName, $returnRevId = false) { | ||||
| 		return Change::fullFieldText($this->id, BaseModel::enumId('field', $fieldName), null, $returnRevId); | ||||
| 	} | ||||
|  | ||||
| 	public function setVersionedFieldValue($fieldName, $fieldValue) { | ||||
| 		$this->changedVersionedFieldValues[$fieldName] = $fieldValue; | ||||
| 	} | ||||
|  | ||||
| 	static public function createId() { | ||||
| 		return openssl_random_pseudo_bytes(16); | ||||
| 	} | ||||
|  | ||||
| 	static public function hex($id) { | ||||
| 		if (is_array($id)) { | ||||
| 			foreach ($id as $k => $v) { | ||||
| 				$id[$k] = self::hex($v); | ||||
| 			} | ||||
| 			return $id; | ||||
| 		} | ||||
| 		return bin2hex($id); | ||||
| 	} | ||||
|  | ||||
| 	static public function unhex($s) { | ||||
| 		if (!strlen($s)) return null; | ||||
| 		 | ||||
| 		if (is_array($s)) { | ||||
| 			foreach ($s as $k => $v) { | ||||
| 				$s[$k] = self::unhex($v); | ||||
| 			} | ||||
| 			return $s; | ||||
| 		} | ||||
| 		$output = @hex2bin($s); | ||||
| 		if ($output === false) return null; | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	public function owner() { | ||||
| 		if (!isset($this->owner_id)) return null; | ||||
| 		return User::find($this->owner_id); | ||||
| 	} | ||||
|  | ||||
| 	static public function validate($data, $rules = null) { | ||||
| 		if (!$rules) $rules = static::$defaultValidationRules; | ||||
|  | ||||
| 		$errors = array(); | ||||
| 		 | ||||
| 		foreach ($rules as $key => $keyRules) { | ||||
| 			foreach ($keyRules as $rule) { | ||||
| 				$ok = true; | ||||
| 				switch ($rule['type']) { | ||||
|  | ||||
| 					case 'required': | ||||
|  | ||||
| 						if (!array_key_exists($key, $data)) $ok = false; | ||||
| 						break; | ||||
|  | ||||
| 					case 'notEmpty': | ||||
|  | ||||
| 						if (array_key_exists($key, $data) && !strlen((string)$data[$key])) $ok = false; | ||||
| 						break; | ||||
|  | ||||
| 					case 'minLength': | ||||
|  | ||||
| 						if (array_key_exists($key, $data) && strlen((string)$data[$key]) < $rule['args'][0]) $ok = false; | ||||
| 						break; | ||||
|  | ||||
| 					case 'maxLength': | ||||
|  | ||||
| 						if (array_key_exists($key, $data) && strlen((string)$data[$key]) > $rule['args'][0]) $ok = false; | ||||
| 						break; | ||||
|  | ||||
| 					case 'function': | ||||
|  | ||||
| 						$ok = call_user_func_array($rule['args'][0], array($key, $rule, $data)); | ||||
| 						break; | ||||
|  | ||||
| 					default: | ||||
|  | ||||
| 						throw new \Exception(sprintf('unsupported validation rule: "%s"', $rule['name'])); | ||||
|  | ||||
| 				} | ||||
|  | ||||
| 				if (!$ok) { | ||||
| 					$errors[] = array( | ||||
| 						'key' => $key, | ||||
| 						'type' => $rule['type'] == 'function' ? 'other' : $rule['type'], | ||||
| 						'message' => static::validationMessage($key, $rule, $data), | ||||
| 					); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return $errors; | ||||
| 	} | ||||
|  | ||||
| 	static public function validationMessage($key, $rule, $data) { | ||||
| 		$msg = static::$defaultValidationMessages[$rule['type']]; | ||||
| 		if (isset($rule['message'])) $msg = $rule['message']; | ||||
| 		$msg = str_replace('{key}', $key, $msg); | ||||
| 		$msg = str_replace('{value}', isset($data[$key]) ? $data[$key] : '', $msg); | ||||
| 		$args = isset($rule['args']) ? $rule['args'] : array(); | ||||
| 		for ($i = 0; $i < count($args); $i++) { | ||||
| 			$v = $args[$i]; | ||||
| 			if (is_array($v)) $v = ''; | ||||
| 			if (is_object($v) && !method_exists($v, '__toString')) $v = ''; | ||||
| 			$v = (string)$v; | ||||
| 			$msg = str_replace(sprintf('{arg%s}', $i), $v, $msg); | ||||
| 		} | ||||
| 		return $msg; | ||||
| 	} | ||||
|  | ||||
| 	static public function enumName($enumType, $enumId, $returnNullOnError = false) { | ||||
| 		foreach (static::$enums[$enumType] as $index => $name) { | ||||
| 			if ($index + 1 == $enumId) return $name; | ||||
| 		} | ||||
| 		if ($returnNullOnError) return null; | ||||
| 		throw new \Exception(sprintf('Invalid enum: %s/%s', $enumType, $enumId)); | ||||
| 	} | ||||
|  | ||||
| 	static public function enumId($enumType, $enumName, $returnNullOnError = false) { | ||||
| 		if (!isset(static::$enums[$enumType])) throw new \Exception(sprintf('Invalid enum type: %s', $enumType)); | ||||
| 		foreach (static::$enums[$enumType] as $index => $name) { | ||||
| 			if ($name == $enumName) return $index + 1; | ||||
| 		} | ||||
| 		if ($returnNullOnError) return null; | ||||
| 		throw new \Exception(sprintf('Invalid enum: %s/%s', $enumType, $enumName)); | ||||
| 	} | ||||
|  | ||||
| 	// Allows caller to force the model to be detected as new, | ||||
| 	// even when the model already has an ID (to handle PUT | ||||
| 	// calls that manually set the ID). | ||||
| 	public function setIsNew($v) { | ||||
| 		$this->isNew = $v; | ||||
| 	} | ||||
|  | ||||
| 	public function save(Array $options = array()) { | ||||
| 		$isNew = !$this->id || $this->isNew === true; | ||||
|  | ||||
| 		if ($this->useUuid && $isNew && !$this->id) $this->id = self::createId(); | ||||
| 		$this->updated_time = time(); // TODO: maybe only update if one of the fields, or if some of versioned data has changed | ||||
| 		if ($isNew) $this->created_time = time(); | ||||
|  | ||||
| 		parent::save($options); | ||||
|  | ||||
| 		if (count($this->versionedFields)) { | ||||
| 			$this->recordChanges($isNew ? 'create' : 'update', $this->changedVersionedFieldValues); | ||||
| 		} | ||||
| 		$this->changedVersionedFieldValues = array(); | ||||
| 	} | ||||
|  | ||||
| 	public function delete() { | ||||
| 		parent::delete(); | ||||
|  | ||||
| 		if (count($this->versionedFields)) { | ||||
| 			$this->recordChanges('delete'); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	protected function recordChanges($type, $versionedData = array()) { | ||||
| 		if ($type == 'delete') { | ||||
| 			$change = $this->newChange($type); | ||||
| 			$change->save(); | ||||
| 		} else if ($type == 'create' || $type == 'update') { | ||||
| 			foreach ($this->versionedFields as $field) { | ||||
| 				if (!isset($versionedData[$field])) continue; | ||||
|  | ||||
| 				$change = $this->newChange($type); | ||||
| 				$change->item_field = BaseModel::enumId('field', $field); | ||||
| 				$change->createDelta($versionedData[$field]); | ||||
| 				$change->save(); | ||||
| 			} | ||||
| 		} else { | ||||
| 			throw new \Exception('Unknown type: ' . $type); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	private function classItemTypeName() { | ||||
| 		$s = strtolower(get_called_class()); | ||||
| 		$s = explode("\\", $s); | ||||
| 		return $s[count($s) - 1]; | ||||
| 	} | ||||
|  | ||||
| 	private function newChange($type) { | ||||
| 		if (static::clientId() === null) throw new \Exception('Client ID must be specified'); | ||||
|  | ||||
| 		$change = new Change(); | ||||
| 		$change->user_id = $this->owner_id; | ||||
| 		$change->client_id = static::clientId(); | ||||
| 		$change->item_type = FolderItem::enumId('type', $this->classItemTypeName()); | ||||
| 		$change->type = Change::enumId('type', $type); | ||||
| 		$change->item_id = $this->id; | ||||
| 		return $change; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										62
									
								
								src/AppBundle/Model/Change.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								src/AppBundle/Model/Change.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| use AppBundle\Diff; | ||||
|  | ||||
| class Change extends BaseModel { | ||||
|  | ||||
| 	static protected $enums = array( | ||||
| 		'type' => array('create', 'update', 'delete'), | ||||
| 	); | ||||
|  | ||||
| 	static public function changesDoneAfterId($userId, $clientId, $changeId) { | ||||
| 		$limit = 100; | ||||
| 		$items = self::where('id', '>', $changeId) | ||||
| 		             ->where('user_id', '=', $userId) | ||||
| 		             ->where('client_id', '!=', $clientId) | ||||
| 		             ->orderBy('id') | ||||
| 		             ->limit($limit + 1) | ||||
| 		             ->get(); | ||||
| 		$hasMore = $limit < count($items); | ||||
| 		if ($hasMore) array_pop($items); | ||||
|  | ||||
| 		return array( | ||||
| 			'has_more' => $hasMore, | ||||
| 			'items' => $items, | ||||
| 		); | ||||
| 	} | ||||
|  | ||||
| 	static public function itemFieldHistory($itemId, $itemField, $toId = null) { | ||||
| 		$query = self::where('item_id', '=', $itemId); | ||||
| 		$query->where('item_field', '=', $itemField); | ||||
| 		if ($toId) $query->where('id', '<=', $toId); | ||||
| 		$query->orderBy('id'); | ||||
| 		return $query->get(); | ||||
| 	} | ||||
|  | ||||
| 	static public function fullFieldText($itemId, $itemField, $toId = null, $returnRevId = false) { | ||||
| 		$output = ''; | ||||
| 		$changes = self::itemFieldHistory($itemId, $itemField, $toId); | ||||
| 		$revId = 0; | ||||
| 		for ($i = 0; $i < count($changes); $i++) { | ||||
| 			$change = $changes[$i]; | ||||
| 			$result = Diff::patch($output, $change->delta); | ||||
| 			if (!count($result[1])) throw new \Exception('Unexpected result format for patch operation: ' . json_encode($result)); | ||||
| 			if (!$result[1][0]) { | ||||
| 				// Could not patch the string. TODO: handle conflict | ||||
| 			} | ||||
| 			$output = $result[0]; | ||||
|  | ||||
| 			$revId = $change->id; | ||||
| 		} | ||||
|  | ||||
| 		return $returnRevId ? array('text' => $output, 'revId' => $revId) : $output; | ||||
| 	} | ||||
|  | ||||
| 	public function createDelta($newText) { | ||||
| 		$currentText = self::fullFieldText($this->item_id, $this->item_field, $this->previous_id); | ||||
| 		$this->delta = Diff::diff($currentText, $newText); | ||||
| 	} | ||||
| 	 | ||||
| } | ||||
							
								
								
									
										21
									
								
								src/AppBundle/Model/Folder.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/AppBundle/Model/Folder.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| class Folder extends FolderItem { | ||||
|  | ||||
| 	protected $versionedFields = array('title'); | ||||
|  | ||||
| 	public function add($ids) { | ||||
| 		$notes = Note::find($ids); | ||||
| 		foreach ($notes as $note) { | ||||
| 			$note->parent_id = $this->id; | ||||
| 			$note->save(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public function notes() { | ||||
| 		return Note::where('parent_id', '=', $this->id)->get(); | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										14
									
								
								src/AppBundle/Model/FolderItem.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/AppBundle/Model/FolderItem.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| class FolderItem extends BaseModel { | ||||
|  | ||||
| 	public $useUuid = true; | ||||
| 	public $incrementing = false; | ||||
|  | ||||
| 	static protected $enums = array( | ||||
| 		'type' => array('folder', 'note', 'todo'), | ||||
| 	); | ||||
|  | ||||
| } | ||||
							
								
								
									
										9
									
								
								src/AppBundle/Model/Note.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/AppBundle/Model/Note.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| class Note extends FolderItem { | ||||
|  | ||||
| 	protected $versionedFields = array('title', 'body'); | ||||
|  | ||||
| } | ||||
							
								
								
									
										40
									
								
								src/AppBundle/Model/Session.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/AppBundle/Model/Session.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| use AppBundle\Exception\NotFoundException; | ||||
| use AppBundle\Exception\AuthException; | ||||
|  | ||||
| class Session extends BaseModel { | ||||
|  | ||||
| 	public $useUuid = true; | ||||
| 	public $incrementing = false; | ||||
|  | ||||
| 	static public function hashPassword($password) { | ||||
| 		return password_hash($password, PASSWORD_DEFAULT); | ||||
| 	} | ||||
|  | ||||
| 	static public function verifyPassword($password, $hash) { | ||||
| 		return password_verify($password, $hash); | ||||
| 	} | ||||
|  | ||||
| 	static public function passwordMinLength() { | ||||
| 		return 8; | ||||
| 	} | ||||
|  | ||||
| 	static public function login($email, $password, $clientId) { | ||||
| 		$user = User::byEmail($email); | ||||
| 		if (!$user) throw new NotFoundException(); | ||||
|  | ||||
| 		$ok = self::verifyPassword($password, $user->password); | ||||
| 		if (!$ok) throw new AuthException(); | ||||
|  | ||||
| 		$session = new Session(); | ||||
| 		$session->owner_id = $user->id; | ||||
| 		$session->client_id = $clientId; | ||||
| 		$session->save(); | ||||
|  | ||||
| 		return $session; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										52
									
								
								src/AppBundle/Model/User.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/AppBundle/Model/User.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| <?php | ||||
|  | ||||
| namespace AppBundle\Model; | ||||
|  | ||||
| use AppBundle\Exception\ValidationException; | ||||
|  | ||||
| class User extends BaseModel { | ||||
|  | ||||
| 	public $useUuid = true; | ||||
| 	public $incrementing = false; | ||||
|  | ||||
| 	public function __construct($attributes = array()) { | ||||
| 		parent::__construct($attributes); | ||||
|  | ||||
| 		static::$defaultValidationRules['email'] = array( | ||||
| 			array('type' => 'required'), | ||||
| 			array('type' => 'notEmpty'), | ||||
| 			array('type' => 'function', 'args' => array(array('AppBundle\Model\User', 'validateUniqueEmail')), 'message' => 'email "{value}" is already in use'), | ||||
| 		); | ||||
| 		static::$defaultValidationRules['password'] = array( | ||||
| 			array('type' => 'required'), | ||||
| 			array('type' => 'minLength', 'args' => array(8)), | ||||
| 		); | ||||
| 	} | ||||
|  | ||||
| 	public function toPublicArray() { | ||||
| 		$output = parent::toPublicArray(); | ||||
| 		unset($output['password']); | ||||
| 		return $output; | ||||
| 	} | ||||
|  | ||||
| 	public function byEmail($email) { | ||||
| 		return self::where('email', '=', $email)->first(); | ||||
| 	} | ||||
|  | ||||
| 	public function save(Array $options = array()) { | ||||
| 		$isNew = !$this->id; | ||||
|  | ||||
| 		parent::save($options); | ||||
| 		if ($isNew) { | ||||
| 			$this->owner_id = $this->id; | ||||
| 			parent::save($options); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	static public function validateUniqueEmail($key, $rule, $data) { | ||||
| 		if (!isset($data['email'])) return true; | ||||
| 		$u = self::byEmail($data['email']); | ||||
| 		return !$u; | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										57
									
								
								structure.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								structure.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| CREATE TABLE `folders` ( | ||||
| 	`id` binary(16) NOT NULL, | ||||
| 	`created_time` int(11) NOT NULL default '0', | ||||
| 	`updated_time` int(11) NOT NULL default '0', | ||||
| 	`parent_id` binary(16) NULL default NULL, | ||||
| 	`owner_id` binary(16) NULL default NULL, | ||||
| 	`is_encrypted` tinyint(1) NOT NULL default '0', | ||||
| 	`encryption_method` int(11) NOT NULL default '0', | ||||
| PRIMARY KEY (`id`) | ||||
| ) CHARACTER SET=utf8; | ||||
|  | ||||
| CREATE TABLE `notes` ( | ||||
| 	`id` binary(16) NOT NULL, | ||||
| 	`completed` tinyint(1) NOT NULL default '0', | ||||
| 	`created_time` int(11) NOT NULL default '0', | ||||
| 	`updated_time` int(11) NOT NULL default '0', | ||||
| 	`parent_id` binary(16) NULL default NULL, | ||||
| 	`owner_id` binary(16), | ||||
| 	`is_encrypted` tinyint(1) NOT NULL default '0', | ||||
| 	`encryption_method` int(11) NOT NULL default '0', | ||||
| PRIMARY KEY (`id`) | ||||
| ) CHARACTER SET=utf8; | ||||
|  | ||||
| CREATE TABLE `users` ( | ||||
| 	`id` binary(16) NOT NULL, | ||||
| 	`email` varchar(256) NOT NULL default '', | ||||
| 	`password` varchar(256) NOT NULL default '', | ||||
| 	`validated` tinyint(1) NOT NULL default '0', | ||||
| 	`created_time` int(11) NOT NULL default '0', | ||||
| 	`updated_time` int(11) NOT NULL default '0', | ||||
| 	`owner_id` binary(16), | ||||
| PRIMARY KEY (`id`) | ||||
| ) CHARACTER SET=utf8; | ||||
|  | ||||
| CREATE TABLE `sessions` ( | ||||
| 	`id` binary(16) NOT NULL, | ||||
| 	`owner_id` binary(16), | ||||
| 	`client_id` binary(16), | ||||
| 	`created_time` int(11) NOT NULL default '0', | ||||
| 	`updated_time` int(11) NOT NULL default '0', | ||||
| PRIMARY KEY (`id`) | ||||
| ) CHARACTER SET=utf8; | ||||
|  | ||||
| CREATE TABLE `changes` ( | ||||
| 	`id` int(11) NOT NULL AUTO_INCREMENT, | ||||
| 	`user_id` binary(16), | ||||
| 	`client_id` binary(16), | ||||
| 	`created_time` int(11) NOT NULL default '0', | ||||
| 	`updated_time` int(11) NOT NULL default '0', | ||||
| 	`type` int(11) NOT NULL default '0', | ||||
| 	`item_id` binary(16), | ||||
| 	`item_type` int(11) NOT NULL default '0', | ||||
| 	`item_field` int(11) NOT NULL default '0', | ||||
| 	`delta` MEDIUMTEXT, | ||||
| 	`previous_id` int(11) NOT NULL default '0', | ||||
| PRIMARY KEY (`id`) | ||||
| ) CHARACTER SET=utf8; | ||||
							
								
								
									
										69
									
								
								tests/BaseTestCase.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								tests/BaseTestCase.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
|  | ||||
| use PHPUnit\Framework\TestCase; | ||||
|  | ||||
| use AppBundle\Model\BaseModel; | ||||
| use AppBundle\Model\User; | ||||
| use AppBundle\Model\Session; | ||||
|  | ||||
| class BaseTestCase extends TestCase { | ||||
|  | ||||
| 	protected function createModelId($type, $num = 1) { | ||||
| 		$c = ''; | ||||
| 		if ($type == 'user') { | ||||
| 			$c = 'A'; | ||||
| 		} else if ($type == 'client') { | ||||
| 			$c = 'C'; | ||||
| 		} else if ($type == 'session') { | ||||
| 			$c = 'B'; | ||||
| 		} else if ($type == 'note') { | ||||
| 			$c = 'D'; | ||||
| 		} | ||||
| 		return BaseModel::unhex(str_repeat($c . $num, 16)); | ||||
| 	} | ||||
|  | ||||
| 	protected function clientId($num = 1) { | ||||
| 		return $this->createModelId('client', $num); | ||||
| 	} | ||||
|  | ||||
| 	protected function user($num = 1) { | ||||
| 		$id = $this->createModelId('user', $num); | ||||
| 		$user = User::find($id); | ||||
| 		if ($user) return $user; | ||||
|  | ||||
| 		$user = new User(); | ||||
| 		$user->id = $id; | ||||
| 		$user->owner_id = $user->id; | ||||
| 		$user->email = BaseModel::hex($id) . '@example.com'; | ||||
| 		$user->password = '$2y$10$YJeArRNypSbmpWG3RA83n.o78EVlyyVCFN71lWJ7.Omc1VEdwmX5W'; // Session::hashPassword('12345678'); | ||||
| 		$user->save(); | ||||
|  | ||||
| 		return $user; | ||||
| 	} | ||||
|  | ||||
| 	protected function session($userNum = 1, $clientNum = 1, $sessionNum = 1) { | ||||
| 		$userId = $this->createModelId('user', $userNum); | ||||
| 		$clientId = $this->createModelId('client', $clientNum); | ||||
| 		$sessionId = $this->createModelId('session', $sessioNum); | ||||
|  | ||||
| 		$session = Session::find($sessionId); | ||||
| 		if ($session) return $session; | ||||
|  | ||||
| 		$session = new Session(); | ||||
| 		$session->id = $sessionId; | ||||
| 		$session->owner_id = $userId; | ||||
| 		$session->client_id = $clientId; | ||||
| 		$session->save(); | ||||
|  | ||||
| 		return $session; | ||||
| 	} | ||||
|  | ||||
| 	public function setUp() { | ||||
| 		BaseModel::setClientId($this->clientId()); | ||||
| 	} | ||||
|  | ||||
| 	public function tearDown() { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| } | ||||
							
								
								
									
										132
									
								
								tests/Model/ChangeTest.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								tests/Model/ChangeTest.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | ||||
| <?php | ||||
|  | ||||
| require_once dirname(dirname(__FILE__)) . '/setup.php'; | ||||
|  | ||||
| use AppBundle\Model\BaseModel; | ||||
| use AppBundle\Model\FolderItem; | ||||
| use AppBundle\Model\Note; | ||||
| use AppBundle\Model\Change; | ||||
|  | ||||
| class ChangeTest extends BaseTestCase { | ||||
|  | ||||
| 	public function setUp() { | ||||
| 		parent::setUp(); | ||||
|  | ||||
| 		Change::truncate(); | ||||
| 		Note::truncate(); | ||||
| 	} | ||||
|  | ||||
| 	public function tearDown() { | ||||
|  | ||||
| 	} | ||||
|  | ||||
| 	public function testDiff() { | ||||
| 		$text1 = 'abcd efgh ijkl'; | ||||
|  | ||||
| 		$itemId = $this->createModelId('note'); | ||||
|  | ||||
| 		$change = new Change(); | ||||
| 		$change->user_id = $this->user()->id; | ||||
| 		$change->client_id = $this->clientId(); | ||||
| 		$change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		$change->item_field = BaseModel::enumId('field', 'body'); | ||||
| 		$change->type = Change::enumId('type', 'create'); | ||||
| 		$change->item_id = $itemId; | ||||
| 		$change->createDelta($text1); | ||||
| 		$change->save(); | ||||
| 		 | ||||
| 		$text2 = 'cd efgh NEW ijkl'; | ||||
|  | ||||
| 		$change = new Change(); | ||||
| 		$change->user_id = $this->user()->id; | ||||
| 		$change->client_id = $this->clientId(); | ||||
| 		$change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		$change->item_field = BaseModel::enumId('field', 'body'); | ||||
| 		$change->type = Change::enumId('type', 'update'); | ||||
| 		$change->item_id = $itemId; | ||||
| 		$change->createDelta($text2); | ||||
| 		$change->save(); | ||||
|  | ||||
| 		$r = Change::fullFieldText($itemId, BaseModel::enumId('field', 'body')); | ||||
|  | ||||
| 		$this->assertEquals($r, $text2); | ||||
| 	} | ||||
|  | ||||
| 	public function testDiff3Ways() { | ||||
| 		// Scenario where two different clients change the same note at the same time. | ||||
| 		// | ||||
| 		// Client 1: 'abcd efgh ijkl' => 'cd efgh ijkl FROMCLIENT2' | ||||
| 		// Client 2: 'abcd efgh ijkl' => 'abcd CLIENT1 efgh ijkl' | ||||
| 		// Expected: 'cd CLIENT1 efgh ijkl FROMCLIENT2' | ||||
|  | ||||
| 		$text1 = 'abcd efgh ijkl'; | ||||
|  | ||||
| 		$itemId = $this->createModelId('note'); | ||||
|  | ||||
| 		$change = new Change(); | ||||
| 		$change->user_id = $this->user()->id; | ||||
| 		$change->client_id = $this->clientId(1); | ||||
| 		$change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		$change->item_field = BaseModel::enumId('field', 'body'); | ||||
| 		$change->type = Change::enumId('type', 'create'); | ||||
| 		$change->item_id = $itemId; | ||||
| 		$change->createDelta($text1); | ||||
| 		$change->save(); | ||||
|  | ||||
| 		$changeId1 = $change->id; | ||||
| 		 | ||||
| 		$text2 = 'cd efgh ijkl FROMCLIENT2'; | ||||
|  | ||||
| 		$change = new Change(); | ||||
| 		$change->user_id = $this->user()->id; | ||||
| 		$change->client_id = $this->clientId(2); | ||||
| 		$change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		$change->item_field = BaseModel::enumId('field', 'body'); | ||||
| 		$change->type = Change::enumId('type', 'update'); | ||||
| 		$change->item_id = $itemId; | ||||
| 		$change->previous_id = $changeId1; | ||||
| 		$change->createDelta($text2); | ||||
| 		$change->save(); | ||||
|  | ||||
| 		$changeId2 = $change->id; | ||||
|  | ||||
| 		$text3 = 'abcd CLIENT1 efgh ijkl'; | ||||
|  | ||||
| 		$change = new Change(); | ||||
| 		$change->user_id = $this->user()->id; | ||||
| 		$change->client_id = $this->clientId(1); | ||||
| 		$change->item_type = FolderItem::enumId('type', 'note'); | ||||
| 		$change->item_field = BaseModel::enumId('field', 'body'); | ||||
| 		$change->type = Change::enumId('type', 'update'); | ||||
| 		$change->item_id = $itemId; | ||||
| 		$change->previous_id = $changeId1; | ||||
| 		$change->createDelta($text3); | ||||
| 		$change->save(); | ||||
|  | ||||
| 		$changeId3 = $change->id; | ||||
|  | ||||
| 		$r = Change::fullFieldText($itemId, BaseModel::enumId('field', 'body')); | ||||
|  | ||||
| 		$this->assertEquals($r, 'cd CLIENT1 efgh ijkl FROMCLIENT2'); | ||||
| 	} | ||||
|  | ||||
| 	public function testRevId() { | ||||
| 		$n = new Note(); | ||||
| 		$n->setVersionedFieldValue('body', 'abcd efgh'); | ||||
| 		$n->save(); | ||||
|  | ||||
| 		$noteId = $n->id; | ||||
|  | ||||
| 		$n = Note::find($noteId); | ||||
| 		$d = $n->toPublicArray(); | ||||
| 		$this->assertEquals(1, $d['rev_id']); | ||||
|  | ||||
| 		$n->setVersionedFieldValue('body', '123456'); | ||||
| 		$n->save(); | ||||
|  | ||||
| 		$n = Note::find($noteId); | ||||
| 		$d = $n->toPublicArray(); | ||||
| 		$this->assertEquals(2, $d['rev_id']); | ||||
| 	} | ||||
| 	 | ||||
| } | ||||
							
								
								
									
										42
									
								
								tests/Model/NoteTest.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								tests/Model/NoteTest.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| <?php | ||||
|  | ||||
| require_once dirname(dirname(__FILE__)) . '/setup.php'; | ||||
|  | ||||
| use AppBundle\Model\Note; | ||||
|  | ||||
| class NoteTest extends BaseTestCase { | ||||
|  | ||||
| 	public function setUp() { | ||||
| 		parent::setUp(); | ||||
| 	} | ||||
|  | ||||
| 	public function testCanSaveAndLoad() { | ||||
| 		$note = new Note(); | ||||
| 		$note->setVersionedFieldValue('title', 'the title'); | ||||
| 		$note->setVersionedFieldValue('body', 'the body'); | ||||
| 		$note->save(); | ||||
|  | ||||
| 		$note = $note->find($note->id); | ||||
| 		$this->assertNotNull($note); | ||||
| 		$this->assertEquals('the title', $note->versionedFieldValue('title')); | ||||
| 		$this->assertEquals('the body', $note->versionedFieldValue('body')); | ||||
| 	} | ||||
|  | ||||
| 	public function testFromToPublicArray() { | ||||
| 		$a = array( | ||||
| 			'title' => 'the title', | ||||
| 			'body' => 'the body', | ||||
| 		); | ||||
| 		$note = new Note(); | ||||
| 		$note->fromPublicArray($a); | ||||
| 		$note->save(); | ||||
|  | ||||
| 		$note = $note::find($note->id); | ||||
| 		$b = $note->toPublicArray(); | ||||
|  | ||||
| 		$this->assertEquals('the title', $b['title']); | ||||
| 		$this->assertEquals('the body', $b['body']); | ||||
| 		$this->assertArrayHasKey('rev_id', $b); | ||||
| 	} | ||||
| 	 | ||||
| } | ||||
							
								
								
									
										27
									
								
								tests/setup.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								tests/setup.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| <?php | ||||
|  | ||||
| require_once dirname(__FILE__) . '/BaseTestCase.php'; | ||||
|  | ||||
| $dbName = 'notes_test'; | ||||
| $structureFile = dirname(dirname(__FILE__)) . '/structure.sql'; | ||||
|  | ||||
| $cmd = sprintf("mysql -u root -ppass -e 'DROP DATABASE IF EXISTS %s; CREATE DATABASE %s;'", $dbName, $dbName); | ||||
| exec($cmd); | ||||
|  | ||||
| $cmd = sprintf("mysql -u root -ppass %s < '%s'", $dbName, $structureFile); | ||||
| exec($cmd); | ||||
|  | ||||
| $capsule = new \Illuminate\Database\Capsule\Manager(); | ||||
|  | ||||
| $capsule->addConnection([ | ||||
| 	'driver'    => 'mysql', | ||||
| 	'host'      => 'localhost', | ||||
| 	'database'  => 'notes_test', | ||||
| 	'username'  => 'root', | ||||
| 	'password'  => 'pass', | ||||
| 	'charset'   => 'utf8', | ||||
| 	'collation' => 'utf8_unicode_ci', | ||||
| 	'prefix'    => '', | ||||
| ]); | ||||
|  | ||||
| $capsule->bootEloquent(); | ||||
							
								
								
									
										819
									
								
								var/SymfonyRequirements.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										819
									
								
								var/SymfonyRequirements.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,819 @@ | ||||
| <?php | ||||
|  | ||||
| /* | ||||
|  * This file is part of the Symfony package. | ||||
|  * | ||||
|  * (c) Fabien Potencier <fabien@symfony.com> | ||||
|  * | ||||
|  * For the full copyright and license information, please view the LICENSE | ||||
|  * file that was distributed with this source code. | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Users of PHP 5.2 should be able to run the requirements checks. | ||||
|  * This is why the file and all classes must be compatible with PHP 5.2+ | ||||
|  * (e.g. not using namespaces and closures). | ||||
|  * | ||||
|  * ************** CAUTION ************** | ||||
|  * | ||||
|  * DO NOT EDIT THIS FILE as it will be overridden by Composer as part of | ||||
|  * the installation/update process. The original file resides in the | ||||
|  * SensioDistributionBundle. | ||||
|  * | ||||
|  * ************** CAUTION ************** | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * Represents a single PHP requirement, e.g. an installed extension. | ||||
|  * It can be a mandatory requirement or an optional recommendation. | ||||
|  * There is a special subclass, named PhpIniRequirement, to check a php.ini configuration. | ||||
|  * | ||||
|  * @author Tobias Schultze <http://tobion.de> | ||||
|  */ | ||||
| class Requirement | ||||
| { | ||||
|     private $fulfilled; | ||||
|     private $testMessage; | ||||
|     private $helpText; | ||||
|     private $helpHtml; | ||||
|     private $optional; | ||||
|  | ||||
|     /** | ||||
|      * Constructor that initializes the requirement. | ||||
|      * | ||||
|      * @param bool        $fulfilled   Whether the requirement is fulfilled | ||||
|      * @param string      $testMessage The message for testing the requirement | ||||
|      * @param string      $helpHtml    The help text formatted in HTML for resolving the problem | ||||
|      * @param string|null $helpText    The help text (when null, it will be inferred from $helpHtml, i.e. stripped from HTML tags) | ||||
|      * @param bool        $optional    Whether this is only an optional recommendation not a mandatory requirement | ||||
|      */ | ||||
|     public function __construct($fulfilled, $testMessage, $helpHtml, $helpText = null, $optional = false) | ||||
|     { | ||||
|         $this->fulfilled = (bool) $fulfilled; | ||||
|         $this->testMessage = (string) $testMessage; | ||||
|         $this->helpHtml = (string) $helpHtml; | ||||
|         $this->helpText = null === $helpText ? strip_tags($this->helpHtml) : (string) $helpText; | ||||
|         $this->optional = (bool) $optional; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether the requirement is fulfilled. | ||||
|      * | ||||
|      * @return bool true if fulfilled, otherwise false | ||||
|      */ | ||||
|     public function isFulfilled() | ||||
|     { | ||||
|         return $this->fulfilled; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the message for testing the requirement. | ||||
|      * | ||||
|      * @return string The test message | ||||
|      */ | ||||
|     public function getTestMessage() | ||||
|     { | ||||
|         return $this->testMessage; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the help text for resolving the problem. | ||||
|      * | ||||
|      * @return string The help text | ||||
|      */ | ||||
|     public function getHelpText() | ||||
|     { | ||||
|         return $this->helpText; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the help text formatted in HTML. | ||||
|      * | ||||
|      * @return string The HTML help | ||||
|      */ | ||||
|     public function getHelpHtml() | ||||
|     { | ||||
|         return $this->helpHtml; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether this is only an optional recommendation and not a mandatory requirement. | ||||
|      * | ||||
|      * @return bool true if optional, false if mandatory | ||||
|      */ | ||||
|     public function isOptional() | ||||
|     { | ||||
|         return $this->optional; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Represents a PHP requirement in form of a php.ini configuration. | ||||
|  * | ||||
|  * @author Tobias Schultze <http://tobion.de> | ||||
|  */ | ||||
| class PhpIniRequirement extends Requirement | ||||
| { | ||||
|     /** | ||||
|      * Constructor that initializes the requirement. | ||||
|      * | ||||
|      * @param string        $cfgName           The configuration name used for ini_get() | ||||
|      * @param bool|callback $evaluation        Either a boolean indicating whether the configuration should evaluate to true or false, | ||||
|      *                                         or a callback function receiving the configuration value as parameter to determine the fulfillment of the requirement | ||||
|      * @param bool          $approveCfgAbsence If true the Requirement will be fulfilled even if the configuration option does not exist, i.e. ini_get() returns false. | ||||
|      *                                         This is helpful for abandoned configs in later PHP versions or configs of an optional extension, like Suhosin. | ||||
|      *                                         Example: You require a config to be true but PHP later removes this config and defaults it to true internally. | ||||
|      * @param string|null   $testMessage       The message for testing the requirement (when null and $evaluation is a boolean a default message is derived) | ||||
|      * @param string|null   $helpHtml          The help text formatted in HTML for resolving the problem (when null and $evaluation is a boolean a default help is derived) | ||||
|      * @param string|null   $helpText          The help text (when null, it will be inferred from $helpHtml, i.e. stripped from HTML tags) | ||||
|      * @param bool          $optional          Whether this is only an optional recommendation not a mandatory requirement | ||||
|      */ | ||||
|     public function __construct($cfgName, $evaluation, $approveCfgAbsence = false, $testMessage = null, $helpHtml = null, $helpText = null, $optional = false) | ||||
|     { | ||||
|         $cfgValue = ini_get($cfgName); | ||||
|  | ||||
|         if (is_callable($evaluation)) { | ||||
|             if (null === $testMessage || null === $helpHtml) { | ||||
|                 throw new InvalidArgumentException('You must provide the parameters testMessage and helpHtml for a callback evaluation.'); | ||||
|             } | ||||
|  | ||||
|             $fulfilled = call_user_func($evaluation, $cfgValue); | ||||
|         } else { | ||||
|             if (null === $testMessage) { | ||||
|                 $testMessage = sprintf('%s %s be %s in php.ini', | ||||
|                     $cfgName, | ||||
|                     $optional ? 'should' : 'must', | ||||
|                     $evaluation ? 'enabled' : 'disabled' | ||||
|                 ); | ||||
|             } | ||||
|  | ||||
|             if (null === $helpHtml) { | ||||
|                 $helpHtml = sprintf('Set <strong>%s</strong> to <strong>%s</strong> in php.ini<a href="#phpini">*</a>.', | ||||
|                     $cfgName, | ||||
|                     $evaluation ? 'on' : 'off' | ||||
|                 ); | ||||
|             } | ||||
|  | ||||
|             $fulfilled = $evaluation == $cfgValue; | ||||
|         } | ||||
|  | ||||
|         parent::__construct($fulfilled || ($approveCfgAbsence && false === $cfgValue), $testMessage, $helpHtml, $helpText, $optional); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * A RequirementCollection represents a set of Requirement instances. | ||||
|  * | ||||
|  * @author Tobias Schultze <http://tobion.de> | ||||
|  */ | ||||
| class RequirementCollection implements IteratorAggregate | ||||
| { | ||||
|     /** | ||||
|      * @var Requirement[] | ||||
|      */ | ||||
|     private $requirements = array(); | ||||
|  | ||||
|     /** | ||||
|      * Gets the current RequirementCollection as an Iterator. | ||||
|      * | ||||
|      * @return Traversable A Traversable interface | ||||
|      */ | ||||
|     public function getIterator() | ||||
|     { | ||||
|         return new ArrayIterator($this->requirements); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a Requirement. | ||||
|      * | ||||
|      * @param Requirement $requirement A Requirement instance | ||||
|      */ | ||||
|     public function add(Requirement $requirement) | ||||
|     { | ||||
|         $this->requirements[] = $requirement; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a mandatory requirement. | ||||
|      * | ||||
|      * @param bool        $fulfilled   Whether the requirement is fulfilled | ||||
|      * @param string      $testMessage The message for testing the requirement | ||||
|      * @param string      $helpHtml    The help text formatted in HTML for resolving the problem | ||||
|      * @param string|null $helpText    The help text (when null, it will be inferred from $helpHtml, i.e. stripped from HTML tags) | ||||
|      */ | ||||
|     public function addRequirement($fulfilled, $testMessage, $helpHtml, $helpText = null) | ||||
|     { | ||||
|         $this->add(new Requirement($fulfilled, $testMessage, $helpHtml, $helpText, false)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds an optional recommendation. | ||||
|      * | ||||
|      * @param bool        $fulfilled   Whether the recommendation is fulfilled | ||||
|      * @param string      $testMessage The message for testing the recommendation | ||||
|      * @param string      $helpHtml    The help text formatted in HTML for resolving the problem | ||||
|      * @param string|null $helpText    The help text (when null, it will be inferred from $helpHtml, i.e. stripped from HTML tags) | ||||
|      */ | ||||
|     public function addRecommendation($fulfilled, $testMessage, $helpHtml, $helpText = null) | ||||
|     { | ||||
|         $this->add(new Requirement($fulfilled, $testMessage, $helpHtml, $helpText, true)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a mandatory requirement in form of a php.ini configuration. | ||||
|      * | ||||
|      * @param string        $cfgName           The configuration name used for ini_get() | ||||
|      * @param bool|callback $evaluation        Either a boolean indicating whether the configuration should evaluate to true or false, | ||||
|      *                                         or a callback function receiving the configuration value as parameter to determine the fulfillment of the requirement | ||||
|      * @param bool          $approveCfgAbsence If true the Requirement will be fulfilled even if the configuration option does not exist, i.e. ini_get() returns false. | ||||
|      *                                         This is helpful for abandoned configs in later PHP versions or configs of an optional extension, like Suhosin. | ||||
|      *                                         Example: You require a config to be true but PHP later removes this config and defaults it to true internally. | ||||
|      * @param string        $testMessage       The message for testing the requirement (when null and $evaluation is a boolean a default message is derived) | ||||
|      * @param string        $helpHtml          The help text formatted in HTML for resolving the problem (when null and $evaluation is a boolean a default help is derived) | ||||
|      * @param string|null   $helpText          The help text (when null, it will be inferred from $helpHtml, i.e. stripped from HTML tags) | ||||
|      */ | ||||
|     public function addPhpIniRequirement($cfgName, $evaluation, $approveCfgAbsence = false, $testMessage = null, $helpHtml = null, $helpText = null) | ||||
|     { | ||||
|         $this->add(new PhpIniRequirement($cfgName, $evaluation, $approveCfgAbsence, $testMessage, $helpHtml, $helpText, false)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds an optional recommendation in form of a php.ini configuration. | ||||
|      * | ||||
|      * @param string        $cfgName           The configuration name used for ini_get() | ||||
|      * @param bool|callback $evaluation        Either a boolean indicating whether the configuration should evaluate to true or false, | ||||
|      *                                         or a callback function receiving the configuration value as parameter to determine the fulfillment of the requirement | ||||
|      * @param bool          $approveCfgAbsence If true the Requirement will be fulfilled even if the configuration option does not exist, i.e. ini_get() returns false. | ||||
|      *                                         This is helpful for abandoned configs in later PHP versions or configs of an optional extension, like Suhosin. | ||||
|      *                                         Example: You require a config to be true but PHP later removes this config and defaults it to true internally. | ||||
|      * @param string        $testMessage       The message for testing the requirement (when null and $evaluation is a boolean a default message is derived) | ||||
|      * @param string        $helpHtml          The help text formatted in HTML for resolving the problem (when null and $evaluation is a boolean a default help is derived) | ||||
|      * @param string|null   $helpText          The help text (when null, it will be inferred from $helpHtml, i.e. stripped from HTML tags) | ||||
|      */ | ||||
|     public function addPhpIniRecommendation($cfgName, $evaluation, $approveCfgAbsence = false, $testMessage = null, $helpHtml = null, $helpText = null) | ||||
|     { | ||||
|         $this->add(new PhpIniRequirement($cfgName, $evaluation, $approveCfgAbsence, $testMessage, $helpHtml, $helpText, true)); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Adds a requirement collection to the current set of requirements. | ||||
|      * | ||||
|      * @param RequirementCollection $collection A RequirementCollection instance | ||||
|      */ | ||||
|     public function addCollection(RequirementCollection $collection) | ||||
|     { | ||||
|         $this->requirements = array_merge($this->requirements, $collection->all()); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns both requirements and recommendations. | ||||
|      * | ||||
|      * @return Requirement[] | ||||
|      */ | ||||
|     public function all() | ||||
|     { | ||||
|         return $this->requirements; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns all mandatory requirements. | ||||
|      * | ||||
|      * @return Requirement[] | ||||
|      */ | ||||
|     public function getRequirements() | ||||
|     { | ||||
|         $array = array(); | ||||
|         foreach ($this->requirements as $req) { | ||||
|             if (!$req->isOptional()) { | ||||
|                 $array[] = $req; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $array; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the mandatory requirements that were not met. | ||||
|      * | ||||
|      * @return Requirement[] | ||||
|      */ | ||||
|     public function getFailedRequirements() | ||||
|     { | ||||
|         $array = array(); | ||||
|         foreach ($this->requirements as $req) { | ||||
|             if (!$req->isFulfilled() && !$req->isOptional()) { | ||||
|                 $array[] = $req; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $array; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns all optional recommendations. | ||||
|      * | ||||
|      * @return Requirement[] | ||||
|      */ | ||||
|     public function getRecommendations() | ||||
|     { | ||||
|         $array = array(); | ||||
|         foreach ($this->requirements as $req) { | ||||
|             if ($req->isOptional()) { | ||||
|                 $array[] = $req; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $array; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the recommendations that were not met. | ||||
|      * | ||||
|      * @return Requirement[] | ||||
|      */ | ||||
|     public function getFailedRecommendations() | ||||
|     { | ||||
|         $array = array(); | ||||
|         foreach ($this->requirements as $req) { | ||||
|             if (!$req->isFulfilled() && $req->isOptional()) { | ||||
|                 $array[] = $req; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return $array; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns whether a php.ini configuration is not correct. | ||||
|      * | ||||
|      * @return bool php.ini configuration problem? | ||||
|      */ | ||||
|     public function hasPhpIniConfigIssue() | ||||
|     { | ||||
|         foreach ($this->requirements as $req) { | ||||
|             if (!$req->isFulfilled() && $req instanceof PhpIniRequirement) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns the PHP configuration file (php.ini) path. | ||||
|      * | ||||
|      * @return string|false php.ini file path | ||||
|      */ | ||||
|     public function getPhpIniConfigPath() | ||||
|     { | ||||
|         return get_cfg_var('cfg_file_path'); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * This class specifies all requirements and optional recommendations that | ||||
|  * are necessary to run the Symfony Standard Edition. | ||||
|  * | ||||
|  * @author Tobias Schultze <http://tobion.de> | ||||
|  * @author Fabien Potencier <fabien@symfony.com> | ||||
|  */ | ||||
| class SymfonyRequirements extends RequirementCollection | ||||
| { | ||||
|     const LEGACY_REQUIRED_PHP_VERSION = '5.3.3'; | ||||
|     const REQUIRED_PHP_VERSION = '5.5.9'; | ||||
|  | ||||
|     /** | ||||
|      * Constructor that initializes the requirements. | ||||
|      */ | ||||
|     public function __construct() | ||||
|     { | ||||
|         /* mandatory requirements follow */ | ||||
|  | ||||
|         $installedPhpVersion = phpversion(); | ||||
|         $requiredPhpVersion = $this->getPhpRequiredVersion(); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             $requiredPhpVersion, | ||||
|             'Vendors should be installed in order to check all requirements.', | ||||
|             'Run the <code>composer install</code> command.', | ||||
|             'Run the "composer install" command.' | ||||
|         ); | ||||
|  | ||||
|         if (false !== $requiredPhpVersion) { | ||||
|             $this->addRequirement( | ||||
|                 version_compare($installedPhpVersion, $requiredPhpVersion, '>='), | ||||
|                 sprintf('PHP version must be at least %s (%s installed)', $requiredPhpVersion, $installedPhpVersion), | ||||
|                 sprintf('You are running PHP version "<strong>%s</strong>", but Symfony needs at least PHP "<strong>%s</strong>" to run. | ||||
|                 Before using Symfony, upgrade your PHP installation, preferably to the latest version.', | ||||
|                     $installedPhpVersion, $requiredPhpVersion), | ||||
|                 sprintf('Install PHP %s or newer (installed version is %s)', $requiredPhpVersion, $installedPhpVersion) | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             version_compare($installedPhpVersion, '5.3.16', '!='), | ||||
|             'PHP version must not be 5.3.16 as Symfony won\'t work properly with it', | ||||
|             'Install PHP 5.3.17 or newer (or downgrade to an earlier PHP version)' | ||||
|         ); | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             is_dir(__DIR__.'/../vendor/composer'), | ||||
|             'Vendor libraries must be installed', | ||||
|             'Vendor libraries are missing. Install composer following instructions from <a href="http://getcomposer.org/">http://getcomposer.org/</a>. '. | ||||
|                 'Then run "<strong>php composer.phar install</strong>" to install them.' | ||||
|         ); | ||||
|  | ||||
|         $cacheDir = is_dir(__DIR__.'/../var/cache') ? __DIR__.'/../var/cache' : __DIR__.'/cache'; | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             is_writable($cacheDir), | ||||
|             'app/cache/ or var/cache/ directory must be writable', | ||||
|             'Change the permissions of either "<strong>app/cache/</strong>" or  "<strong>var/cache/</strong>" directory so that the web server can write into it.' | ||||
|         ); | ||||
|  | ||||
|         $logsDir = is_dir(__DIR__.'/../var/logs') ? __DIR__.'/../var/logs' : __DIR__.'/logs'; | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             is_writable($logsDir), | ||||
|             'app/logs/ or var/logs/ directory must be writable', | ||||
|             'Change the permissions of either "<strong>app/logs/</strong>" or  "<strong>var/logs/</strong>" directory so that the web server can write into it.' | ||||
|         ); | ||||
|  | ||||
|         if (version_compare($installedPhpVersion, '7.0.0', '<')) { | ||||
|             $this->addPhpIniRequirement( | ||||
|                 'date.timezone', true, false, | ||||
|                 'date.timezone setting must be set', | ||||
|                 'Set the "<strong>date.timezone</strong>" setting in php.ini<a href="#phpini">*</a> (like Europe/Paris).' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         if (false !== $requiredPhpVersion && version_compare($installedPhpVersion, $requiredPhpVersion, '>=')) { | ||||
|             $timezones = array(); | ||||
|             foreach (DateTimeZone::listAbbreviations() as $abbreviations) { | ||||
|                 foreach ($abbreviations as $abbreviation) { | ||||
|                     $timezones[$abbreviation['timezone_id']] = true; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $this->addRequirement( | ||||
|                 isset($timezones[@date_default_timezone_get()]), | ||||
|                 sprintf('Configured default timezone "%s" must be supported by your installation of PHP', @date_default_timezone_get()), | ||||
|                 'Your default timezone is not supported by PHP. Check for typos in your <strong>php.ini</strong> file and have a look at the list of deprecated timezones at <a href="http://php.net/manual/en/timezones.others.php">http://php.net/manual/en/timezones.others.php</a>.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             function_exists('iconv'), | ||||
|             'iconv() must be available', | ||||
|             'Install and enable the <strong>iconv</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             function_exists('json_encode'), | ||||
|             'json_encode() must be available', | ||||
|             'Install and enable the <strong>JSON</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             function_exists('session_start'), | ||||
|             'session_start() must be available', | ||||
|             'Install and enable the <strong>session</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             function_exists('ctype_alpha'), | ||||
|             'ctype_alpha() must be available', | ||||
|             'Install and enable the <strong>ctype</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             function_exists('token_get_all'), | ||||
|             'token_get_all() must be available', | ||||
|             'Install and enable the <strong>Tokenizer</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             function_exists('simplexml_import_dom'), | ||||
|             'simplexml_import_dom() must be available', | ||||
|             'Install and enable the <strong>SimpleXML</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         if (function_exists('apc_store') && ini_get('apc.enabled')) { | ||||
|             if (version_compare($installedPhpVersion, '5.4.0', '>=')) { | ||||
|                 $this->addRequirement( | ||||
|                     version_compare(phpversion('apc'), '3.1.13', '>='), | ||||
|                     'APC version must be at least 3.1.13 when using PHP 5.4', | ||||
|                     'Upgrade your <strong>APC</strong> extension (3.1.13+).' | ||||
|                 ); | ||||
|             } else { | ||||
|                 $this->addRequirement( | ||||
|                     version_compare(phpversion('apc'), '3.0.17', '>='), | ||||
|                     'APC version must be at least 3.0.17', | ||||
|                     'Upgrade your <strong>APC</strong> extension (3.0.17+).' | ||||
|                 ); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         $this->addPhpIniRequirement('detect_unicode', false); | ||||
|  | ||||
|         if (extension_loaded('suhosin')) { | ||||
|             $this->addPhpIniRequirement( | ||||
|                 'suhosin.executor.include.whitelist', | ||||
|                 create_function('$cfgValue', 'return false !== stripos($cfgValue, "phar");'), | ||||
|                 false, | ||||
|                 'suhosin.executor.include.whitelist must be configured correctly in php.ini', | ||||
|                 'Add "<strong>phar</strong>" to <strong>suhosin.executor.include.whitelist</strong> in php.ini<a href="#phpini">*</a>.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         if (extension_loaded('xdebug')) { | ||||
|             $this->addPhpIniRequirement( | ||||
|                 'xdebug.show_exception_trace', false, true | ||||
|             ); | ||||
|  | ||||
|             $this->addPhpIniRequirement( | ||||
|                 'xdebug.scream', false, true | ||||
|             ); | ||||
|  | ||||
|             $this->addPhpIniRecommendation( | ||||
|                 'xdebug.max_nesting_level', | ||||
|                 create_function('$cfgValue', 'return $cfgValue > 100;'), | ||||
|                 true, | ||||
|                 'xdebug.max_nesting_level should be above 100 in php.ini', | ||||
|                 'Set "<strong>xdebug.max_nesting_level</strong>" to e.g. "<strong>250</strong>" in php.ini<a href="#phpini">*</a> to stop Xdebug\'s infinite recursion protection erroneously throwing a fatal error in your project.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $pcreVersion = defined('PCRE_VERSION') ? (float) PCRE_VERSION : null; | ||||
|  | ||||
|         $this->addRequirement( | ||||
|             null !== $pcreVersion, | ||||
|             'PCRE extension must be available', | ||||
|             'Install the <strong>PCRE</strong> extension (version 8.0+).' | ||||
|         ); | ||||
|  | ||||
|         if (extension_loaded('mbstring')) { | ||||
|             $this->addPhpIniRequirement( | ||||
|                 'mbstring.func_overload', | ||||
|                 create_function('$cfgValue', 'return (int) $cfgValue === 0;'), | ||||
|                 true, | ||||
|                 'string functions should not be overloaded', | ||||
|                 'Set "<strong>mbstring.func_overload</strong>" to <strong>0</strong> in php.ini<a href="#phpini">*</a> to disable function overloading by the mbstring extension.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         /* optional recommendations follow */ | ||||
|  | ||||
|         if (file_exists(__DIR__.'/../vendor/composer')) { | ||||
|             require_once __DIR__.'/../vendor/autoload.php'; | ||||
|  | ||||
|             try { | ||||
|                 $r = new ReflectionClass('Sensio\Bundle\DistributionBundle\SensioDistributionBundle'); | ||||
|  | ||||
|                 $contents = file_get_contents(dirname($r->getFileName()).'/Resources/skeleton/app/SymfonyRequirements.php'); | ||||
|             } catch (ReflectionException $e) { | ||||
|                 $contents = ''; | ||||
|             } | ||||
|             $this->addRecommendation( | ||||
|                 file_get_contents(__FILE__) === $contents, | ||||
|                 'Requirements file should be up-to-date', | ||||
|                 'Your requirements file is outdated. Run composer install and re-check your configuration.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             version_compare($installedPhpVersion, '5.3.4', '>='), | ||||
|             'You should use at least PHP 5.3.4 due to PHP bug #52083 in earlier versions', | ||||
|             'Your project might malfunction randomly due to PHP bug #52083 ("Notice: Trying to get property of non-object"). Install PHP 5.3.4 or newer.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             version_compare($installedPhpVersion, '5.3.8', '>='), | ||||
|             'When using annotations you should have at least PHP 5.3.8 due to PHP bug #55156', | ||||
|             'Install PHP 5.3.8 or newer if your project uses annotations.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             version_compare($installedPhpVersion, '5.4.0', '!='), | ||||
|             'You should not use PHP 5.4.0 due to the PHP bug #61453', | ||||
|             'Your project might not work properly due to the PHP bug #61453 ("Cannot dump definitions which have method calls"). Install PHP 5.4.1 or newer.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             version_compare($installedPhpVersion, '5.4.11', '>='), | ||||
|             'When using the logout handler from the Symfony Security Component, you should have at least PHP 5.4.11 due to PHP bug #63379 (as a workaround, you can also set invalidate_session to false in the security logout handler configuration)', | ||||
|             'Install PHP 5.4.11 or newer if your project uses the logout handler from the Symfony Security Component.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             (version_compare($installedPhpVersion, '5.3.18', '>=') && version_compare($installedPhpVersion, '5.4.0', '<')) | ||||
|             || | ||||
|             version_compare($installedPhpVersion, '5.4.8', '>='), | ||||
|             'You should use PHP 5.3.18+ or PHP 5.4.8+ to always get nice error messages for fatal errors in the development environment due to PHP bug #61767/#60909', | ||||
|             'Install PHP 5.3.18+ or PHP 5.4.8+ if you want nice error messages for all fatal errors in the development environment.' | ||||
|         ); | ||||
|  | ||||
|         if (null !== $pcreVersion) { | ||||
|             $this->addRecommendation( | ||||
|                 $pcreVersion >= 8.0, | ||||
|                 sprintf('PCRE extension should be at least version 8.0 (%s installed)', $pcreVersion), | ||||
|                 '<strong>PCRE 8.0+</strong> is preconfigured in PHP since 5.3.2 but you are using an outdated version of it. Symfony probably works anyway but it is recommended to upgrade your PCRE extension.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             class_exists('DomDocument'), | ||||
|             'PHP-DOM and PHP-XML modules should be installed', | ||||
|             'Install and enable the <strong>PHP-DOM</strong> and the <strong>PHP-XML</strong> modules.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             function_exists('mb_strlen'), | ||||
|             'mb_strlen() should be available', | ||||
|             'Install and enable the <strong>mbstring</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             function_exists('iconv'), | ||||
|             'iconv() should be available', | ||||
|             'Install and enable the <strong>iconv</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             function_exists('utf8_decode'), | ||||
|             'utf8_decode() should be available', | ||||
|             'Install and enable the <strong>XML</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             function_exists('filter_var'), | ||||
|             'filter_var() should be available', | ||||
|             'Install and enable the <strong>filter</strong> extension.' | ||||
|         ); | ||||
|  | ||||
|         if (!defined('PHP_WINDOWS_VERSION_BUILD')) { | ||||
|             $this->addRecommendation( | ||||
|                 function_exists('posix_isatty'), | ||||
|                 'posix_isatty() should be available', | ||||
|                 'Install and enable the <strong>php_posix</strong> extension (used to colorize the CLI output).' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             extension_loaded('intl'), | ||||
|             'intl extension should be available', | ||||
|             'Install and enable the <strong>intl</strong> extension (used for validators).' | ||||
|         ); | ||||
|  | ||||
|         if (extension_loaded('intl')) { | ||||
|             // in some WAMP server installations, new Collator() returns null | ||||
|             $this->addRecommendation( | ||||
|                 null !== new Collator('fr_FR'), | ||||
|                 'intl extension should be correctly configured', | ||||
|                 'The intl extension does not behave properly. This problem is typical on PHP 5.3.X x64 WIN builds.' | ||||
|             ); | ||||
|  | ||||
|             // check for compatible ICU versions (only done when you have the intl extension) | ||||
|             if (defined('INTL_ICU_VERSION')) { | ||||
|                 $version = INTL_ICU_VERSION; | ||||
|             } else { | ||||
|                 $reflector = new ReflectionExtension('intl'); | ||||
|  | ||||
|                 ob_start(); | ||||
|                 $reflector->info(); | ||||
|                 $output = strip_tags(ob_get_clean()); | ||||
|  | ||||
|                 preg_match('/^ICU version +(?:=> )?(.*)$/m', $output, $matches); | ||||
|                 $version = $matches[1]; | ||||
|             } | ||||
|  | ||||
|             $this->addRecommendation( | ||||
|                 version_compare($version, '4.0', '>='), | ||||
|                 'intl ICU version should be at least 4+', | ||||
|                 'Upgrade your <strong>intl</strong> extension with a newer ICU version (4+).' | ||||
|             ); | ||||
|  | ||||
|             if (class_exists('Symfony\Component\Intl\Intl')) { | ||||
|                 $this->addRecommendation( | ||||
|                     \Symfony\Component\Intl\Intl::getIcuDataVersion() <= \Symfony\Component\Intl\Intl::getIcuVersion(), | ||||
|                     sprintf('intl ICU version installed on your system is outdated (%s) and does not match the ICU data bundled with Symfony (%s)', \Symfony\Component\Intl\Intl::getIcuVersion(), \Symfony\Component\Intl\Intl::getIcuDataVersion()), | ||||
|                     'To get the latest internationalization data upgrade the ICU system package and the intl PHP extension.' | ||||
|                 ); | ||||
|                 if (\Symfony\Component\Intl\Intl::getIcuDataVersion() <= \Symfony\Component\Intl\Intl::getIcuVersion()) { | ||||
|                     $this->addRecommendation( | ||||
|                         \Symfony\Component\Intl\Intl::getIcuDataVersion() === \Symfony\Component\Intl\Intl::getIcuVersion(), | ||||
|                         sprintf('intl ICU version installed on your system (%s) does not match the ICU data bundled with Symfony (%s)', \Symfony\Component\Intl\Intl::getIcuVersion(), \Symfony\Component\Intl\Intl::getIcuDataVersion()), | ||||
|                         'To avoid internationalization data inconsistencies upgrade the symfony/intl component.' | ||||
|                     ); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $this->addPhpIniRecommendation( | ||||
|                 'intl.error_level', | ||||
|                 create_function('$cfgValue', 'return (int) $cfgValue === 0;'), | ||||
|                 true, | ||||
|                 'intl.error_level should be 0 in php.ini', | ||||
|                 'Set "<strong>intl.error_level</strong>" to "<strong>0</strong>" in php.ini<a href="#phpini">*</a> to inhibit the messages when an error occurs in ICU functions.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $accelerator = | ||||
|             (extension_loaded('eaccelerator') && ini_get('eaccelerator.enable')) | ||||
|             || | ||||
|             (extension_loaded('apc') && ini_get('apc.enabled')) | ||||
|             || | ||||
|             (extension_loaded('Zend Optimizer+') && ini_get('zend_optimizerplus.enable')) | ||||
|             || | ||||
|             (extension_loaded('Zend OPcache') && ini_get('opcache.enable')) | ||||
|             || | ||||
|             (extension_loaded('xcache') && ini_get('xcache.cacher')) | ||||
|             || | ||||
|             (extension_loaded('wincache') && ini_get('wincache.ocenabled')) | ||||
|         ; | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             $accelerator, | ||||
|             'a PHP accelerator should be installed', | ||||
|             'Install and/or enable a <strong>PHP accelerator</strong> (highly recommended).' | ||||
|         ); | ||||
|  | ||||
|         if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') { | ||||
|             $this->addRecommendation( | ||||
|                 $this->getRealpathCacheSize() >= 5 * 1024 * 1024, | ||||
|                 'realpath_cache_size should be at least 5M in php.ini', | ||||
|                 'Setting "<strong>realpath_cache_size</strong>" to e.g. "<strong>5242880</strong>" or "<strong>5M</strong>" in php.ini<a href="#phpini">*</a> may improve performance on Windows significantly in some cases.' | ||||
|             ); | ||||
|         } | ||||
|  | ||||
|         $this->addPhpIniRecommendation('short_open_tag', false); | ||||
|  | ||||
|         $this->addPhpIniRecommendation('magic_quotes_gpc', false, true); | ||||
|  | ||||
|         $this->addPhpIniRecommendation('register_globals', false, true); | ||||
|  | ||||
|         $this->addPhpIniRecommendation('session.auto_start', false); | ||||
|  | ||||
|         $this->addRecommendation( | ||||
|             class_exists('PDO'), | ||||
|             'PDO should be installed', | ||||
|             'Install <strong>PDO</strong> (mandatory for Doctrine).' | ||||
|         ); | ||||
|  | ||||
|         if (class_exists('PDO')) { | ||||
|             $drivers = PDO::getAvailableDrivers(); | ||||
|             $this->addRecommendation( | ||||
|                 count($drivers) > 0, | ||||
|                 sprintf('PDO should have some drivers installed (currently available: %s)', count($drivers) ? implode(', ', $drivers) : 'none'), | ||||
|                 'Install <strong>PDO drivers</strong> (mandatory for Doctrine).' | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Loads realpath_cache_size from php.ini and converts it to int. | ||||
|      * | ||||
|      * (e.g. 16k is converted to 16384 int) | ||||
|      * | ||||
|      * @return int | ||||
|      */ | ||||
|     protected function getRealpathCacheSize() | ||||
|     { | ||||
|         $size = ini_get('realpath_cache_size'); | ||||
|         $size = trim($size); | ||||
|         $unit = strtolower(substr($size, -1, 1)); | ||||
|         switch ($unit) { | ||||
|             case 'g': | ||||
|                 return $size * 1024 * 1024 * 1024; | ||||
|             case 'm': | ||||
|                 return $size * 1024 * 1024; | ||||
|             case 'k': | ||||
|                 return $size * 1024; | ||||
|             default: | ||||
|                 return (int) $size; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Defines PHP required version from Symfony version. | ||||
|      * | ||||
|      * @return string|false The PHP required version or false if it could not be guessed | ||||
|      */ | ||||
|     protected function getPhpRequiredVersion() | ||||
|     { | ||||
|         if (!file_exists($path = __DIR__.'/../composer.lock')) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         $composerLock = json_decode(file_get_contents($path), true); | ||||
|         foreach ($composerLock['packages'] as $package) { | ||||
|             $name = $package['name']; | ||||
|             if ('symfony/symfony' !== $name && 'symfony/http-kernel' !== $name) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             return (int) $package['version'][1] > 2 ? self::REQUIRED_PHP_VERSION : self::LEGACY_REQUIRED_PHP_VERSION; | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										0
									
								
								var/cache/.gitkeep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								var/cache/.gitkeep
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								var/logs/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								var/logs/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										0
									
								
								var/sessions/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								var/sessions/.gitkeep
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										68
									
								
								web/.htaccess
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								web/.htaccess
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| # Use the front controller as index file. It serves as a fallback solution when | ||||
| # every other rewrite/redirect fails (e.g. in an aliased environment without | ||||
| # mod_rewrite). Additionally, this reduces the matching process for the | ||||
| # start page (path "/") because otherwise Apache will apply the rewriting rules | ||||
| # to each configured DirectoryIndex file (e.g. index.php, index.html, index.pl). | ||||
| DirectoryIndex app.php | ||||
|  | ||||
| # By default, Apache does not evaluate symbolic links if you did not enable this | ||||
| # feature in your server configuration. Uncomment the following line if you | ||||
| # install assets as symlinks or if you experience problems related to symlinks | ||||
| # when compiling LESS/Sass/CoffeScript assets. | ||||
| # Options FollowSymlinks | ||||
|  | ||||
| # Disabling MultiViews prevents unwanted negotiation, e.g. "/app" should not resolve | ||||
| # to the front controller "/app.php" but be rewritten to "/app.php/app". | ||||
| <IfModule mod_negotiation.c> | ||||
|     Options -MultiViews | ||||
| </IfModule> | ||||
|  | ||||
| <IfModule mod_rewrite.c> | ||||
|     RewriteEngine On | ||||
|  | ||||
|     # Determine the RewriteBase automatically and set it as environment variable. | ||||
|     # If you are using Apache aliases to do mass virtual hosting or installed the | ||||
|     # project in a subdirectory, the base path will be prepended to allow proper | ||||
|     # resolution of the app.php file and to redirect to the correct URI. It will | ||||
|     # work in environments without path prefix as well, providing a safe, one-size | ||||
|     # fits all solution. But as you do not need it in this case, you can comment | ||||
|     # the following 2 lines to eliminate the overhead. | ||||
|     RewriteCond %{REQUEST_URI}::$1 ^(/.+)/(.*)::\2$ | ||||
|     RewriteRule ^(.*) - [E=BASE:%1] | ||||
|  | ||||
|     # Sets the HTTP_AUTHORIZATION header removed by Apache | ||||
|     RewriteCond %{HTTP:Authorization} . | ||||
|     RewriteRule ^ - [E=HTTP_AUTHORIZATION:%{HTTP:Authorization}] | ||||
|  | ||||
|     # Redirect to URI without front controller to prevent duplicate content | ||||
|     # (with and without `/app.php`). Only do this redirect on the initial | ||||
|     # rewrite by Apache and not on subsequent cycles. Otherwise we would get an | ||||
|     # endless redirect loop (request -> rewrite to front controller -> | ||||
|     # redirect -> request -> ...). | ||||
|     # So in case you get a "too many redirects" error or you always get redirected | ||||
|     # to the start page because your Apache does not expose the REDIRECT_STATUS | ||||
|     # environment variable, you have 2 choices: | ||||
|     # - disable this feature by commenting the following 2 lines or | ||||
|     # - use Apache >= 2.3.9 and replace all L flags by END flags and remove the | ||||
|     #   following RewriteCond (best solution) | ||||
|     RewriteCond %{ENV:REDIRECT_STATUS} ^$ | ||||
|     RewriteRule ^app\.php(?:/(.*)|$) %{ENV:BASE}/$1 [R=301,L] | ||||
|  | ||||
|     # If the requested filename exists, simply serve it. | ||||
|     # We only want to let Apache serve files and not directories. | ||||
|     RewriteCond %{REQUEST_FILENAME} -f | ||||
|     RewriteRule ^ - [L] | ||||
|  | ||||
|     # Rewrite all other queries to the front controller. | ||||
|     RewriteRule ^ %{ENV:BASE}/app.php [L] | ||||
| </IfModule> | ||||
|  | ||||
| <IfModule !mod_rewrite.c> | ||||
|     <IfModule mod_alias.c> | ||||
|         # When mod_rewrite is not available, we instruct a temporary redirect of | ||||
|         # the start page to the front controller explicitly so that the website | ||||
|         # and the generated links can still be used. | ||||
|         RedirectMatch 302 ^/$ /app.php/ | ||||
|         # RedirectTemp cannot be used instead | ||||
|     </IfModule> | ||||
| </IfModule> | ||||
							
								
								
									
										18
									
								
								web/app.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								web/app.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| <?php | ||||
|  | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
|  | ||||
| /** @var \Composer\Autoload\ClassLoader $loader */ | ||||
| $loader = require __DIR__.'/../app/autoload.php'; | ||||
| include_once __DIR__.'/../var/bootstrap.php.cache'; | ||||
|  | ||||
| $kernel = new AppKernel('prod', false); | ||||
| $kernel->loadClassCache(); | ||||
| //$kernel = new AppCache($kernel); | ||||
|  | ||||
| // When using the HttpCache, you need to call the method in your front controller instead of relying on the configuration parameter | ||||
| //Request::enableHttpMethodParameterOverride(); | ||||
| $request = Request::createFromGlobals(); | ||||
| $response = $kernel->handle($request); | ||||
| $response->send(); | ||||
| $kernel->terminate($request, $response); | ||||
							
								
								
									
										40
									
								
								web/app_dev.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								web/app_dev.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| <?php | ||||
|  | ||||
| use Symfony\Component\HttpFoundation\Request; | ||||
| // use Symfony\Component\Debug\Debug; | ||||
|  | ||||
| // If you don't want to setup permissions the proper way, just uncomment the following PHP line | ||||
| // read http://symfony.com/doc/current/book/installation.html#checking-symfony-application-configuration-and-setup | ||||
| // for more information | ||||
| //umask(0000); | ||||
|  | ||||
| // This check prevents access to debug front controllers that are deployed by accident to production servers. | ||||
| // Feel free to remove this, extend it, or make something more sophisticated. | ||||
| if (isset($_SERVER['HTTP_CLIENT_IP']) | ||||
|     || isset($_SERVER['HTTP_X_FORWARDED_FOR']) | ||||
|     || !(in_array(@$_SERVER['REMOTE_ADDR'], ['127.0.0.1', '::1']) || php_sapi_name() === 'cli-server') | ||||
| ) { | ||||
|     header('HTTP/1.0 403 Forbidden'); | ||||
|     exit('You are not allowed to access this file. Check '.basename(__FILE__).' for more information.'); | ||||
| } | ||||
|  | ||||
| /** @var \Composer\Autoload\ClassLoader $loader */ | ||||
| $loader = require __DIR__.'/../app/autoload.php'; | ||||
|  | ||||
| // Disable error handling page: | ||||
| // Debug::enable(); | ||||
|  | ||||
| set_exception_handler(function($e) { | ||||
| 	$msg = array(); | ||||
| 	$msg[] = 'Exception: ' . $e->getMessage() . ' at ' . $e->getFile() . ':' . $e->getLine(); | ||||
| 	$msg[] = ''; | ||||
| 	$msg[] = $e->getTraceAsString(); | ||||
| 	echo implode("\n", $msg); | ||||
| }); | ||||
|  | ||||
| $kernel = new AppKernel('dev', true); | ||||
| $kernel->loadClassCache(); | ||||
| $request = Request::createFromGlobals(); | ||||
| $response = $kernel->handle($request); | ||||
| $response->send(); | ||||
| $kernel->terminate($request, $response); | ||||
							
								
								
									
										
											BIN
										
									
								
								web/apple-touch-icon.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								web/apple-touch-icon.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										422
									
								
								web/config.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										422
									
								
								web/config.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,422 @@ | ||||
| <?php | ||||
|  | ||||
| /* | ||||
|  * ************** CAUTION ************** | ||||
|  * | ||||
|  * DO NOT EDIT THIS FILE as it will be overridden by Composer as part of | ||||
|  * the installation/update process. The original file resides in the | ||||
|  * SensioDistributionBundle. | ||||
|  * | ||||
|  * ************** CAUTION ************** | ||||
|  */ | ||||
|  | ||||
| if (!isset($_SERVER['HTTP_HOST'])) { | ||||
|     exit('This script cannot be run from the CLI. Run it from a browser.'); | ||||
| } | ||||
|  | ||||
| if (!in_array(@$_SERVER['REMOTE_ADDR'], array( | ||||
|     '127.0.0.1', | ||||
|     '::1', | ||||
| ))) { | ||||
|     header('HTTP/1.0 403 Forbidden'); | ||||
|     exit('This script is only accessible from localhost.'); | ||||
| } | ||||
|  | ||||
| require_once dirname(__FILE__).'/../var/SymfonyRequirements.php'; | ||||
|  | ||||
| $symfonyRequirements = new SymfonyRequirements(); | ||||
|  | ||||
| $majorProblems = $symfonyRequirements->getFailedRequirements(); | ||||
| $minorProblems = $symfonyRequirements->getFailedRecommendations(); | ||||
| $hasMajorProblems = (bool) count($majorProblems); | ||||
| $hasMinorProblems = (bool) count($minorProblems); | ||||
|  | ||||
| ?> | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
|     <head> | ||||
|         <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/> | ||||
|         <meta name="robots" content="noindex,nofollow" /> | ||||
|         <title>Symfony Configuration Checker</title> | ||||
|         <style> | ||||
|             /* styles copied from symfony framework bundle */ | ||||
|             html { | ||||
|                 background: #eee; | ||||
|             } | ||||
|             body { | ||||
|                 font: 11px Verdana, Arial, sans-serif; | ||||
|                 color: #333; | ||||
|             } | ||||
|             .sf-reset, .sf-reset .block, .sf-reset #message { | ||||
|                 margin: auto; | ||||
|             } | ||||
|             img { | ||||
|                 border: 0; | ||||
|             } | ||||
|             .clear { | ||||
|                 clear: both; | ||||
|                 height: 0; | ||||
|                 font-size: 0; | ||||
|                 line-height: 0; | ||||
|             } | ||||
|             .clear-fix:after { | ||||
|                 content: "\0020"; | ||||
|                 display: block; | ||||
|                 height: 0; | ||||
|                 clear: both; | ||||
|                 visibility: hidden; | ||||
|             } | ||||
|             .clear-fix { | ||||
|                 display: inline-block; | ||||
|             } | ||||
|             * html .clear-fix { | ||||
|                 height: 1%; | ||||
|             } | ||||
|             .clear-fix { | ||||
|                 display: block; | ||||
|             } | ||||
|             .header { | ||||
|                 padding: 30px 30px 20px 30px; | ||||
|             } | ||||
|             .header-logo { | ||||
|                 float: left; | ||||
|             } | ||||
|             .search { | ||||
|                 float: right; | ||||
|                 padding-top: 20px; | ||||
|             } | ||||
|             .search label { | ||||
|                 line-height: 28px; | ||||
|                 vertical-align: middle; | ||||
|             } | ||||
|             .search input { | ||||
|                 width: 195px; | ||||
|                 font-size: 12px; | ||||
|                 border: 1px solid #dadada; | ||||
|                 background: #fff url() repeat-x left top; | ||||
|                 padding: 5px 6px; | ||||
|                 color: #565656; | ||||
|             } | ||||
|             .search input[type="search"] { | ||||
|                 -webkit-appearance: textfield; | ||||
|             } | ||||
|             #content { | ||||
|                 width: 970px; | ||||
|                 margin: 0 auto; | ||||
|             } | ||||
|             #content pre { | ||||
|                 white-space: normal; | ||||
|                 font-family: Arial, Helvetica, sans-serif; | ||||
|             } | ||||
|  | ||||
|             /* | ||||
|             Copyright (c) 2010, Yahoo! Inc. All rights reserved. | ||||
|             Code licensed under the BSD License: | ||||
|             http://developer.yahoo.com/yui/license.html | ||||
|             version: 3.1.2 | ||||
|             build: 56 | ||||
|             */ | ||||
|             .sf-reset div,.sf-reset dl,.sf-reset dt,.sf-reset dd,.sf-reset ul,.sf-reset ol,.sf-reset li,.sf-reset h1,.sf-reset h2,.sf-reset h3,.sf-reset h4,.sf-reset h5,.sf-reset h6,.sf-reset pre,.sf-reset code,.sf-reset form,.sf-reset fieldset,.sf-reset legend,.sf-reset input,.sf-reset textarea,.sf-reset p,.sf-reset blockquote,.sf-reset th,.sf-reset td{margin:0;padding:0;}.sf-reset table{border-collapse:collapse;border-spacing:0;}.sf-reset fieldset,.sf-reset img{border:0;}.sf-reset address,.sf-reset caption,.sf-reset cite,.sf-reset code,.sf-reset dfn,.sf-reset em,.sf-reset strong,.sf-reset th,.sf-reset var{font-style:normal;font-weight:normal;}.sf-reset li{list-style:none;}.sf-reset caption,.sf-reset th{text-align:left;}.sf-reset h1,.sf-reset h2,.sf-reset h3,.sf-reset h4,.sf-reset h5,.sf-reset h6{font-size:100%;font-weight:normal;}.sf-reset q:before,.sf-reset q:after{content:'';}.sf-reset abbr,.sf-reset acronym{border:0;font-variant:normal;}.sf-reset sup{vertical-align:text-top;}.sf-reset sub{vertical-align:text-bottom;}.sf-reset input,.sf-reset textarea,.sf-reset select{font-family:inherit;font-size:inherit;font-weight:inherit;}.sf-reset input,.sf-reset textarea,.sf-reset select{font-size:100%;}.sf-reset legend{color:#000;} | ||||
|             .sf-reset abbr { | ||||
|                 border-bottom: 1px dotted #000; | ||||
|                 cursor: help; | ||||
|             } | ||||
|             .sf-reset p { | ||||
|                 font-size: 14px; | ||||
|                 line-height: 20px; | ||||
|                 padding-bottom: 20px; | ||||
|             } | ||||
|             .sf-reset strong { | ||||
|                 color: #313131; | ||||
|                 font-weight: bold; | ||||
|             } | ||||
|             .sf-reset a { | ||||
|                 color: #6c6159; | ||||
|             } | ||||
|             .sf-reset a img { | ||||
|                 border: none; | ||||
|             } | ||||
|             .sf-reset a:hover { | ||||
|                 text-decoration: underline; | ||||
|             } | ||||
|             .sf-reset em { | ||||
|                 font-style: italic; | ||||
|             } | ||||
|             .sf-reset h2, | ||||
|             .sf-reset h3 { | ||||
|                 font-weight: bold; | ||||
|             } | ||||
|             .sf-reset h1 { | ||||
|                 font-family: Georgia, "Times New Roman", Times, serif; | ||||
|                 font-size: 20px; | ||||
|                 color: #313131; | ||||
|                 word-wrap: break-word; | ||||
|             } | ||||
|             .sf-reset li { | ||||
|                 padding-bottom: 10px; | ||||
|             } | ||||
|             .sf-reset .block { | ||||
|                 -moz-border-radius: 16px; | ||||
|                 -webkit-border-radius: 16px; | ||||
|                 border-radius: 16px; | ||||
|                 margin-bottom: 20px; | ||||
|                 background-color: #FFFFFF; | ||||
|                 border: 1px solid #dfdfdf; | ||||
|                 padding: 40px 50px; | ||||
|                 word-break: break-all; | ||||
|             } | ||||
|             .sf-reset h2 { | ||||
|                 font-size: 16px; | ||||
|                 font-family: Arial, Helvetica, sans-serif; | ||||
|             } | ||||
|             .sf-reset li a { | ||||
|                 background: none; | ||||
|                 color: #868686; | ||||
|                 text-decoration: none; | ||||
|             } | ||||
|             .sf-reset li a:hover { | ||||
|                 background: none; | ||||
|                 color: #313131; | ||||
|                 text-decoration: underline; | ||||
|             } | ||||
|             .sf-reset ol { | ||||
|                 padding: 10px 0; | ||||
|             } | ||||
|             .sf-reset ol li { | ||||
|                 list-style: decimal; | ||||
|                 margin-left: 20px; | ||||
|                 padding: 2px; | ||||
|                 padding-bottom: 20px; | ||||
|             } | ||||
|             .sf-reset ol ol li { | ||||
|                 list-style-position: inside; | ||||
|                 margin-left: 0; | ||||
|                 white-space: nowrap; | ||||
|                 font-size: 12px; | ||||
|                 padding-bottom: 0; | ||||
|             } | ||||
|             .sf-reset li .selected { | ||||
|                 background-color: #ffd; | ||||
|             } | ||||
|             .sf-button { | ||||
|                 display: -moz-inline-box; | ||||
|                 display: inline-block; | ||||
|                 text-align: center; | ||||
|                 vertical-align: middle; | ||||
|                 border: 0; | ||||
|                 background: transparent none; | ||||
|                 text-transform: uppercase; | ||||
|                 cursor: pointer; | ||||
|                 font: bold 11px Arial, Helvetica, sans-serif; | ||||
|             } | ||||
|             .sf-button span { | ||||
|                 text-decoration: none; | ||||
|                 display: block; | ||||
|                 height: 28px; | ||||
|                 float: left; | ||||
|             } | ||||
|             .sf-button .border-l { | ||||
|                 text-decoration: none; | ||||
|                 display: block; | ||||
|                 height: 28px; | ||||
|                 float: left; | ||||
|                 padding: 0 0 0 7px; | ||||
|                 background: transparent url() no-repeat top left; | ||||
|             } | ||||
|             .sf-button .border-r { | ||||
|                 padding: 0 7px 0 0; | ||||
|                 background: transparent url() right top no-repeat; | ||||
|             } | ||||
|             .sf-button .btn-bg { | ||||
|                 padding: 0 14px; | ||||
|                 color: #636363; | ||||
|                 line-height: 28px; | ||||
|                 background: transparent url() repeat-x top left; | ||||
|             } | ||||
|             .sf-button:hover .border-l, | ||||
|             .sf-button-selected .border-l { | ||||
|                 background: transparent url() no-repeat top left; | ||||
|             } | ||||
|             .sf-button:hover .border-r, | ||||
|             .sf-button-selected .border-r { | ||||
|                 background: transparent url() right top no-repeat; | ||||
|             } | ||||
|             .sf-button:hover .btn-bg, | ||||
|             .sf-button-selected .btn-bg { | ||||
|                 color: #FFFFFF; | ||||
|                 text-shadow:0 1px 1px #6b9311; | ||||
|                 background: transparent url() repeat-x top left; | ||||
|             } | ||||
|  | ||||
|             /* styles copied from bundles/sensiodistribution/webconfigurator/css/install.css */ | ||||
|             body { | ||||
|                 font-size: 14px; | ||||
|                 font-family: "Lucida Sans Unicode", "Lucida Grande", Verdana, Arial, Helvetica, sans-serif; | ||||
|             } | ||||
|             .sf-reset h1.title { | ||||
|                 font-size: 45px; | ||||
|                 padding-bottom: 30px; | ||||
|             } | ||||
|             .sf-reset h2 { | ||||
|                 font-weight: bold; | ||||
|                 color: #FFFFFF; | ||||
|                 /* Font is reset to sans-serif (like body) */ | ||||
|                 font-family: "Lucida Sans Unicode", "Lucida Grande", Verdana, Arial, Helvetica, sans-serif; | ||||
|                 margin-bottom: 10px; | ||||
|                 background-color: #aacd4e; | ||||
|                 padding: 2px 4px; | ||||
|                 display: inline-block; | ||||
|                 text-transform: uppercase; | ||||
|             } | ||||
|             .sf-reset ul a, | ||||
|             .sf-reset ul a:hover { | ||||
|                 background: url(../images/blue-arrow.png) no-repeat right 6px; | ||||
|                 padding-right: 10px; | ||||
|             } | ||||
|             .sf-reset ul, ol { | ||||
|                 padding-left: 20px; | ||||
|             } | ||||
|             .sf-reset li { | ||||
|                 padding-bottom: 18px; | ||||
|             } | ||||
|             .sf-reset ol li { | ||||
|                 list-style-type: decimal; | ||||
|             } | ||||
|             .sf-reset ul li { | ||||
|                 list-style-type: none; | ||||
|             } | ||||
|             .sf-reset .symfony-blocks-install { | ||||
|                 overflow: hidden; | ||||
|             } | ||||
|             .sf-reset .symfony-install-continue { | ||||
|                 font-size: 0.95em; | ||||
|                 padding-left: 0; | ||||
|             } | ||||
|             .sf-reset .symfony-install-continue li { | ||||
|                 padding-bottom: 10px; | ||||
|             } | ||||
|             .sf-reset .ok { | ||||
|                 color: #fff; | ||||
|                 font-family: "Lucida Sans Unicode", "Lucida Grande", Verdana, Arial, Helvetica, sans-serif; | ||||
|                 background-color: #6d6; | ||||
|                 padding: 10px; | ||||
|                 margin-bottom: 20px; | ||||
|             } | ||||
|             .sf-reset .ko { | ||||
|                 background-color: #d66; | ||||
|             } | ||||
|             .sf-reset p.help { | ||||
|                 padding: 12px 16px; | ||||
|                 word-break: break-word; | ||||
|             } | ||||
|             .version { | ||||
|                 text-align: right; | ||||
|                 font-size: 10px; | ||||
|                 margin-right: 20px; | ||||
|             } | ||||
|             .sf-reset a, | ||||
|             .sf-reset li a { | ||||
|                 color: #08C; | ||||
|                 text-decoration: none; | ||||
|             } | ||||
|             .sf-reset a:hover, | ||||
|             .sf-reset li a:hover { | ||||
|                 color: #08C; | ||||
|                 text-decoration: underline; | ||||
|             } | ||||
|             .sf-reset textarea { | ||||
|                 padding: 7px; | ||||
|             } | ||||
|         </style> | ||||
|     </head> | ||||
|     <body> | ||||
|         <div id="content"> | ||||
|             <div class="header clear-fix"> | ||||
|                 <div class="header-logo"> | ||||
|                     <img src="" alt="Symfony" /> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="search"> | ||||
|                   <form method="get" action="http://symfony.com/search"> | ||||
|                     <div class="form-row"> | ||||
|  | ||||
|                       <label for="search-id"> | ||||
|                                 <img src="" alt="Search on Symfony website" /> | ||||
|                       </label> | ||||
|  | ||||
|                       <input name="q" id="search-id" type="search" placeholder="Search on Symfony website" /> | ||||
|  | ||||
|                       <button type="submit" class="sf-button"> | ||||
|                           <span class="border-l"> | ||||
|                             <span class="border-r"> | ||||
|                                 <span class="btn-bg">OK</span> | ||||
|                             </span> | ||||
|                         </span> | ||||
|                       </button> | ||||
|                     </div> | ||||
|                    </form> | ||||
|                 </div> | ||||
|             </div> | ||||
|  | ||||
|             <div class="sf-reset"> | ||||
|                 <div class="block"> | ||||
|                     <div class="symfony-block-content"> | ||||
|                         <h1 class="title">Configuration Checker</h1> | ||||
|                         <p> | ||||
|                             This script analyzes your system to check whether is | ||||
|                             ready to run Symfony applications. | ||||
|                         </p> | ||||
|  | ||||
|                         <?php if ($hasMajorProblems): ?> | ||||
|                             <h2 class="ko">Major problems</h2> | ||||
|                             <p>Major problems have been detected and <strong>must</strong> be fixed before continuing:</p> | ||||
|                             <ol> | ||||
|                                 <?php foreach ($majorProblems as $problem): ?> | ||||
|                                     <li><?php echo $problem->getTestMessage() ?> | ||||
|                                         <p class="help"><em><?php echo $problem->getHelpHtml() ?></em></p> | ||||
|                                     </li> | ||||
|                                 <?php endforeach; ?> | ||||
|                             </ol> | ||||
|                         <?php endif; ?> | ||||
|  | ||||
|                         <?php if ($hasMinorProblems): ?> | ||||
|                             <h2>Recommendations</h2> | ||||
|                             <p> | ||||
|                                 <?php if ($hasMajorProblems): ?>Additionally, to<?php else: ?>To<?php endif; ?> enhance your Symfony experience, | ||||
|                                 it’s recommended that you fix the following: | ||||
|                             </p> | ||||
|                             <ol> | ||||
|                                 <?php foreach ($minorProblems as $problem): ?> | ||||
|                                     <li><?php echo $problem->getTestMessage() ?> | ||||
|                                         <p class="help"><em><?php echo $problem->getHelpHtml() ?></em></p> | ||||
|                                     </li> | ||||
|                                 <?php endforeach; ?> | ||||
|                             </ol> | ||||
|                         <?php endif; ?> | ||||
|  | ||||
|                         <?php if ($symfonyRequirements->hasPhpIniConfigIssue()): ?> | ||||
|                             <p id="phpini">* | ||||
|                                 <?php if ($symfonyRequirements->getPhpIniConfigPath()): ?> | ||||
|                                     Changes to the <strong>php.ini</strong> file must be done in "<strong><?php echo $symfonyRequirements->getPhpIniConfigPath() ?></strong>". | ||||
|                                 <?php else: ?> | ||||
|                                     To change settings, create a "<strong>php.ini</strong>". | ||||
|                                 <?php endif; ?> | ||||
|                             </p> | ||||
|                         <?php endif; ?> | ||||
|  | ||||
|                         <?php if (!$hasMajorProblems && !$hasMinorProblems): ?> | ||||
|                             <p class="ok">All checks passed successfully. Your system is ready to run Symfony applications.</p> | ||||
|                         <?php endif; ?> | ||||
|  | ||||
|                         <ul class="symfony-install-continue"> | ||||
|                             <?php if ($hasMajorProblems || $hasMinorProblems): ?> | ||||
|                                 <li><a href="config.php">Re-check configuration</a></li> | ||||
|                             <?php endif; ?> | ||||
|                         </ul> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div class="version">Symfony Standard Edition</div> | ||||
|         </div> | ||||
|     </body> | ||||
| </html> | ||||
							
								
								
									
										
											BIN
										
									
								
								web/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								web/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 6.4 KiB | 
							
								
								
									
										5
									
								
								web/robots.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								web/robots.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| # www.robotstxt.org/ | ||||
| # www.google.com/support/webmasters/bin/answer.py?hl=en&answer=156449 | ||||
|  | ||||
| User-agent: * | ||||
| Disallow: | ||||
		Reference in New Issue
	
	Block a user