
Собственно, поведение Золотова (T-вирус просроченный завезли) абсолютно неудивительно. Как и всех росгвардейцев.
В Windows 7 и выше есть дополнительный механизм безопасности UAC (Контроль учетных записей пользователей). Обычно пользователь заходит в систему под учетной записью с ограниченными правами, а когда приложению для каких-то целей (установка драйверов, изменение файлов в системных папках и т.д.) требуются админские права, то соответствующий запрос появляется на экране у пользователя.
В принципе, можно вообще отключить UAC, но это понизит безопасность системы, т.к. любое приложение (в т.ч. и зловредное) сможет без всякого запроса поднять себе права до уровня администратора, и сделать с системой все, что ему угодно.
А вот для отдельных программ полезно было бы иметь возможность отключить уведомления. Например, у меня в автозагрузке есть несколько служебных программ, которым требуются права администратора, но неудобно каждый раз копытом набирать номер при каждой загрузке системы прихлопывать вылетающие окна с этим запросом 🙂
На самом деле способов есть минимум 3:
— Скачать программку AdmLink и ей воспользоваться (тогда приложение можно будет запускать и пользователю вообще не входящему в группу «Администраторы»)
— Скачать Microsoft Application Compatibility Toolkit, и воспользоваться им.
— И третий, приглянувшийся мне, запускать программу с повышением прав с помощью «Планировщика заданий»
Способ собственно тем и приглянулся, что не надо ничего качать, планировщик заданий есть на любой машине, а вот интернет не везде. Да, способ будет работать только для пользователей, входящих в группу «Администраторы» (для домашнего компьютера, при установке Windows по умолчанию, пользователь обычно в нее входит).
Создаем задачу
Для примера создадим ярлык для запуска Редактора реестра (regedit.exe
)
1. Запускаем планировщик заданий Пуск —> Все программы —> Стандартные —> Служебные —> Планировщик заданий или в командной строке/через «Выполнить» (Win+R):
taskschd.msc /s
2. Справа в колонке Действие кликаем на Создать задачу:
3. В появившемся окне на вкладке Общие вводим в поле Имя какое-нибудь название, желательно латиницей и без пробелов, чтоб не морочиться потом с кавычками. Например, regedit_adm
. Можно также оставить описание (в поле Описание), чтоб не запутаться если что, например, Запуск Regedit с правами администратора
.
4. Обязательно ставим галочку Выполнить с наивысшими правами.
5. Переходим на вкладку Действия. и жмем кнопку Создать…
6. В появившемся окне в поле Действие оставляем Запуск программы, а в поле Программа или сценарий прописываем путь к нужной программе или BAT/CMD файлу. При необходимости в соответствующих полях можно указать рабочую папку или необходимые аргументы командной строки.
7. Нажимаем ОК, потом еще раз ОК, таким образом сохраняем задачу.
Задача появится в Библиотеке планировщика заданий, где ее можно при необходимости посмотреть, изменить или удалить, когда станет не нужна.
Закрываем планировщик задач
1. Создаем ярлык на рабочем столе. Клик правой кнопкой и Создать —> Ярлык
2. В появившемся окне прописываем следующее:
C:\Windows\System32\schtasks.exe /run /tn regedit_adm
где regedit_adm
— имя созданной вами задачи.
3. Нажимаем Далее, вводим имя ярлыка, например Regedit с правами администратора
4. Можно поменять в свойствах ярлыка значок для красоты. Кстати, внутри Regedit32.exe
ВНЕЗАПНО обнаружился прикольный значок в винтажном стиле Windows 3.11
Запускаем ярлык. Окно UAC не появилось
ФАНФАРЫ!
Понадобилось тут перекодировать URL с русскими буквами в нормальный текст, т.е. нечто вида https://ru.wikipedia.org/wiki/URL#%D0%9A%D0%BE%D0%B4%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5_URL
в читабельный вид https://ru.wikipedia.org/wiki/URL#Кодирование_URL
В MSDN сходу нашлось решение для .NET Framework 4, а вот для 2.0 так сразу не нашлось, но оказалось, что все-таки решение есть.
string text = https://ru.wikipedia.org/wiki/URL#%D0%9A%D0%BE%D0%B4%D0%B8%D1%80%D0%BE%D0%B2%D0%B0%D0%BD%D0%B8%D0%B5_URL
text = text.Replace("+", " ");
text = System.Uri.UnescapeDataString(text);
string text = https://ru.wikipedia.org/wiki/URL#Кодирование_URL
text = System.Uri.EscapeDataString(text);
На самом деле, тот случай, когда мопед совершенно не мой, и я просто объяву разместил.
1. https://ru.wikipedia.org/wiki/URL#Кодирование_URL
2. Сам способ нашел здесь, у няшного серфера и программера.
Html and Uri String Encoding without System.Web
Там еще есть и способ кодирования эскейп-последовательностей HTML.
Ну чтоб не потерять, еще и класс на PasteBin и Mega.NZ
Жалуются тут, что последний на старых компах падает и виснет.
Нашел старый, кому надо, вот ссылка на DriverPackSolutions 11:
https://mega.nz/#!dQUTlaxL.
Ссылка без ключа, тому ще, кто сильно жаловался, ключ был по почте отправлен. Но если кому сей раритет вдруг будет нужен, пишите в комментарии на ЛЖР, я дам, если вежливо попросить.
Про зарезервированные диапазоны и то, как проверить, попадает ли в них адрес IPv4, я уже писал и приводил решение данной задачи на PHP. Теперь вариант на C#.
Понадобится функция IPConverter.IPToInt32(IP)
, которую я описывал в прошлой заметке, аналог функции ip2long()
PHP
В остальном все делается также, как и в PHP-примере.
1. Заводим двумерный массив, содержащий IP начала диапазона, конечный IP и описание диапазона:
string[,] SpecList = new string[,] { {"0.0.0.0","0.255.255.255", "Current network"}, {"255.255.255.255","255.255.255.255", "Broadcast"}, {"255.0.0.0","255.255.255.255", "Reserved by the IETF, broadcast"}, {"10.0.0.0","10.255.255.255", "Private network"}, {"100.64.0.0","100.127.255.255", "Shared Address Space"}, {"127.0.0.0","127.255.255.255", "Loopback"}, {"169.254.0.0","169.254.255.255", "Link-local"}, {"172.16.0.0","172.31.255.255", "Private network"}, {"192.0.0.0","192.0.0.7", "DS-Lite"}, {"192.0.0.170","192.0.0.170", "NAT64"}, {"192.0.0.171","192.0.0.171", "DNS64"}, {"192.0.2.0","192.0.2.255", "Documentation example"}, {"192.0.0.0","192.0.0.255", "Reserved by the IETF"}, {"192.88.99.1","192.88.99.1", "IPv6 to IPv4 Incapsulation"}, {"192.88.99.0","192.88.99.255", "Anycast"}, {"192.168.0.0","192.168.255.255", "Private network"}, {"198.51.100.0","198.51.100.255", "Documentation example"}, {"198.18.0.0","198.19.255.255", "Test IP"}, {"203.0.113.0","203.0.113.255", "Documentation example"}, {"224.0.0.0","224.255.255.255", "Multicast"}, {"240.0.0.0","240.255.255.255", "Future reserved"} };
2. Пишем функцию, которая перебирает массив, и проверяет, попал ли указанный IP в один из диапазонов. Если попал — возвращает описание диапазона, если нет — пустую строку. IP-адреса конвертируются функцией IPConverter.IPToInt32(IP)
в целые числа и сравниваются:
string CheckSpecDiaps(string IP) { int uip = IPConverter.IPToInt32(IP); for (int i = 0; i < SpecList.GetLength(0); i++) { int start = IPConverter.IPToInt32(SpecList[i, 0]); int end = IPConverter.IPToInt32(SpecList[i, 1]); string desr = SpecList[i, 2]; if (uip >= start && uip <= end) { return desr; } } return string.Empty; }
Для работы с IP-адресами в .NET Framework есть класс IPAddress
из пространства имен System.Net
bool IsIP(string IP) { try { IPAddress addr = IPAddress.Parse(IP); } catch { return false; } return true; }
Класс IPAddress
и его функция Parse
распознает сокращенную форму записи IP-адреса, например, адрес 4.0.0.1
можно записать как 4.1
, и он будет таки корректным. Функция ToString()
класса IPAddress
всегда возвращает строку IP-адреса в полном виде, т.е. n.n.n.n
, это можно использовать для преобразования адреса в полную форму записи:
string ToStandForm(string IP) { if (!IsIP(IP)) { return string.Empty; } IPAddress addr = IPAddress.Parse(IP); return addr.ToString(); }
byte[] GetBytesBE(string IP) { IPAddress addr = IPAddress.Parse(IP); return addr.GetAddressBytes(); }
Примечание: GetAddressBytes()
выдает байты IP-адреса всегда в сетевом порядке, т.е. big-endian, в таком же, в котором байты идут при обычной записи IP. Если вдруг понадобится little-endian порядок, например, для того, чтобы преобразовать IP-адрес в целое число на little-endian системе, то массив надо перевернуть.
byte[] GetBytesLE(string IP) { IPAddress addr = IPAddress.Parse(IP); byte[] addrbytes = addr.GetAddressBytes(); Array.Reverse(addrbytes); return addrbytes; }
Аналог PHP-функции ip2long()
или сишной inet_addr()
int IPToInt32(string IP) { IPAddress addr = IPAddress.Parse(IP); //получаем байты адреса, они всегда в big-endian byte[] addrbytes = addr.GetAddressBytes(); //IP в виде Int32 big-endian int n = BitConverter.ToInt32(addrbytes,0); //если в системе little-endian порядок if (BitConverter.IsLittleEndian) { n = IPAddress.NetworkToHostOrder(n); //надо перевернуть } return n; }
Изменить порядок байт можно и без использования Array.Reverse()
, если число типа short
или int
. Для этого в классе IPAddress
присутствуют функции NetworkToHostOrder()
, которая меняет порядок байт в переменной с сетевого (big-endian) на используемый на локальной машине, и функция HostToNetworkOrder()
, которая делает обратную операцию.
Скорее, чтоб два раза не вставать, делается точно так же, как и преобразование в знаковый int
, но функции NetworkToHostOrder()
и HostToNetworkOrder()
тут уже не помогут, так что придется переворачивать массив байт адреса с помощью Array.Reverse()
:
public static uint IPToUInt32(string IP) { IPAddress addr = IPAddress.Parse(IP); byte[] addrbytes = addr.GetAddressBytes(); if (BitConverter.IsLittleEndian) { Array.Reverse(addrbytes); } return BitConverter.ToUInt32(addrbytes, 0); }
Навеяно вопросом на stackowerflow
Объединил все вышеуказанные функции в отдельный класс.
Известно, что любые данные хранятся в памяти компьютера в виде последовательности байт, байт обычно равен 8 битам (но бывают и исключения, о которых сейчас не будем). Числа не исключение, например, когда мы определяем переменную int
, фактически мы говорим компилятору «выдай нам 4 байта по такому-то адресу в памяти, для записи числа от -2147483648 до 2147483647». Конкретный адрес, конечно, заботливо выбирает для нас среда .NET, и обычно, лезть туда руками не нужно, среда все прекрасно за нас сделает. Пока не случается какой-нибудь хитрый случай, и тут уже приходится думать самому. Когда работаешь с однобайтовыми числами, обычно никаких чудес не происходит — байт он и на десктопной машине с Windows или Linux байт, и на сервере байт, и даже на роутере, телефоне и утюге, скорее всего будет тем же самым байтом. Но все было бы хорошо, если бы одного байта хватало всем 🙂
Как уже сказано выше, тот же int
, это не один байт, а 4, и вот тут кроется проблема. Эти самые байты можно хранить в памяти как угодно, хоть через один, хоть в шахматном порядке.
На наше счастье, такие извращения, если и встречаются, то очень редко. В большинстве вычислительных устройств байты (одного числа, например типа int
), хранятся либо в последовательности от старшего к младшему big-endian, либо от младшего к старшему little-endian. В случае big-endian первым идет старший разряд числа, а остальные по ранжиру за ним, в случае ltiite-endian — наоборот, первым идет самый младший, последним — самый старший.
Чтобы было проще представить, предположим, что в 1 байт влезает не десятичное число от 0 до 255 (256 значений), а всего 10 значений, т.е. числа от 0 до 9.
В таком случае число 1234
будет «четырехбайтовым», и в случае, если оно записывается в порядке big—endian, то в памяти оно будет храниться в обычном для нас виде, как последовательность 1 2 3 4
, поэтому, порядок big-endian еще называют прямым порядком байт. Если же число хранится в порядке little-endian, то в памяти оно будет выглядеть, как последовательность 4 3 2 1
, поэтому порядок little-endian называют обратным порядком байт.
В реальном компьютере, в котором байт восьмибитный, и хранит положенные ему 256 значений, все происходит точно так же.
Итак:
Big-endian — «от старшего к младшему», он же прямой, сетевой (поскольку принят в качестве стандартного порядка байт при передаче данных по сети), Motorola byte order (использовался в процессорах Motorola, а не в честь уехавшего на социальном лифте деятеля).
В big-endian формате хранятся IP-адреса.
Little-endian — «От младшего к старшему», обратный, интеловский (используется в процессорах Intel), VAX (использовался на платформе VAX).
int
, содержащую число 16909060
и два массива байт byte[] LittleEndian
и byte[] BigEndian
, содержащие его представление в виде последовательности байт в обратном и прямом порядке:
int Constant = 16909060;
byte[] LittleEndian = new byte[] {4, 3, 2, 1};
byte[] BigEndian = new byte[] {1, 2, 3, 4 };
Да, число подобрано специально, чтобы было красивое представление его в массиве байт. 🙂
Попробуем провести обратное преобразование с помощью класса BitConverter
, который как раз и предназначен для получения из последовательности байт чисел соответствующих типов:
BitConverter.ToInt32(LittleEndian,0); //результат - 16909060
BitConverter.ToInt32(BigEndian, 0); //результат - 67305985
Класс BitConverter
в теории должен использовать при преобразовании тот порядок байт, который используется на данной машине, соответственно, правильное число 16909060
было получено при преобразовании массива LittleEndian
.
Этот простой пример иллюстрирует почему так важен порядок байт. Если вы получили данные в порядке, отличном от того, который используется в среде, где данные в результате обрабатываются, и соответствующей проверки не было произведено, то вы рискуете получить ошибочные данные.
Средствами .NET
Для того, чтобы средствами .NET узнать, какой порядок байт используется на машине, где выполняется ваша программа, можно посмотреть в переменную IsLittleEndian
класса BitConverter
. Если она принимает значение true
, то используется, соответственно, порядок little-endian (обратный):
if (BitConverter.IsLittleEndian) { Console.WriteLine("little-endian"); } else { Console.WriteLine("most likely big-endian"); }
Создадим перечисление ByteOrder
(для красоты :):
private enum ByteOrder { BigEndian = 0, LittleEndian = 1, Unknow = 3 }
И напишем функцию:
ByteOrder DetectBO() { int Constant = 16909060; byte[] LittleEndian = new byte[] { 4, 3, 2, 1 }; byte[] BigEndian = new byte[] { 1, 2, 3, 4 }; if (BitConverter.ToInt32(BigEndian, 0) == Constant) return ByteOrder.BigEndian; if (BitConverter.ToInt32(LittleEndian, 0) == Constant) return ByteOrder.LittleEndian; return ByteOrder.Unknow; }
Примечание: в тексте в кодировке UTF-16 можно определить UTF-16LE или UTF-16BE при помощи BOM, при его наличии
А вот тут у .NET Framework’а все как-то печально, частично могут помочь статические функции класса System.Net.IPAddress
NetworkToHostOrder()
и HostToNetworkOrder()
, преобразующие, соответственно, число, полученное в big-endian в формат, используемый на данной машине и наоборот, но они довольно ограничены, поддерживают только long
, int
и short
значения, не поддерживают работу с беззнаковыми числами и числами с плавающей запятой, а также с массивами байт. Вот способ преобразования порядка байт:
1. Преобразовать исходное число в массив байт с помощью BitConverter.GetBytes()
или взять готовый массив, если он есть.
2. Перевернуть массив функцией Array.Reverse()
3. Преобразовать развернутый массив обратно, с помощью одной из функций класса BitConverter
:
byte [] ConvArray = BitConverter.GetBytes(BigEndianValue);
Array.Reverse(ConvArray);
ushort LittleEndianValue = BitConverter.ToUInt16(ConvArray,0);
Минус — относительно медленная функция Array.Reverse
. И BitConverter
тоже не самый быстрый класс, зато довольно наглядно.
Внимание! Функция Array.Reverse()
не создает копии массива, а работает с указанным массивом прямо в памяти. Если в функцию, в которой используется Array.Reverse()
будет передан массив из вызывающей подпрограммы, и функция Array.Reverse()
будет к массиву применена, то массив изменится. Такое поведение может породить труднообнаруживаемую ошибку, поэтому, если массив в оригинальном виде планируется еще где-либо использовать, то перед Array.Reverse()
надо сделать его копию с помощью Array.Copy()
.
1. Порядок байтов
2. Разбираемся с прямым и обратным порядком байтов
3. MSDN
Зачем:
— Во-первых, уязвимость, которая вроде как давно устранена, но осадочек остался.
— Во-вторых, кривым SVG можно положить браузер, чем, например, пользуется вандал-кремлебот, пакостя на самом свободном блогосервисе рунета
В общем, проще нахуй отрезать, чем вылечить.
Как:
1. Набираем в адресной строке about:config
2. Устанавливаем опцию svg.enabled
в false
Способ универсальный
Кроме Firefox и в Chrome работает, но тут инструкция для Firefox, для остального должно быть также.
1. Ставим блокировщик рекламы и прочей гадости uBlock Origin
2. Нажимаем на значок плагина (1 на картинке ниже) и входим в панель управления плагином (2 на картинке ниже)
Переключаемся на вкладку Мои фильтры и добавляем туда следующий код:
! block all svg
*.svg
Баста! Зловредный SVG вырезан с корнями!
Да, все, наверное, уже видели, слышали и читали. Но раз от меня требуют комментариев, то прокомментирую. Хотелось бы поиронизировать на тему, что единственная церковь которая несет свет — это горящая церковь, но не буду. Во-первых, объект был все-таки музейный, а не РПЦ-шный, хотя службы там проводились. А во-вторых, об этом еще Дима писал, при церкви жила куча жидохвостов (кошек), и если какие-то кошки сгорели, то это пиздец, и виновный будет радоваться, если ему придется гореть в аду.
Предварительная причина возгорания — неосторожное обращение с огнем.
Вот здесь можно посмотреть на фото уже бывшей церкви
А на церкви было лого инопланетян из сериала Предел
Поскольку я сейчас ~ в 150 Км от города, подробности из первых рук не могу сообщить. Но если пострадали коты — тогда очень жаль.
Unix-time — способ хранения времени, используемый обычно в POSIX-совместимых системах. Определяется, как количество секунд, прошедших с 01.01.1970 00:00:00, может встречаться в логах, базах данных, и т.д., как отметка времени (timestamp). Следующий алгоритм поможет преобразовать Unix-время в DateTime
:
1. Заводим переменную DateTime
и инициализируем ее точкой отсчета Unix-time, т.е. полночью 1 января 1970 года:
DateTime origin = new DateTime(1970, 1, 1, 0, 0, 0);
2. Используем метод
AddSecounds()
, чтобы добавить количество секунд, в качестве аргумента передаем переменную, содержащую Unix-time:
origin.AddSeconds(UnixTime);
Вся функция:
private DateTime UnixTimeToDateTime(double UnixTime) { DateTime origin = new DateTime(1970, 1, 1, 0, 0, 0); return origin.AddSeconds(UnixTime); }
Если Unix-time хранится в целочисленной переменной, меняем double
на нужный тип, компилятор сам преобразует ее в тип double
, нужный AddSecounds
.
Наконец я нашел откомпилированную версию сабжа, и теперь ее можно скачать бесплатно, без СМС и вирусов отсюда. IDE, тащемта, дерьмо, но это можно простить, ибо писалось все это дело в 2008 году, но до сей поры ничего лучшего никто не написал.
Не буду подробно рассказывать, чего мне это стоило, но было нагажено одному болгарскому файл-хостингу и инсталлятор пришлось избавлять от вирей, распространяющих рекламу и майнящих криптовалюту.
Избавили и выкладываем в нормальный открытый доступ. Кто умеет в торренты, помогите и выложите на Рутрекер и TPB, оно маленькое 20 мегабайт всего. Но очень полезное.
Итак: Roadsend PHP Studio (IDE) + Roadsend PHP Compiler
Компилятор PHP для Windows.
Умеет компилировать PHP в EXE в нескольких режимах:
— Console application — на выходе получается стандартное консольное приложение для Win32
— Desktop GUI Application — десктопное оконное приложение, для того, чтоб его собрать нужны еще библиотеки и инструментарий PHP-GTK, в комплекте нет
— Compiled Web Application (FastCGI) — скомпилированное Web-приложение, в экзешник встраивается FastCGI.
— Standalone Web Application (MicroServer) — почти тоже самое, но в экзешник встраивается Web-сервер от Roadsend.
-Library — компилирует полноценный DLL из PHP-кода, который можно встроить в свое приложение, хоть на C#, хоть на C++, хоть на небе, хоть на Аллахе.
Компилировать может в двух режимах — dynamic и static, В статическом режиме в экзешник внедряются DLL от Sheme на котором написан сам компилятор, и их не надо таскать за собой, в динамическом режиме — будет надо, но в статическом вырастает размер готового экзешника, примерно на мегабайт.
Проблемы:
— В Windows 7 не работает штатный лаунчер для IDE (сам компилятор работает).
Пришлось вспомнить молодость и расчехлить OllyDebug, вытащить из лаунчера команду, которой он запускает основной файл IDE. По итогу вышел вот такой небольшой батник, который заменяет оригинальный лаунчер:
@echo off set ROADSEND_ROOT=#put you Roadsend root directory here, e.g. C:\roadsend# set SH=\bin\sh.exe set IDE=\pcc\bin\loon.exe if not exist %ROADSEND_ROOT%%SH% goto NOSH if not exist %ROADSEND_ROOT%%IDE% goto NOIDE echo Files OK. echo Launch Roadsend IDE... %ROADSEND_ROOT%%SH% --login -c "cd /pcc/bin && start loon" :NOSH echo File %ROADSEND_ROOT%%SH% not found exit /b 1 :NOIDE echo File %ROADSEND_ROOT%%IDE% not found exit /b 1
куда вместо #put you Roadsend root directory here, e.g. C:\roadsend#
надо вставить директорию, куда установлена IDE с компилятором.
Впрочем, Леша, заглаживая вчерашний демарш, написал свой лаунчер с азартными играми и продажными женщинами на C# и даже с инсталлятором.
Подробнее про Roadsend PHP Compiler и Roadsend PHP Studio можно прочесть в статье на Хабре
Оригинальный инсталлятор Roadsend PHP Studio (IDE + компилятор)
Лаунчер для Windows 7 (C#) с Mega.nz
Лаунчер для Windows 7 (C#) с GitHub
Лаунчер для Windows 7 (BAT/CMD) с Mega.nz
Лаунчер для Windows 7 (BAT/CMD) с GitHub
Статья о Roadsend PHP Compiler на Хабр Копия
Исходники лаунчера
BAT/CMD лаунчер на PasteBin
НЕФОРМАТ УДАЛЕН.
Леша Сайлент
Не потерплю такого неформатного говна на наших ресурсах. Типа давай гусей не дразнить
Консольный вариант скрипта.
Т.е. это уже не совсем «определение местоположения посетителя сайта», это просто интерфейс к базе данных Sypex Geo, который можно использовать в консольных приложениях на PHP.
Что переделано:
Переделан изначальный скрипт, для работы под консоль, или, если хотите, а качестве stand-alone скрипта. Для консольного приложения очень важно вернуть код завершения, чтобы далее его могло обработать либо следующая программа в командном файле (BAT/CMD), либо вызвавшая скрипт, как внешний процесс, программа.
Таким же образом был немного подправлен SxGeo.php, распространяемый разработчиками Cypex Geo.
Исправления таковы:
— Везде, где скрипт завершает работу, была добавлена выдача кода завершения
— Таким же образом подправлен класс SxGeo, поставляемый разработчиками Cypex Geo.
Краткая справка скрипта:
Вызов скрипта:
consxgeo.php <ip-address> <database-path> [FIELD_NAME]
consxgeo.php --spec
consxgeo.php --help
--help - Помощь
--spec - Список зарезервированных диапазонов IP-адресов
<ip-address> - Адрес IPv4 (например, 8.8.8.8)
<database-path> - Путь к файлу SxGeoCity.dat
[FIELD_NAME] - Для включения в вывод результата работы скрипта источника IP-адреса. По умолчанию, это значение = MANUAL
Коды завершения:
0 - OK
1 - Предупреждение
3 - Вывод информационных сообщений
4 - Ошибка или неправильные параметры скрипта
Структура ответа:
IP|ISO_CODE|COUNTRY_NAME|CTNR_LAT|CTNR_LON|
REGION_ISO|REGION_NAME|CITY_NAME|CTY_LAT|CTY_LON
|FIELD|MESSAGE|
Частный IP:
Сокращенный IP-адрес:
Конь-Пилятор
(из классификатора «Эсхатологические мутанты»)
ketmar@ljr справедливо заметил, что наши скрипты принимают только IP-адреса в формате
xxx.xxx.xxx.xxx
, где xxx
— десятеричное число от 0 до 255, а IP может быть записан и в виде шестнадцатеричных, и в виде восьмеричных чисел, и даже смешано. Также IP-адрес может быть неполным. Например, адрес 4.0.0.1
можно записать как 4.1
В наших скриптах корректность IP проверялась регулярным выражением, т.к. все равно им же производился поиск IP в полях HTTP_*
, в которых IP может передаваться прокси-серверами.
Но валидацию IP можно провести и без использования регулярных выражений. Другое дело, поиск без него уже не произведешь, и это может создать определенные проблемы, если прокси-сервер, например, будет передавать IP в виде шестнадцатеричных чисел или преобразовывать, где возможно, полные адреса в неполные.
В PHP 5.2.0 появилась функция filter_var()
, которая позволяет проверять строки на соответствие разным форматам, не только IP, но и, например URL или e-mail.
Для проверки IP она вызывается так:
filter_var($ip,FILTER_VALIDATE_IP);
где:
$ip
— строка, содержащая IP-адрес
FILTER_VALIDATE_IP
— встроенный фильтр для IP-адреса
Функция возвращает TRUE
, если строка прошла проверку, и FALSE
если нет.
А не очень хорош способ тем, что валидацию проходят только IP, соответствующие формату «четыре десятеричных числа от 0 до 255, разделенных точкой», т.е. работает точно также, как и наша старая функция
Такое ощущение, что разработчики PHP пытаются избавить мир от зоопарка способов записи IP-адреса, и привести все, наконец, к единому знаменателю 🙂
Пример с использованием функции filter_var
ketmar@ljr предлагал использовать сишную функцию
inet_addr()
, но где ж я ее возьму-то в PHP, подумал я, и зря подумал. 🙂 Функция ip2long($ip)
, которая использовалась для проверки попадания IP-адреса в определенный диапазон Копия ее аналог. Если переменная $ip
содержит корректный IP-адрес, то функция вернет целое число, если в IP-адресе ошибка, то функция вернет FALSE
или -1
.
Примечание: Функция ip2long()
возвратит FALSE для IP 255.255.255.255
в PHP 5 <= 5.0.2 и -1 в PHP 5 <=5.2.4 на 64-битных системах. Это было исправлено в PHP 5.2.5, теперь возвращается 4294967295. В 32-битных системах будет возвращено -1 из-за целочисленного переполнения.
Так что проверку на корректность IP можно делать в функции-обертке, добавив код-заглушку для IP 255.255.255.255
:
function IsIP($ip) { //преобразуем в нижний регистр, на случай шестнадцатиричных чисел $ip=strtoupper($ip); //ip2long в некоторых версиях php //некорректно реагирует на адрес 255.255.255.255 //делаем небольшую заглушку if ($ip == '255.255.255.255'||$ip == '0xff.0xff.0xff.0xff'|| $ip == '0377.0377.0377.0377') { return true; } $tolong=ip2long($ip); if ($tolong == -1||$tolong===FALSE) return FALSE; else return TRUE; }
В PHP есть функция long2ip()
, которая делает обратное преобразование, т.е. преобразует целое число в IP-адрес в привычном виде. Если их совместить, то можно преобразовывать неполные адреса в полные:
function fulladdr($ip) { //преобразует неполные адреса в полные //для информации $tolong=ip2long($ip); return long2ip ($tolong); }
Пример работы скрипта:
isip.php?ip=4.1
Внедрил последний метод в рабочие скрипты, залил на сервер, и оказалось, что метод не работает. Ну не то, чтобы не совсем не работает, а работает точно так же, как старая функция с регулярным выражением, т.е. возвращает TRUE
только если IP-адрес записан в виде четырех десятеричных чисел от 0 до 255, разделенных точкой. Неполные адреса, или адреса в других системах счисления IP-адресами не считаются. Функция ip2long()
возвращает FALSE
, если ей скормить такой адрес. Уж не знаю, и сходу не нагуглил, отчего такая оказия, то ли такое поведение зависит от настроек PHP, то ли от сервера. На локальном сервере работало, а на том, который в интернете — перестало. Можно сказать, зря делал, но зато сделал валидацию IP без регулярного выражения.
1. ip2long
2. Ещё раз о filter_var
Нифига не окончание, а второй сезон.
Пока исключительно небольшие исправления.
— Добавили зарезервированные диапазоны IPv4 адресов и соответствующую проверку Копия в скрипты для машинной обработки данных и в красивый скрипт с оформлением, таким образом немножко усовершенствовали скрипт из предыдущей серии.
— Сделали тестовый скрипт, который IP сам ничего не детектит, а просто обращается к SxGeo. Он все равно еще понадобится.
Использование:
onlysxgeo.php?ip=ip-address
Например:
onlysxgeo.php?ip=8.8.8.8
Или:
onlysxgeo.php?ip=ip-address&f=FIELD_NAME
для включения имя поля в ответ скрипта, по умолчанию имя поля MANUAL
Или
onlysxgeo.php?specdiap
Для вывода списка зарезервированных диапазонов адресов IPv4
— По многочисленным просьбам зрителей сделали самый простой скрипт, который просто выводит все IP из заголовков HTTP_*
(например HTTP_VIA
и/или HTTP_X_FORWARDED_FOR
) и REMOTE_ADDR
и источник адреса (имя поля HTTP_*
или REMOTE_ADDR
).
— Завели отдельный репозиторий на GitHub, куда выложили код из всех предыдущих и новых серий
Дальше будет банальщина, наверняка всем известная, но все же, вдруг кто не знает, да и себе от склероза.
Есть у меня маленький, но очень полезный горшочек скрипт, за которым надо постоянно ходить на сервер, а тут оказалось, что он понадобился на машине, где никаких интернетов нет, а данные, которые нужно обработать — есть. Я сначала решил сделать из него экзешник, но потом оставил эту затею, ибо с компиляторами PHP под Виндовоз неожиданно образовался какой-то жуткий геморрой. HipHop от Facebook напрочь отказался собираться, сайт Roadsend PHP Compiler сдох, а опять собирать с исходников GitHub стало после шаманства с HipHop откровенно влом, ну плюс MinGW особым удобством не отличается.
Заметил, кстати, удивительную странность — на машине с 12 Гб оперативной памяти и четвероядерным процессором, но под MinGW и, соответственно, Виндой, сборка происходит медленнее, чем на компьютере с Linux, скромным двухведерным процессором и двумя же Гб ОП.
Впрочем, все эти танцы с бубном были совершенно лишними.
Все оказалось банально и элементарно, php прекрасно работает из консоли. Формат команды:
php.exe [параметры] <скрипт> [параметры_скрипта]
$_SERVER
:$_SERVER['argc']
хранит количество параметров$_SERVER['argv']
их значения. $_SERVER['argv']
сам является индексированным массивом. В элементе 0 содержится имя файла скрипта.
Примечание:
Начиная с PHP 4.3.0, при использовании CLI SAPI переменные $argc и and $argv зарегистрированы и заполнены соответствующими значениями. В более ранних версиях создание этих переменных, так же, как и для CGI или модуля веб-сервера, требует значение on директивы register_globals. Независимо от версии PHP или статуса опции register_global они всегда доступны как элементы массива $_SERVER или $HTTP_SERVER_VARS. Например: $_SERVER[‘argv’]
Скрипт, выводящий в консоль свои параметры:
<?php echo "Console test \n\n"; echo "Arguments count: ".$_SERVER['argc']."\n"; echo "Arguments values:\n"; $i=0; foreach ($_SERVER['argv'] as $arg) { echo $i.": ".$arg."\n"; $i++; } ?>
Если запустить скрипт, например так:
php.exe con-test.php param1 param2 param3 tramparamparamparam
Все это не имело бы большого смысла, если бы для запуска скрипта пришлось бы тащить с собой Web-сервер или весь дистрибьютив PHP, со всеми модулями и прочим. Для минимального запуска PHP под Windows нужно всего 2 файла: php.exe
и php*ts.dll
, где * — версия PHP. Например, для использованного мной PHP5, это php5ts.dll
— Инклюды. Если скрипт подразумевает использование инклюдов, то их лучше держать либо в директории с главным скриптом, либо в поддиректориях, и прописывать в основном скрипте относительные пути. Т.е. так, как это обычно делается на сервере. Иначе PHP будет искать их либо в include_path, который прописан в php.ini, либо вообще неведомо где, точнее ведомо — это указывается как параметр при сборке самого PHP, если мне не изменяет склероз.
— php.ini
Вообще, для запуска PHP он не нужен, но тут может вкрасться чущественная гадость. Если на машине уже установлен PHP, или php.ini
подложен каким-то злоумышленником в каталог по умолчанию, для многих сборок под Windows это %WINDIR%
, то может случиться неприятность.
Сам PHP ищет свой php.ini
сначала в каталоге с самим собой, так что можно туда его подложить.
— Расширения PHP. Можно подложить нужные DLL расширений в подкаталог в каталоге с php.exe и прописать в extension_dir
в php.ini
относительный путь к каналогу, а в соответствующих параметрах extension
— имена DLL
— PHP я цельностянул из Denwer 🙂
— Архив с примером и «портабельным» PHP
1. Создание EXE приложений на PHP Копия
2. Оффлайновые лекционные тетради в клеточку.
Джа-Кузя
(из классификатора «Эсхатологические мутанты»)
В IPv4 имеются зарезервированные диапазоны адресов, в некоторых случаях надо проверить, не попал ли адрес в один из них
Надеюсь, ничего не забыл. Таблица нагло выдрана из лекции, но я сверялся с Википедией и другими интернетами.
Для порядку заведем регулярное выражение для проверки IP
$ip_pattern="#(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)#";
И функцию, которая проверяет, а IP ли вообще у нас на входе:
function isip($ip_str) //соответствие данных формату IP { global $ip_pattern; $ret=FALSE; if (preg_match($ip_pattern,$ip_str)) { $ret=TRUE; } return $ret; }
Теперь функцию, которая будет проверять, попал ли IP в нужный диапазон:
function chkdiapip ($user_ip, $ip_from, $ip_to) //попадает ли ip в нужный диапазон { return ( ip2long($user_ip)>=ip2long($ip_from) && ip2long($user_ip)<=ip2long($ip_to) ); }
Создадим двумерный массив со списком диапазонов:
$spec_list = array( array ("0.0.0.0","0.255.255.255", "Current network"), array ("255.255.255.255","255.255.255.255", "Broadcast"), array ("255.0.0.0","255.255.255.255", "Reserved by the IETF, broadcast"), array ("10.0.0.0","10.255.255.255", "Private network"), array ("100.64.0.0","100.127.255.255", "Shared Address Space"), array ("127.0.0.0","127.255.255.255", "Loopback"), array ("169.254.0.0","169.254.255.255", "Link-local"), array ("172.16.0.0","172.31.255.255", "Private network"), array ("192.0.0.0","192.0.0.7", "DS-Lite"), array ("192.0.0.170","192.0.0.170", "NAT64"), array ("192.0.0.171","192.0.0.171", "DNS64"), array ("192.0.2.0","192.0.2.255", "Documentation example"), array ("192.0.0.0","192.0.0.255", "Reserved by the IETF"), array ("192.88.99.0","192.88.99.255", "Anycast"), array ("192.88.99.1","192.88.99.1", "IPv6 to IPv4 Incapsulation"), array ("192.168.0.0","192.168.255.255", "Private network"), array ("198.51.100.0","198.51.100.255", "Documentation example"), array ("198.18.0.0","198.19.255.255", "Test IP"), array ("203.0.113.0","203.0.113.255", "Documentation example"), array ("224.0.0.0","224.255.255.255", "Multicast"), array ("240.0.0.0","240.255.255.255", "Future reserved") );
И функцию, которая будет все это безобразие в цикле перебирать, и если IP попадет в какой-то из диапазонов - выдаст диапазон и описание. Если не попадет - вернет -1.
function get_spec_diap ($user_ip, $listspec) { for ($i=0;$i<sizeof($listspec);$i++) { $item = $listspec[$i]; if (chkdiapip($user_ip, $item[0], $item[1])) { return $item[0]."\t".$item[1]."\t".$item[2]."\t\n"; } } return -1; }
Ну шоб два раза не вставать, сделаем еще и функцию, оглашающую весь список:
function printspecdiap ($listspec) { for ($i=0;$i<sizeof($listspec);$i++) { $item = $listspec[$i]; echo $item[0]."\t\t".$item[1]."\t\t\t".$item[2], "\t\t\t\n"; } }