https://certbot.eff.org/lets-encrypt/debianstretch-apache
# apt-get install python-certbot-apache -t stretch-backports
# certbot --authenticator webroot --installer apache
https://certbot.eff.org/lets-encrypt/debianstretch-apache
# apt-get install python-certbot-apache -t stretch-backports
# certbot --authenticator webroot --installer apache
app.php w sekcji Log => [
'auth' => [ 'className' => 'Cake\Log\Engine\FileLog', 'path' => LOGS, 'file' => 'auth', 'url' => env('LOG_AUTH_URL', null), 'scopes' => ['auth'], ],
.htaccess – 14 dni czas sesji
php_value session.gc_maxlifetime 1209600
https://github.com/servocoder/RichFilemanager/wiki/Configuration-options
Instalacja projektu w katalogu filemanager
# composer create-project --prefer-dist servocoder/richfilemanager filemanager
# chown -R www-data.www-data filemanager/userdata
– ustawienie virtualnego serwera np. https://mymanager.pl
– dostęp poprzez podany URL
Zmiana userfiles na media
# nano filemanager/connectors/php/filemanager.php $local->setRoot('media', true, true);
Dostęp do pliku
https://mymanager.pl/media/dir/file.png
# apt-get -y install ifupdown resolvconf # ifconfig
Obecna konfiguracja DHCP
enp0s3: flags=4163<UP,BROADCAST,RUNNING,MULTICAST> mtu 1500 inet 192.168.1.6 netmask 255.255.255.0 broadcast 192.168.1.255
# nano /etc/network/interfaces
Master
nano /etc/bind/named.conf.local
https://www.itzgeek.com/debian/configure-dns-server-on-debian-9.html
Instalacja bind9
# apt-get install -y bind9 bind9utils bind9-doc dnsutils
Globalna konfiguracja
# cd /etc/bind/ # less named.conf - nie zmieniany # nano named.conf.local - plik do edycji
https://wkhtmltopdf.org
https://github.com/FriendsOfCake/CakePdf
Pobrać plik instalacyjny silnika ze strony i zainstalować
# dpkg -i wkhtmltox_0.12.5-1.stretch_amd64.deb
Podlinkować do domyślnej lokalizacji
# ln -s /usr/local/bin/wkhtmltopdf /usr/bin/wkhtmltopdf
use Cake\Database\Expression\QueryExpression; ... function incrementCounters() { $expression = new QueryExpression('view_count=view_count +1'); $this->updateAll([$expression], ['published' => true]); }
// In src/Model/Table/OrdersTable.php public function buildRules(RulesChecker $rules) { $check = function($order) { if($order->shipping_mode !== 'free') { return true; } return $order->price >= 100; }; $rules->add($check, [ 'errorField' => 'shipping_mode' , 'message' => 'Nie ma bezpłatnej dostawy poniżej 100zł' ]); return $rules; } // W kontrolerze $order->price = 50; $order->shipping_mode = 'free' ; $ordersTable->save($order); // Returns false
// In src/Model/Table/UsersTable.php public function validatePasswords($validator) { $validator->add('password2', 'no-misspelling' , [ 'rule' => [ 'compareWith' , 'password' ], 'message' => 'Hasła nie są takie same.' , ]); ... return $validator; }
Dołączenie modeli do pobranego już zasobu
$articles = $this->Articles->find()->all(); $withMore = $this->Articles->loadInto($articles, ['Comments', 'Users']);
Plik konfiguracyjny tinyMce
Przechowuje w tabeli ilość rekordów podrzędnych (dzieci). Definiowany jest na tabeli podrzędnej ( belongsTo() ). Tabela nadrzędna musi mieć kolumnę typu int (np. comment_count)
class CommentsTable extends Table { public function initialize(array $config) { $this->belongsTo('Articles'); $this->addBehavior('CounterCache' , [ 'Articles' => [ 'comment_count' ] ]); } }
Można zastosować CounterCache w powiązaniach belongsToMany() tylko z opcją through. Wówczas behaviour definiuje się w tabeli łączącej gdyż posiada powiązania belongsTo().
Wprowadzenie warunku – zlicza tylko komentarze bez spamu
$this->addBehavior('CounterCache' , [ 'Articles' => [ 'comment_count' => [ 'conditions' => [ 'Comments.spam' => false] ] ] ]);
Obsługuje złożone struktury hierarchiczne. Zapewnia wiele metod dostępu, prezentacji, generowania list, ścieżek i manipulacji elementami.
Kolumny wymagane w tabeli
parent_id lft rght level - przechowuje poziom (opcjonalna) model - lub country własna kolumna dla opcji skope (opcjonalna)
Izolowanie procesorów renderowania stron – bezpieczeństwo
chrome://flags Strict site isolation Site isolation trial opt-out
chrome://flags/#enable-site-per-process
class ArticlesTable extends Table
{
public function initialize(array $config)
{
$this->setDisplayField('title');
}
}
$query = $articles->find('list' , [ 'keyField' => 'slug' , 'valueField' => 'title' 'groupField' => 'author_id' ]); $data = $query->toArray();
$query = $articles->find('list' , [ 'keyField' => 'id' , 'valueField' => 'author.name' ])->contain([ 'Authors' ]);
$query->enableHydration(false);
Dostępne do zmiany przez formularz
class User extends Entity
{
protected $_accessible = [
'id' => false,
'*' => true
]
}
$article->accessible('title' , false); $article->set($properties, [ 'guard' => false]);
Właściwości wirtualne uwzgl. w konwersji na array/json
class User extends Entity
{
protected $_virtual = [ 'full_name' ];
}
$user->virtualProperties([ 'full_name' , 'is_admin' ]);
Nie będą exportowane do formatu array/json
class User extends Entity
{
protected $_hidden = [ 'password' ];
}
$user->hiddenProperties([ 'password' , 'recovery_question' ]);
$article->has('title' ); // czy jest i czy ma wartość $article->isEmpty('title' ); // czy jest pusta $article->hasValue('title' ); // czy ma wartość $article->isNew(); // nie zapisany jeszcze w bazie $article->isDirty('title' ); // czy zmieniono title $article->comments[] = $newComment; $article->setDirty('comments' , true); // ustaw - zmieniono $dirtyFields = $entity->getDirty(); // pobierz wszystkie falgi $article->clean(); // wyczyść wszystkie flagi dirty
$entity->unsetProperty('name'); $user->unsetProperty('beers'); // user belongsToMany beers $user->has('beers'); // false // Aby znowu mieć dostęp - ładujemy ręcznie $user= $this->Programmers->loadInto($user, ['Beers']);
Definiowanie właściwości wirtualnej (nie będzie zapisana do bazy)
protected function _getFullName() { return $this->_properties[ 'first_name' ] . ' ' . $this->_properties[ 'last_name' ]; }
echo $user->full_name;
Cake – korzystanie z drugiej bazy danych
namespace App\Model\Table; use Cake\ORM\Table; class ArticlesTable extends Table { public static function defaultConnectionName() { return 'replica_db' ; } }
$user['age'] = $user['birth_date']->diff(new \DateTime)->y;
$controllerName = $this->request->getParam('controller' ); $params = $this->request->getAttribute('params' ); $passedArgs = $this->request->getParam('pass' ); // URL is /posts/index?page=1&sort=title $page = $this->request->getQuery('page' ); $query = $this->request->getQueryParams(); // Input with a name 'MyModel[title]' $title = $this->request->getData('MyModel.title' ); $host = $this->request->env('HTTP_HOST' ); $env = $this->request->getServerParams(); // /subdir/articles/edit/1?page=1 $here = $request->getRequestTarget(); $base = $request->getAttribute('base'); // /subdir $base = $request->getAttribute('webroot'); // /subdir/ $jsonData = $this->request->input('json_decode' ); $data = $this->request->input('Cake\Utility\Xml::build' , [ 'return' => 'domdocument' ]); $this->request->is('post') $this->request->is('ajax') // X-Requested-With = XMLHttpRequest $this->request->is('json') // application/xml lub text/xml .json (ext) $userName = $this->request->session() ->read('Auth.User.name' );
$request->domain(); $request->subdomains(); $request->host(); $request->getMethod(); $this->request->allowMethod([ 'post' , 'delete']);
1. Przeniesienie kodu – FTP
2. Kopia bazy danych – takie same: charset, collate
3. Zmiany parametrów nowej bazy danych
# nano wp-config.php dbname dbuser dbpass dbhost
4. Zmiany domeny w tabeli wp_options:
mysql> select * from wp_options where option_id=1 or option_id=2; +-----------+-------------+-----------------+----------+ | option_id | option_name | option_value | autoload | +-----------+-------------+-----------------+----------+ | 1 | siteurl | https://old.pl | yes | | 2 | home | https://old.pl | yes | +-----------+-------------+-----------------+----------+ mysql> update wp_options set option_value='https://new.pl' where option_value='https://old.pl'
5. Opcjonalnie można zmienić wpisy w tabeli wp_posts. Wszystkie tabele nowej bazy danych można przeszukać ze względu na nazwę starej domeny.
User z dostępem z sieci
mysql> GRANT SELECT,INSERT,UPDATE,DELETE, EXECUTE ON dbname.* TO 'username'@'%' IDENTIFIED BY 'passwd';
Pakiety do pracy z CakePHP3, Moodle3
# dpkg -l | grep php . ii libapache2-mod-php7.0 ii php-common ii php7.0 ii php7.0-cli ii php7.0-common ii php7.0-curl ii php7.0-gd ii php7.0-intl ii php7.0-json ii php7.0-mbstring ii php7.0-mcrypt ii php7.0-mysql ii php7.0-opcache ii php7.0-readline ii php7.0-soap ii php7.0-sqlite3 ii php7.0-xml ii php7.0-xmlrpc ii php7.0-zip
Klient – Kod HTML
-- head <script src='https://www.google.com/recaptcha/api.js'></script> -- form - przycisk reCaptcha <div class="g-recaptcha" data-sitekey="62LczhlYUAAAAAAYW6nsSPsXJ-mulLH6w7TxFH4Pa"></div>
Serwer – kod PHP
if($_POST['send']) { $response = $_POST['g-recaptcha-response']; if($response) { $url = "https://www.google.com/recaptcha/api/siteverify"; $secret = "62LczhlYUAAAAADTnGOz8GRl9c66Rwn4GDSVsChAg"; $remoteip = $_SERVER['REMOTE_ADDR']; $params = [ 'secret' => $secret, 'remoteip' => $remoteip, 'response' => $response ]; $post_params = http_build_query($params); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_HEADER, false); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); curl_setopt($ch, CURLOPT_POST, true); curl_setopt($ch, CURLOPT_POSTFIELDS, $post_params); $response = curl_exec($ch); curl_close($ch); $resp_tab = json_decode($response, true); if( $resp_tab['success'] ) { //Przetwarzanie i wysłanie formularza ... $info = "Formularz wysłany"; } else { $info = "Nieprawidłowa weryfikacja testu w Google. Kod błęd: " .$resp_tab['error-codes'][0]; } } else { $info = "Błędny wybór lub nie wypełniłeś chptcha."; } $smarty->assign('info', $info); }
Przykład obsługi rezerwacji po obu stronach – klient serwer
https://www.darkoperator.com/installing-metasploit-in-ubunt/
https://www.webupd8.org/2014/03/how-to-install-oracle-java-8-in-debian.html