2012年11月14日水曜日

PHP フォームでアップロードできるファイル数の上限

ちょいとハマったので覚書です。

PHP は CentOS 5.7 の標準である 5.1.6 を使用しています。

フォームに <input type="file" /> を30個ほど並べたところ、$_FILES に20個分しか入っていません???

アップロードできるファイル数に制限があったかなぁと調べてみたのですが見当たりません???

ファイルアップロード設定オプション
名前デフォルト変更可能変更履歴
file_uploads"1"PHP_INI_SYSTEMPHP <= 4.2.3 では PHP_INI_ALL、PHP 4.0.3 以降で利用可能
upload_tmp_dirNULLPHP_INI_SYSTEM
max_input_nesting_level64PHP_INI_PERDIRPHP 5.3.9 以降で利用可能。
max_input_vars1000PHP_INI_PERDIRPHP 5.3.9 以降で利用可能。
upload_max_filesize"2M"PHP_INI_PERDIRPHP <= 4.2.3 では PHP_INI_ALL
max_file_uploads20PHP_INI_SYSTEMPHP 5.2.12 以降で利用可能。

上記を見ると 5.2.12 以降では php.ini の設定で変えられるようです。

PHP_INI_SYSTEM となっていますので、httpd.conf のバーチャルホスト設定に下記の1行を追加
php_value max_file_uploads  40

アパッチを再起動してみましたが状況が変わりません。

やっぱ、5.1.6 では実装されていないのかな?と思いましたが諦めきれずに php.ini に直接書いてやりました。

アパッチを再起動すると全部取得できるではないですか!?

max_file_uploads は 5.1.6 でも効果はあるが、php.ini に直接書かなければならないということのようです。

2012年5月11日金曜日

SJIS で HTMLPurifier を使う悩み

ひさしぶりの書き込みです^^;

今回管理画面から設定するメッセージにHTMLを使用できるようにしました。
携帯サイトがあるので文字コードは SJIS です。

管理画面からの入力なので、比較的セキュリティは緩くてもいいかなと思ったのですが、何買ったら心配なので HTML Purifier にて、ヤバいタグや属性を除去するように。

$config = HTMLPurifier_Config::createDefault();
$config->set( 'Core.Encoding', 'SJIS' );
$config->set( 'Core.Language', 'ja' );
$config->set( 'HTML.AllowedElements', $g_aryHTMLPurifierAllowTag );
$objPurifier = new HTMLPurifier( $config );
$str = $objPurifier->purify( $str );

ここで困ったのが © や ® などの HTML 特殊文字です。

フォームからはちゃんと © や ® という形で送られてくるのですが HTML Purifier を通すと消えてなくなっている・・・

で調べてみると、HTML  Purifier は、© や ® を © や ® に変換してしまうことが判明、また、set( 'Core.Encoding', 'SJIS' ); していると SJIS に変換できない文字は消してしまうようです。

とりあえず、© ® を別のキーワード(全角が安全?)に変換してから purify() してその後に戻してやるというヤクザな方法で逃げました^^;

2011年11月4日金曜日

JSON を使って配列の初期化

JSON の使い方のページを見るとオブジェクトの初期化方法を書かれているものが多く、単なるの配列の初期化方法はあまり書かれていません。 備忘録として記しておきます。

 1. 単純な配列を初期化
var a = ["abc", "def"];

2. 連想配列(オブジェクト)を初期化
var a = {"a":"abc", "b":"def"};

2011年10月6日木曜日

PHP HTMLタグの属性をパースする

HTMLタグの属性をパースする関数を作って見ました。
//====================================================================
//! タグの属性をパースする
//! 属性名はアルファベット, 記号(-_), 数字を認めます。
//! 値は "' での括りを認めます。
//! @param string $str タグの属性文字列
//! 例 type=submit type="submit" type='submit' checked
//! @return array キーが属性名の配列
//====================================================================
function aryParseTagAttribute( $str )
{
 //=== 正規表現 ===================================================
// $ma の要素番号
//  1                2 34                  5 67                    8 9
 $re = <<<_EOL_
/\s*([A-Za-z0-9_\-]+)(=(([^"'][^\s]*[^"'])|('(([^'\\\\]|\\\\.)*)')|("(([^"\\\\]|\\\\.)*)")))?/
_EOL_;
 $iMatches = preg_match_all( $re, $str, $ma );

 $aryAttr = array( );
 for ( $i = 0; $i < $iMatches; ++ $i )
 {
  //=== 属性名 =================================================
  $strName = $ma[1][$i];

  //=== 値を取得 ===========================================
  // type="submit"
  // $ma[] は (かっこの順にサブ文字列を格納している
  if ( !empty( $ma[4][$i] ) )
   $strValue = $ma[4][$i]; // 括り無し
  else if ( !empty( $ma[6][$i] ) )
   $strValue = $ma[6][$i]; // '' 括り
  elseif ( !empty( $ma[9][$i] ) )
   $strValue = $ma[9][$i]; // "" 括り
  else
   $strValue = '';   // 値なし

  //============================================================
  $aryAttr[$strName] = $strValue;
 }

 //================================================================
 return $aryAttr;
}

◆テスト
$s = <<<_EOL_
type="text" name='DATE' onclick="func( '\"test\"' ); return" width=100
_EOL_;

$a = aryParseTagAttribute( $s );
print_r( $a );

◆実行結果
Array
(
    [type] => text
    [name] => DATE
    [onclick] => func( '\"test\"' ); return
    [width] => 100
)

2011年7月14日木曜日

PHP 住所から緯度/経度を取得

Google Geocoding API を使って住所から緯度経度を求めます。(めちゃめちゃ簡単です)

Windows 環境ですので文字コードは SJIS です。API呼び出し時に UTF-8 に変換しています。

APIのレザルトは JSON で受け取って配列にデコードしています。

このレザルトには住所などの情報も含まれていますが JSON ですので Windows で使用する場合は SJIS への変換が必要となります。

<?php
//================================================
// 住所 → 緯度/経度変換
//================================================
function strAddrToLatLng( $strAddr )
{
    $strRes = file_get_contents(
         'http://maps.google.com/maps/api/geocode/json'
        . '?address=' . urlencode( mb_convert_encoding( $strAddr, 'UTF-8' ) )
        . '&sensor=false'
    );
    $aryGeo = json_decode( $strRes, TRUE );
    if ( !isset( $aryGeo['results'][0] ) )
        return '';


    $strLat = (string)$aryGeo['results'][0]['geometry']['location']['lat'];
    $strLng = (string)$aryGeo['results'][0]['geometry']['location']['lng'];
    return $strLat . ',' . $strLng;
}


//================================================
echo strAddrToLatLng( '東京都青梅市' );
?>

実行結果

35.7879958,139.2758298

詳しい情報は Google Geocoding API サービス をご覧ください。

Google Static Maps API

携帯などで使う Java Script を使わない地図表示 API です。通常の Google Maps API に比べるととても手軽に使えます。

例: <img src="http://maps.google.co.jp/maps/api/staticmap?center=東京都青梅市&zoom=18&size=640x400&maptype=roadmap&sensor=false&markers=color:red|label:A|東京都青梅市" />



簡単に説明します。

URL: http://maps.google.co.jp/maps/api/staticmap

パラメーター

center カンマ区切り緯度経度または住所を示す文字列(UTF-8 & URL-encode)
zoom ズームレベル(0 ~ 21)
size(必須) 画像のサイズです。書式は [x]x[y] (例: 640x480)最大 640x640
format 画像のフォーマット
png8 または png(デフォルト): 8 ビットの PNG 形式
png32: 32 ビットの PNG 形式
gif: GIF 形式
jpg: JPEG 圧縮形式
jpg-baseline: 非プログレッシブの JPEG 圧縮形式
maptype 作成する地図のタイプ
roadmap:(デフォルト)Google マップ ウェブサイトで通常表示される標準の道路地図画像
satellite: 航空写真です。
terrain: 物理的な地形図画像
hybrid: 航空写真と道路地図を組み合わせたもの
mobile 地図を携帯端末で表示するかどうかを指定 (true or false)
language マップ タイルのラベルを表示する言語
markers マーカー
sensor(必須)アプリケーションがユーザーの位置情報を取得するのにセンサーを用いているか (true or false)

マーカー

マーカーはパラメーターを|で区切って記述
markers=markerStyles|markerLocation1|markerLocation2|...

マーカースタイル

size: マーカーのサイズを {tiny, mid, small} の中から指定
color: 24 ビット カラー(color=0xFFFFCC など)、または {black, brown, green, purple, yellow, blue, gray, orange, red, white}
label: {A~Z、0~-9} のセットから大文字の英数字を 1 文字
icon: マーカーのカスタム アイコンとして使用するための URL を指定。使用できる画像形式は PNG、JPEG、GIF(PNG推奨)
shadow:(デフォルトは true)画像に影を付けるかどうかを指定

2011年6月24日金曜日

Google Maps API 住所から地図を表示する

Google Maps API を使って住所から緯度経度を求めその地点の地図を表示してみました。

先にmapオブジェクトを作ってから住所を元に緯度経度をセットすると灰色の地図が結構長い時間表示されてしまうので、緯度経度取得のコールバックの中でmapオブジェクトを作成しています。

この中ではまったのはマーカーの表示でした。最初、タイトルを設定せずにマーカーを構築していたのですが、それだとマーカーが表示されませんでした><;
function OnLoad( )
{
 var latlng;
 var strAddress = "東京都青梅市東青梅1-7-7";

 var geocoder = new google.maps.Geocoder();
 geocoder.geocode
 ( 
  {
   'address': strAddress
  }
  , function( results, status )
  {
   if ( status != google.maps.GeocoderStatus.OK )
   {
    Debug( "status=" + status );
    return;
   }

   latlng = results[0].geometry.location;

   // 地図を作成
   var map = new google.maps.Map
   (
    document.getElementById("map_canvas")
    , { 
     zoom: 15,
     center: latlng,
     mapTypeId: google.maps.MapTypeId.ROADMAP
    }
   );

   // マーカーを作成
   var marker = new google.maps.Marker( { positon: latlng, map: map } );
   var marker = new google.maps.Marker
   (
    {
       position: latlng
     , map: map
     , title: "(株)クレアンスメアード"
    }
   ); 

   // ポップアップ設定
   var infowindow = new google.maps.InfoWindow
   (
    {
       content: '(株)クレアンスメアード<br>0428-23-3120<br>'
     , size: new google.maps.Size( 450, 150 )
    }
   );
   google.maps.event.addListener
   (
      marker
    , 'click'
    , function( )
    {
     infowindow.open( map, marker );
    }
   );
  }
 );
}