2024年5月11日土曜日

メモ:Visual StudioとMicrosoft C++ (MSVC) コンパイラ ツールのバージョンの対応

  • Visual Studio 2015: v140
  • Visual Studio 2017: v141
  • Visual Studio 2019: v142
  • Visual Studio 2022: v143

方法: ターゲット フレームワークおよびプラットフォームのツールセットを変更する | Microsoft Learn

log4php

  • log4phpを試した備忘録です
  • Apache log4php - Welcome - Apache log4php の[Download]より apache-log4php-2.3.0-src.zip をダウンロード
  • 解凍先は任意、今回は C:\DATA01\TempDel の下に解凍
  • C:\DATA01\TempDel\log4php_config.xmlを作成
<?xml version="1.0" encoding="UTF-8"?>
<configuration xmlns="http://logging.apache.org/log4php/">
    <appender name="appenderfile" class="LoggerAppenderFile">
        <param name="file" value="C:\DATA01\TempDel\application.log" />
        <param name="append" value="true" />
        <layout class="LoggerLayoutPattern">
            <param name="conversionPattern" value="%date{Y-m-d H:i:s.u} %logger %-5level  %sessionid %file(%line) %msg%n" />
        </layout>
    </appender>
    <root>
        <appender_ref ref="appenderfile" />
    </root>
</configuration>
  • C:\DATA01\TempDel\log4php_test.phpを作成
<?php
// ログの時間を日本時間にするため
date_default_timezone_set("Asia/Tokyo");

require_once(dirname(__FILE__) . "/apache-log4php-2.3.0/src/main/php/Logger.php");

// 設定ファイルの読み込み
Logger::configure(dirname(__FILE__) . "/log4php_config.xml");

// ロガー名:任意
$logger = Logger::getLogger("myLogger");

// 以下ログの出力
$logger->debug("メッセージ debug");
$logger->info("メッセージ info");
$logger->warn("メッセージ warning");
$logger->error("メッセージ error");
$logger->fatal("メッセージ fatal");
?>
  • ブラウザからlog4php_test.phpにアクセスすると、C:\DATA01\TempDel\application.log に以下のようにログが出力される
2024-05-10 20:11:04.171 myLogger DEBUG   C:\DATA01\TempDel\log4php_test.php(14) メッセージ debug
2024-05-10 20:11:04.173 myLogger INFO    C:\DATA01\TempDel\log4php_test.php(15) メッセージ info
2024-05-10 20:11:04.173 myLogger WARN    C:\DATA01\TempDel\log4php_test.php(16) メッセージ warning
2024-05-10 20:11:04.173 myLogger ERROR   C:\DATA01\TempDel\log4php_test.php(17) メッセージ error
2024-05-10 20:11:04.173 myLogger FATAL   C:\DATA01\TempDel\log4php_test.php(18) メッセージ fatal
  • サイズでローテーションさせる場合は、log4php_config.xmlを以下のように記載
<?xml version="1.0" encoding="UTF-8"?>
<configuration xmlns="http://logging.apache.org/log4php/">
    <appender name="rollingFile" class="LoggerAppenderRollingFile">
        <param name="file" value="C:\DATA01\TempDel\application.log" />
        <param name="append" value="true" />
        <param name="maxFileSize" value="10MB" />
        <param name="maxBackupIndex" value="10" />
        <param name="compress" value="false" />
        <layout class="LoggerLayoutPattern">
            <param name="conversionPattern" value="%date{Y-m-d H:i:s.u} %logger %-5level  %sessionid %file(%line) %msg%n" />
        </layout>
    </appender>
    <root>
        <appender_ref ref="rollingFile" />
    </root>
</configuration>
  • 日付でローテーションさせる場合は、log4php_config.xmlを以下のように記載
<?xml version="1.0" encoding="UTF-8"?>
<configuration xmlns="http://logging.apache.org/log4php/">
    <appender name="dailyFile" class="LoggerAppenderDailyFile">
        <param name="file" value="C:\DATA01\TempDel\application_%s.log" />
        <param name="append" value="true" />
        <param name="datePattern" value="Y-m-d" />
        <layout class="LoggerLayoutPattern">
            <param name="conversionPattern" value="%date{Y-m-d H:i:s.u} %logger %-5level  %sessionid %file(%line) %msg%n" />
        </layout>
    </appender>
    <root>
        <appender_ref ref="dailyFile" />
    </root>
</configuration>
  • application_2024-05-10.log というファイル名でログが出力される
    日付のフォーマットは datePattern で指定できる
    月ごとにする場合は Y-m など
  • 日付でログ出力しつつ、ブラウザにも表示する場合は、log4php_config.xmlを以下のように記載
<?xml version="1.0" encoding="UTF-8"?>
<configuration xmlns="http://logging.apache.org/log4php/">
    <appender name="dailyFile" class="LoggerAppenderDailyFile">
        <param name="file" value="C:\DATA01\TempDel\application_%s.log" />
        <param name="append" value="true" />
        <param name="datePattern" value="Y-m-d" />
        <layout class="LoggerLayoutPattern">
            <param name="conversionPattern" value="%date{Y-m-d H:i:s.u} %logger %-5level  %sessionid %file(%line) %msg%n" />
        </layout>
    </appender>
    <appender name="echo" class="LoggerAppenderEcho">
        <param name="htmlLineBreaks" value="true" />
        <layout class="LoggerLayoutSimple" />
    </appender>
    <root>
        <appender_ref ref="dailyFile" />
        <appender_ref ref="echo" />
    </root>
</configuration>

2024年5月6日月曜日

Apache バージョンアップ 2024

はじめに

以下のソフトをバージョンアップします

バージョンアップ前 バージョンアップ後
Apache 2.2.19 (Win32) Apache 2.4.59 (Win64)
PHP 5.3.5 (Win32) PHP 8.3.6 (Win64)
PukiWiki 1.3.5 (euc-jp) PukiWiki 1.5.4 (utf-8)
(新規) JDK 22.0.1 (Win64)
(新規) Tomcat 10.1.23 (Win64)
Apache Subversion 1.8.0 (Win32) Apache Subversion 1.14.3 (Win64)

 

Apache 2.4.59

  • Apache Loungeのサイトから httpd-2.4.59-240404-win64-VS17.zip をダウロードします
  • 解凍してフォルダ名を「Apache2.4.59」として「C:\APPL01\Apache Software Foundation\」にコピーします    
  • 配置後のフォルダパス : C:\APPL01\Apache Software Foundation\Apache2.4.59
  • conf/httpd.confを編集します(日本語を含むパスがある場合はutf-8でファイルを保存します)
修正前 修正後
修正 Define SRVROOT "c:/Apache24" Define SRVROOT "C:/APPL01/Apache Software Foundation/Apache2.4.59"
修正 ServerAdmin admin@example.com ServerAdmin <自分のメールアドレス>
追加   ServerName MYHOSTNAME:80
修正 <IfModule dir_module>
    DirectoryIndex index.html
</IfModule>
<IfModule dir_module>
    DirectoryIndex index.html index.php index.jsp
</IfModule>
追加       AddType image/x-icon .ico
  • Apache2.2からApache2.4でのhttpd.confの書き方の違い  
Apache 2.2 Apache 2.4
Alias /MyPub01/ "C:/DATA01/MyPagePub01/" Alias /MyPub01/ "C:/DATA01/MyPagePub01/"
<Directory "C:/DATA01/MyPagePub01">
    AllowOverride None
    Require all granted
</Directory>
※Aliasだけ指定してDirectoryタグの設定がない場合
Forbidden
You don't have permission to access this resource.
となる
<Directory "C:/xxx/~">
    Order allow,deny
    Allow from all
</Directory>
<Directory "C:/xxx/~">
    Require all granted
</Directory>
すべて許可する
  すべて拒否する場合は
    Require all denied
<Directory "C:/xxx/~">
    Order deny,allow
    Deny from all
    Allow from 192.168.11.8 HOSTNAME1
</Directory>
<Directory "C:/xxx/~">
    Require ip 192.168.11.8
    Require host HOSTNAME1
</Directory>
IP指定、ホスト名指定の許可
  ローカルホストからのアクセスを許可する場合は
    Require local

参考サイト
Apache | Requireディレクティブ:アクセスの許可や拒否などのアクセス制限を行う(Apache2.4以降)

  • ログローテーションの設定(conf/httpd.conf)
修正前 修正後
修正 ErrorLog "logs/error.log" ErrorLog "| bin/rotatelogs.exe logs/error_%Y%m.log 86400"
※rotatelogsは拡張子まで記載が必要
修正 CustomLog "logs/access.log" common CustomLog "| bin/rotatelogs.exe logs/access_%Y%m.log 86400" common
※rotatelogsは拡張子まで記載が必要
  • 古いサービスの削除
  • コマンドプロンプトを管理者として開き、以下を実行します
>cd C:\APPL01\Apache Software Foundation\Apache2.2\bin
>httpd -k uninstall
Removing the Apache2.2 service
The Apache2.2 service has been removed successfully.
  • サービスの登録
>cd C:\APPL01\Apache Software Foundation\Apache2.4.59\bin
>httpd -k install -n "Apache2.4.59"
Installing the 'Apache2.4.59' service
The 'Apache2.4.59' service is successfully installed.
Testing httpd.conf....
Errors reported here must be corrected before the service can be started.
  • サービス登録は成功、起動は失敗したが、コントロールパネルのサービスから起動できたのでOKとします

 

PHP 8.3.6

  • PHP For Windows: Binaries and sources Releasesで「VS16 x64 Thread Safe (2024-Apr-10 15:13:46)」「Zip」を選び、php-8.3.6-Win32-vs16-x64.zip をダウンロードします
  • 解凍してC:\APPL01\php\に配置します
  • 配置後のフォルダパス : C:\APPL01\php\php-8.3.6-Win32-vs16-x64\
  • php.ini-productionをphp.iniとコピーして編集します
  修正前 修正後
修正 doc_root = doc_root = "C:/APPL01/Apache Software Foundation/Apache2.4.59/htdocs"
追加   extension_dir = "C:/APPL01/php/php-8.3.6-Win32-vs16-x64/ext"
修正 ;extension=mbstring extension=mbstring
修正 ;extension=odbc extension=odbc
修正 ;sendmail_from = me@example.com sendmail_from = <自分のメールアドレス>
修正 ;session.save_path = "/tmp" session.save_path = "C:/DATA01/TempDel"
修正 ;mbstring.language = Japanese mbstring.language = Japanese
修正 ;mbstring.internal_encoding = mbstring.internal_encoding = UTF-8
修正 ;mbstring.http_input = mbstring.http_input = pass
修正 ;mbstring.http_output = mbstring.http_output = pass
修正 ;mbstring.encoding_translation = Off mbstring.encoding_translation = Off
修正 ;mbstring.detect_order = auto mbstring.detect_order = UTF-8,SJIS,EUC-JP,JIS,ASCII
修正 ;mbstring.substitute_character = none mbstring.substitute_character = none
修正 ;mbstring.strict_detection = Off mbstring.strict_detection = Off
  • Apacheのhttpd.confを編集します
  修正前 修正後
追加   PHPIniDir "C:/APPL01/php/php-8.3.6-Win32-vs16-x64"
LoadModule php_module C:/APPL01/php/php-8.3.6-Win32-vs16-x64/php8apache2_4.dll
追加       AddType application/x-httpd-php .php
  • 動作確認をします
    C:\APPL01\Apache Software Foundation\Apache2.4.59\htdocs\phpinfo.php を作成します
<?php
    phpinfo();
?>

 

Microsoft Access データベース エンジン 2016 再頒布可能コンポーネント

>cd C:\DATA01\TempDel\accessdatabaseengine_X64
>aceredist.msi /passive

参考サイト
Microsoft Acces Database Engine 2016 の32bit/64bit版の共存方法 #access – Qiita

 

PHPアプリの修正

  • Shift_JIS→UTF-8に変更、PHP 5.3.5→PHP 8.3.6に変更したことによる修正をします
  • Shift_JIS→UTF-8に関する変更
修正前 修正後
ファイルの文字コード:Shift_JIS ファイルの文字コード:UTF-8(BOM無し)
以下のソフトを使って一括変換します
FileCode Checker のダウンロード・使い方 - フリーソフト100
<meta http-equiv="Content-Type" content="text/html; charset=Shift_JIS"> <meta charset="UTF-8">
odbc_exec($conn_id,$wkSQL); odbc_exec($conn_id, mb_convert_encoding($wkSQL ,"SJIS-win", "UTF-8"));
MS-ACCESSへSQL実行時にSQLをUTF-8→Shift_JISに変更
odbc_result($re01, 2) mb_convert_encoding(odbc_result($re01, 2) , "UTF-8", "SJIS-win")
MS-ACCESSから結果取得時に値をShift_JIS→UTF-8に変更
  • PHP 5.3.5→PHP 8.3.6に関する変更
修正前 修正後
<? <?php
※phpもつけないと認識されなかった
session_is_registered($key) isset($_SESSION[$key])
session_register($key) $_SESSION[$key] = "value";
※セッション変数名のみ登録したい場合は、記載しない
session_unregister($key) unset($_SESSION[$key])
$HTTP_SESSION_VARS $_SESSION
$HTTP_POST_VARS $_POST
$HTTP_GET_VARS $_GET
※パラメータがないことが想定される場合は、以下のように記載
$p_sheet_name = isset($_GET["SheetName"]) ? $_GET["SheetName"] : "";
コンストラクタ
class test01
{
    function test01()
    {
コンストラクタは「__construct()」にする
class test01
{
    function __construct()
    {
ereg_replace mb_ereg_replace
  • その他の変更
修正前 修正後
define("_DBName","ODBCのDNS名"); define("_DBName", "Driver={Microsoft Access Driver (*.mdb, *.accdb)};Dbq=C:/DATA01/MsAccess/test.mdb");
※直接Accessファイルを指定するように変更

 

PukiWiki 1.5.4

  • PukiWiki/Download/1.5.4 - PukiWiki-official から pukiwiki-1.5.4_utf8.zip をダウンロードします
  • 解凍して「C:\DATA01\」の下に配置します
    配置後のフォルダパス : C:\DATA01\pukiwiki-1.5.4_utf8\
  • pukiwiki.ini.phpを修正します
  修正前 修正後
修正 $nowikiname = 0; $nowikiname = 1;
自動リンクさせない
修正 $adminpass = '{x-php-md5}!'; $adminpass = '{x-php-md5}098f6bcd4621d373cade4e832627b4f6';
※{x-php-md5}の後の文字列は <?php echo md5("パスワード"); ?>の結果文字列を記載します
  • Apacheのhttpd.confを修正します
  修正前 修正後
追加   Alias /my-pukiwiki/ "C:/DATA01/pukiwiki-1.5.4_utf8/"
<Directory "C:/DATA01/pukiwiki-1.5.4_utf8">
    Options Indexes FollowSymLinks
    DirectoryIndex index.php
    AllowOverride All
    Require all granted
</Directory>
  • skin/pukiwiki.cssを修正します
修正前 修正後 備考
.style_table {
  padding:0;
  border:0;
  margin:auto;
  text-align:left;
  color:inherit;
  background-color:#ccd5dd;
}
.style_table {
  padding:0;
  border:0;
  margin:1em auto 1em 0;
  text-align:left;
  color:inherit;
  background-color:#ccd5dd;
}
表を左寄せ
.style_th {
  padding:5px;
  margin:1px;
  text-align:center;
  color:inherit;
  background-color:#EEEEEE;
}
.style_th {
  padding:5px;
  margin:1px;
  text-align:center;
  color:inherit;
  background-color:#EEEEEE;
  vertical-align:top;
}
セル内文字を上寄せ
.style_td {
  padding:5px;
  margin:1px;
  color:inherit;
  background-color:#EEF5FF;
}
.style_td {
  padding:5px;
  margin:1px;
  color:inherit;
  background-color:#EEF5FF;
  vertical-align:top;
}
セル内文字を上寄せ
  • wikiフォルダの下のファイルを移行をします
    wikiフォルダ下には初期インストール時のファイルも含まれているため必要なファイルのみ移行します
    移行時にWiki名についている[ ]は削除したファイル名に変更します
  移行元 移行先
ファイルの文字コード ecu-jp UTF-8
改行コード \n \n
wikiフォルダ下のファイル名 5B5B4C696E7578C0DFC4EA5D5D.txt
{"[[Linux設定]]"をHEXエンコード}.txt
・・・
4C696E7578E8A8ADE5AE9A.txt
{"Linux設定"をHEXエンコード}.txt
・・・
  • 移行に用いたC#プログラム
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.IO;

namespace PukiwikiConversionConsole
{
    class Program
    {
        const string BASE_PATH = @"C:\DATA01\TempDel\";
        const string FROM_FILE_PATH = BASE_PATH + @"wiki_From\";
        const string TO_FILE_PATH = BASE_PATH + @"wiki_To\";

        /// <summary>
        /// ベースパス(BASE_PATH)の下の
        /// wiki_From配下のファイルをUTF-8変換、ファイル名変更して
        /// wiki_Toの配下に保存する
        /// </summary>
        /// <param name="args"></param>
        static void Main(string[] args)
        {
            if (!Directory.Exists(FROM_FILE_PATH))
            {
                Console.WriteLine(FROM_FILE_PATH + " に変換元のwikiファイルを配置してください。");
                Console.ReadKey();
                return;
            }

            if (!Directory.Exists(TO_FILE_PATH))
            {
                Directory.CreateDirectory(TO_FILE_PATH);
            }

            StreamWriter swLog = new System.IO.StreamWriter(
                BASE_PATH + "Log-" + DateTime.Now.ToString("yyyyMMdd-HHmmss") + ".txt",
                false, new System.Text.UTF8Encoding(false));

            swLog.WriteLine(
                "変換元ファイル名"
                + "\t"
                + "更新日時"
                + "\t"
                + "変換元Wiki名"
                + "\t"
                + "変換後Wiki名"
                + "\t"
                + "変換後ファイル名"
                );

            foreach (string fromFileFullName in Directory.GetFiles(FROM_FILE_PATH, "*.txt"))
            {
                ConvertFile(swLog, fromFileFullName);
            }

            swLog.Close();

            Console.WriteLine("完了");
            Console.ReadKey();
        }

        private static void ConvertFile(StreamWriter swLog, string fromFileFullName)
        {
            WikiFile wikiFile = new WikiFile(fromFileFullName);

            // 元ファイル読み込み
            List<string> contents = ReadFromFile(wikiFile.FromFileFullName, Encoding.GetEncoding("euc-jp"));

            // UTF-8の\nで書き込む
            StreamWriter swTo = new System.IO.StreamWriter(
                wikiFile.ToFileFullName,
                false, new System.Text.UTF8Encoding(false));
            foreach (string line in contents) swTo.Write(line + "\n");
            swTo.Close();

            // タイムスタンプを元ファイルに合わせる
            //作成日時の設定
            System.IO.File.SetCreationTime(wikiFile.ToFileFullName, wikiFile.FiFrom.CreationTime);
            //更新日時の設定
            System.IO.File.SetLastWriteTime(wikiFile.ToFileFullName, wikiFile.FiFrom.LastWriteTime);
            //アクセス日時の設定
            System.IO.File.SetLastAccessTime(wikiFile.ToFileFullName, wikiFile.FiFrom.LastAccessTime);

            // ログの書き込み
            swLog.WriteLine(
                wikiFile.FromFileName
                + "\t"
                + wikiFile.FiFrom.LastWriteTime.ToString("yyyy/MM/dd HH:mm:ss")
                + "\t"
                + wikiFile.FromWikiName
                + "\t"
                + wikiFile.ToWikiName
                + "\t"
                + wikiFile.ToFileName
                );

        }

        private static List<string> ReadFromFile(string filePath, Encoding encode)
        {
            string line = "";
            List<string> ret = new List<string>();

            using (StreamReader sr = new StreamReader(filePath, encode))
            {
                while ((line = sr.ReadLine()) != null)
                {
                    ret.Add(line);
                }
            }

            return ret;
        }

        /// <summary>
        /// 変換前のファイル名、変換後のファイル名を格納するクラス
        /// </summary>
        class WikiFile
        {
            private string _fromFileFullName;
            private string _fromFileName;
            private string _fromWikiName;
            private string _toFileName;
            private string _toWikiName;
            private string _toFileFullName;

            FileInfo _fiFrom;

            public WikiFile(string fromFileFullName)
            {
                this._fromFileFullName = fromFileFullName;

                this._fiFrom = new FileInfo(fromFileFullName);

                // 変換元ファイル名と拡張子
                this._fromFileName = this._fiFrom.Name;
                string ext = this._fromFileName.Substring(this._fromFileName.LastIndexOf("."));

                // 変換元Wiki名(拡張子なしでデコード後の変換元ファイル名)
                this._fromWikiName = HexToBin(Encoding.GetEncoding("euc-jp"), this._fromFileName.Substring(0, this._fromFileName.LastIndexOf(".")));

                // 変換後Wiki名(ファイル名内の [] は削除する)
                this._toWikiName = this._fromWikiName.Replace("[", "").Replace("]", "");

                // 変換後ファイル名(エンコード後の変換後ファイル名+拡張子)
                this._toFileName = BinToHex(new System.Text.UTF8Encoding(false), this._toWikiName) + ext; 

                // 変換後ファイル名フルパス
                this._toFileFullName = TO_FILE_PATH + this._toFileName;

            }

            public string FromFileFullName { get { return this._fromFileFullName; } }
            public string FromFileName { get { return this._fromFileName; } }
            public string FromWikiName { get { return this._fromWikiName; } }
            public string ToFileName { get { return this._toFileName; } }
            public string ToWikiName { get { return this._toWikiName; } }
            public string ToFileFullName { get { return this._toFileFullName; } }
            public FileInfo FiFrom { get { return this._fiFrom; } }

            // How to convert this PHP function to C#? - Stack Overflow
            // https://stackoverflow.com/questions/35336150/how-to-convert-this-php-function-to-c
            private static string BinToHex(Encoding enc, string bin)
            {
                byte[] bytes = enc.GetBytes(bin);
                string hexString = HexStringFromBytes(bytes);
                return hexString.ToUpper();
            }

            private static string HexStringFromBytes(byte[] bytes)
            {
                var sb = new StringBuilder();
                foreach (byte b in bytes)
                {
                    var hex = b.ToString("x2");
                    sb.Append(hex);
                }
                return sb.ToString();
            }

            // php hextobin not works exactly in c# - Stack Overflow
            // https://stackoverflow.com/questions/60095429/php-hextobin-not-works-exactly-in-c-sharp
            private static string HexToBin(Encoding enc, string hex)
            {
                return enc.GetString(HexToBinByteArray(hex));
            }

            private static byte[] HexToBinByteArray(string hex)
            {
                int NumberChars = hex.Length;
                byte[] bytes = new byte[NumberChars / 2];
                for (int i = 0; i < NumberChars; i += 2)
                    bytes[i / 2] = Convert.ToByte(hex.Substring(i, 2), 16);
                return bytes;
            }
        }
    }
}

 

JDK 22.0.1

  • JDK Builds from OracleよりOpenJDK JDK 22.0.1 GA Releaseに遷移し、openjdk-22.0.1_windows-x64_bin.zip をダウンロードします
  • openjdk-22.0.1_windows-x64_bin.zip を解凍し、フォルダ名を「openjdk-22.0.1_windows-x64_bin」に変更後に、「C:\APPL01\Ap_Java\」にコピーします
    配置後のフォルダパス : C:\APPL01\Ap_Java\openjdk-22.0.1_windows-x64_bin\
  • バージョンを確認します
>cd C:\APPL01\Ap_Java\openjdk-22.0.1_windows-x64_bin\bin
>java --version
openjdk 22.0.1 2024-04-16
OpenJDK Runtime Environment (build 22.0.1+8-16)
OpenJDK 64-Bit Server VM (build 22.0.1+8-16, mixed mode, sharing)
  • 通常は環境変数を設定しますが、今回は実施しません
変数名 設定内容
JAVA_HOME C:\APPL01\Ap_Java\openjdk-22.0.1_windows-x64_bin
PATH %JAVA_HOME%\bin;~以降は元の値~

参考サイト
2022年のJavaまるわかり!最新バージョンからJavaを取り巻く環境まで | gihyo.jp

 

Tomcat 10.1.23

  • Apache Tomcat® - Welcome!よりApache Tomcat® - Apache Tomcat 10 Software Downloadsに遷移して、apache-tomcat-10.1.23-windows-x64.zip をダウンロードします
  • apache-tomcat-10.1.23-windows-x64.zipを解凍し、フォルダ名を「apache-tomcat-10.1.23-windows-x64」に変更後に、「C:\APPL01\Ap_Java\」にコピーします
    配置後のフォルダパス : C:\APPL01\Ap_Java\apache-tomcat-10.1.23-windows-x64\
  • C:\APPL01\Ap_Java\apache-tomcat-10.1.23-windows-x64\bin\setenv.batを作成し以下を記載します
set JAVA_HOME="C:\APPL01\Ap_Java\openjdk-22.0.1_windows-x64_bin"

 

Apacheとの連携

  • Tomcatのconf/server.xmlを開いて以下を追加します
    (デフォルトではコメントされているので新たに追加)
    <Connector protocol="AJP/1.3"
               address="::1"
               secretRequired="false"
               port="8009"
               redirectPort="8443"
               maxParameterCount="1000"
               />
  • Apacheのconf/httpd.confを編集します
  修正前 修正後
修正 #LoadModule proxy_module modules/mod_proxy.so
#LoadModule proxy_ajp_module modules/mod_proxy_ajp.so
LoadModule proxy_module modules/mod_proxy.so
LoadModule proxy_ajp_module modules/mod_proxy_ajp.so
修正 #LoadModule proxy_wstunnel_module modules/mod_proxy_wstunnel.so LoadModule proxy_wstunnel_module modules/mod_proxy_wstunnel.so
追加   ProxyPass /examples/websocket/ ws://localhost:8080/examples/websocket/
ProxyPassReverse /examples/websocket/ ws://localhost:8080/examples/websocket/
ProxyPass /examples/ ajp://localhost:8009/examples/
#ProxyPassReverse /examples/ ajp://localhost:8009/examples/

 

Apache Subversion 1.14.3

  • GitHub - nono303/win-svn: apache subversion windows builds with httpd modules and java hl から「Download ZIP」を選択して、モジュールをダウンロードします
    今回ダウンロードしたファイルはバージョン1.14.3です
  • ダウンロードしたファイル win-svn-master.zip を解凍します
  • vs17\x64\(depsフォルダ下も含む) の全dllを「C:\APPL01\Apache Software Foundation\Apache2.4.59\bin」の下にコピーします
    同名のファイルがある場合は上書きはしないようにします
    コピーしたファイルは以下のものです
libexpat.dll
libserf-2.dll
libsqlite3.dll
libsvn_client-1.dll
libsvn_delta-1.dll
libsvn_diff-1.dll
libsvn_fs-1.dll
libsvn_fs_fs-1.dll
libsvn_fs_util-1.dll
libsvn_fs_x-1.dll
libsvn_ra-1.dll
libsvn_repos-1.dll
libsvn_subr-1.dll
libsvn_wc-1.dll
libsvnjavahl-1.dll
zlib.dll
  • vs17\x64\ のmod_authz_svn.so、mod_dav_svn.soを「C:\APPL01\Apache Software Foundation\Apache2.4.59\modules」の下にコピーします
mod_authz_svn.so
mod_dav_svn.so
  • (※補足)Apache2.4.59/bin、Apache2.4.59/modulesの下にコピーせずに、vs17/x64を別場所に配置して、soを参照する設定を試しましたが、Apacheは起動するもののリポジトリブラウザでアクセスしようとすると、開けなかったのであきらめました
    Apacheのエラーログで何度もプロセスのリスタートがかかり何度目かにあきらめている感じでした
  • リポジトリ、パスワードファイルの作成は省略します
  • Apacheのconf/httpd.confを修正します
  修正前 修正後
修正 #LoadModule dav_module modules/mod_dav.so
#LoadModule dav_fs_module modules/mod_dav_fs.so
LoadModule dav_module modules/mod_dav.so
LoadModule dav_fs_module modules/mod_dav_fs.so
追加   LoadModule dav_svn_module modules/mod_dav_svn.so
LoadModule authz_svn_module modules/mod_authz_svn.so
追加   <Location /svn/svnrep01>
    SVNListParentPath on
    DAV svn
    SVNPath "C:/DATA01/SVN/SVNREP01"
    #Basic認証の設定
    AuthType Basic
    AuthName "Subversion repositories"
    AuthUserFile "C:/DATA01/SVN/SVNREP01/svnpasswd"
    Require valid-user
</Location>

 

項目 内容
SlikSVN Download SlikSVN Subversion clientからSlik-Subversion-1.14.2b-x64.zipをダウンロード
zip内のSlik-Subversion-1.14.2-x64.msiをUniversal Extractor 2で解凍したがmod_dav_svn.so、mod_authz_svn.soが無かったので断念
TortoiseSVN Downloads · TortoiseSVNからTortoiseSVN-1.14.7.29687-x64-svn-1.14.3.msiをダウンロード
Universal Extractor 2で解凍したがmod_dav_svn.so、mod_authz_svn.soが無かったので断念
VisualSVN Downloads | VisualSVNからApache-Subversion-1.14.3.zip、VisualSVN-Server-5.3.2-x64.msiをダウンロード
解凍後どちらもmod_dav_svn.so、mod_authz_svn.soが無かったので断念
Cirata Premier source for certified subversion binaries | CirataからWANdiscoSubversion_1.14.1-1_apache_24.exeをダウンロード
WANdiscoSubversion_1.14.1-1_apache_24\Apache2\modulesの下にmod_dav_svn.so、mod_authz_svn.soがあったが32bit版のためかロードできなかった
>cd C:\DATA01\TempDel\WANdiscoSubversion_1.14.1-1_apache_24\Apache2\bin
>httpd.exe -v
Server version: Apache/2.4.46 (Win32)
Server built: Oct 13 2020 13:37:23

2015年9月5日土曜日

Apacheサービス登録

【サービス登録】
httpd.exe –k install

【サービス登録解除】
httpd.exe –k uninstall

SVNのユーザ作成

以下のサイトの丸パクリです。

フリーな人生:Windowsで、SVN(Subversion) + Apache Basic認証、ユーザの追加、リポジトリの作成

【インストール】

まずSVNをインストールしてください。
インストールが終了したら、コンソールで以下のコマンドを実行し、正常に行われたかどうか確認します。
svn --version

だめだったらパスが通ってないので、環境変数のPathに追加してください。

あと、Apache2.2もインストールしておいてください。

インストールの仕方とかは他のサイトを参考にしてください。すみません。。。


【Basic認証の設定】

SVNユーザの追加は、ApacheのBasic認証を利用します。
httpd.confで指定した任意のフォルダに、同じくhttpd.confで指定した任意のファイルを作成します。

以下、Apache設定
「httpd.conf」を編集します(Apache2.2/conf下)。

#以下のモジュールのコメントアウトをはずす
LoadModule dav_module modules/mod_dav.so
LoadModule dav_fs_module modules/mod_dav_fs.so

#以下のモジュールの記述を追加。モジュールがなければ、SVNのフォルダから探してmodulesフォルダにコピーする
LoadModule dav_svn_module modules/mod_dav_svn.so
LoadModule authz_svn_module modules/mod_authz_svn.so

#以下の記述を追加
<Location /svn>
  DAV svn
  SVNParentPath "d:/svn"
(※以下が認証の設定)
  AuthType Basic
  AuthName "Subversion repositories"
  AuthUserFile "C:/Apache2.2/etc/svnpasswd"  ←※1
  Require valid-user
</Location>

※1…パスワードファイルの絶対パス。任意の場所で大丈夫なので、拡張子なしの空のファイルを作っておいてください。
あとで説明するユーザの追加時に自動でできたかも。すみません、忘れました。。。

#ちなみに以下のように複数作れます
<Location /svn2>
  DAV svn
  SVNListParentPath on
  SVNParentPath "d:/svn2"

  AuthType Basic
  AuthName "Subversion repositories"
  AuthUserFile "C:/Apache2.2/etc2/svnpasswd"
  Require valid-user
</Location>
<Location /svn3>
  DAV svn
  SVNListParentPath on
  SVNParentPath "d:/svn3"

  AuthType Basic
  AuthName "Subversion repositories"
  AuthUserFile "C:/Apache2.2/etc3/svnpasswd"
  Require valid-user
</Location>


【ユーザの追加】

コマンドプロンプトを起動してください。
最初のユーザは、-cコマンドを付与して作成します。

c:> cd c:/Apache2.2/etc
c:/Apache2.2/etc>
"C:\Apache2.2\bin\htpasswd.exe" -c svnpasswd username(ユーザ名)
このあとパスワードを入力する

2人目以降は-cコマンドを付けずに実行。付けると上書きされるので。

c:/Apache2.2/etc>
"C:\Apache2.2\bin\htpasswd.exe" svnpasswd username(ユーザ名)

これでetcフォルダの下のsvnpasswdファイルに複数のユーザが追加されます。

また、ここで設定したユーザIDがコミット時などのログに利用されるAuthorになります。


【リポジトリ作成】

(1)ディレクトリを作成
mkdir D:\svn\repo

(2)リポジトリとして認識させる
svnadmin create D:\svn\repo


あとはTortoiseSVNとか、Eclipseとか、または他のツールでフォルダごとインポートします。

2015年8月25日火曜日

SVNのディレクトリを完全削除する

一度、ファイルに保存する場合

svnadmin dump 【svnディレクトリ】 | svndumpfilter exclude 【消したいsvnパス】 > 【適当なファイル名】
svnadmin create 【svnディレクトリ】
svnadmin load 【svnディレクトリ】 < 【適当なファイル名】

あらかじめ移動先を作っている場合

svnadmin dump 【svnディレクトリ】 | svndumpfilter exclude 【消したいsvnパス】 | svnadmin load 【svnディレクトリ】


svnadmin dump C:/SvnRep1 | svndumpfilter exclude /trunk/99_XXX | svnadmin load C:/SvnRep2

参照元

SVNでディレクトリを完全削除【Windows】 | grush-blog

[Subversion]巨大なリポジトリから多数の不要ファイルを恒久的に削除する方法 ・ DQNEO起業日記

2012年8月5日日曜日

Operaユーザースクリプトを有効にする

URLに「opera:config#UserPrefs|UserJavaScript」と入力し、「User JavaScript」「User JavaScript on HTTPS」をチェックし、「User JavaScript File」にJSファイルを保存しているフォルダを指定する。
設定後はブラウザの再起動が必要。

20120805_202500_001

2012年7月29日日曜日

FirefoxでYahooのトップページを表示すると重い

FirefoxとFlashプレーヤの相性が悪いことが原因らしい
Flashプレーヤの設定ファイルmms.cfgを編集してオプションを追加すると、スムーズに表示できるようになる

mms.cfgの場所
32bit版 Vista/7の場合
C:\windows\system32\macromed\flash
64bit版 Vista/7の場合
C:\windows\syswow64\macromed\flash

追加するオプション
ProtectedMode=0

参照サイト
FirefoxでFlash、Youtubeが再生出来ない問題2012年6月

2012年6月18日月曜日

[c#]Twitterクライアントを作成するためのリンク集

以下のサイトを参考にすると、作成できるかも。昔はBASIC認証でもアクセスできていたみたいだけ、OAuth認証というものを使わなければアクセスできなくなったらしい。結構難しそう・・・。

C#でOAuthでTwitter - nojimaの日記

Twitterを活用したプログラムをC#で作成しよう

Twitterに自分で作成したアプリケーションを登録する (2012年3月版)

twitterizer(http://code.google.com/p/twitterizer/)

Twitterizer(http://www.twitterizer.net/)

30分で誰でも作れるTwitter Bot開発・運用手順 – Ruby Twitter bot

リンク先の①、②を組み合わせて1分おきに、タイムラインをコンソール出力するプログラムを作ってみました。
初回起動時に保存される設定ファイルは「C:\Users\<ユーザーID>\AppData\Local\ConsoleTwitter」に保存されるようです。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Net;
  6. using System.Xml;
  7. using System.IO;
  8. using System.Security.Cryptography;
  9.  
  10. namespace ConsoleTwitter
  11. {
  12. class Auth {
  13. const string REQUEST_TOKEN_URL = "https://twitter.com/oauth/request_token";
  14. const string ACCESS_TOKEN_URL = "https://twitter.com/oauth/access_token";
  15. const string AUTHORIZE_URL = "https://twitter.com/oauth/authorize";
  16.  
  17. private Random random = new Random();
  18.  
  19. public string ConsumerKey { get; private set; }
  20. public string ConsumerSecret { get; private set; }
  21. public string RequestToken { get; private set; }
  22. public string RequestTokenSecret { get; private set; }
  23. public string AccessToken { get; private set; }
  24. public string AccessTokenSecret { get; private set; }
  25. public string UserId { get; private set; }
  26. public string ScreenName { get; private set; }
  27.  
  28. public Auth(string consumerKey, string consumerSecret) {
  29. ServicePointManager.Expect100Continue = false;
  30. ConsumerKey = consumerKey;
  31. ConsumerSecret = consumerSecret;
  32. }
  33.  
  34. public Auth(string consumerKey, string consumerSecret, string accessToken, string accessTokenSecret, string userId, string screenName) {
  35. ServicePointManager.Expect100Continue = false;
  36. ConsumerKey = consumerKey;
  37. ConsumerSecret = consumerSecret;
  38. AccessToken = accessToken;
  39. AccessTokenSecret = accessTokenSecret;
  40. UserId = userId;
  41. ScreenName = screenName;
  42. }
  43.  
  44. public void GetRequestToken() {
  45. SortedDictionary parameters = GenerateParameters("");
  46. string signature = GenerateSignature("", "GET", REQUEST_TOKEN_URL, parameters);
  47. parameters.Add("oauth_signature", UrlEncode(signature));
  48. string response = HttpGet(REQUEST_TOKEN_URL, parameters);
  49. Dictionary dic = ParseResponse(response);
  50. RequestToken = dic["oauth_token"];
  51. RequestTokenSecret = dic["oauth_token_secret"];
  52. }
  53. public string GetAuthorizeUrl() {
  54. return AUTHORIZE_URL + "?oauth_token=" + RequestToken;
  55. }
  56. public void GetAccessToken(string pin) {
  57. SortedDictionary parameters = GenerateParameters(RequestToken);
  58. parameters.Add("oauth_verifier", pin);
  59. string signature = GenerateSignature(RequestTokenSecret, "GET", ACCESS_TOKEN_URL, parameters);
  60. parameters.Add("oauth_signature", UrlEncode(signature));
  61. string response = HttpGet(ACCESS_TOKEN_URL, parameters);
  62. Dictionary dic = ParseResponse(response);
  63. AccessToken = dic["oauth_token"];
  64. AccessTokenSecret = dic["oauth_token_secret"];
  65. UserId = dic["user_id"];
  66. ScreenName = dic["screen_name"];
  67. }
  68. public string Get(string url, IDictionary parameters) {
  69. SortedDictionary parameters2 = GenerateParameters(AccessToken);
  70. foreach (var p in parameters)
  71. parameters2.Add(p.Key, p.Value);
  72. string signature = GenerateSignature(AccessTokenSecret, "GET", url, parameters2);
  73. parameters2.Add("oauth_signature", UrlEncode(signature));
  74. return HttpGet(url, parameters2);
  75. }
  76. public string Post(string url, IDictionary parameters) {
  77. SortedDictionary parameters2 = GenerateParameters(AccessToken);
  78. foreach (var p in parameters)
  79. parameters2.Add(p.Key, p.Value);
  80. string signature = GenerateSignature(AccessTokenSecret, "POST", url, parameters2);
  81. parameters2.Add("oauth_signature", UrlEncode(signature));
  82. return HttpPost(url, parameters2);
  83. }
  84. private string HttpGet(string url, IDictionary parameters) {
  85. WebRequest req = WebRequest.Create(url + '?' + JoinParameters(parameters));
  86. WebResponse res = req.GetResponse();
  87. Stream stream = res.GetResponseStream();
  88. StreamReader reader = new StreamReader(stream);
  89. string result = reader.ReadToEnd();
  90. reader.Close();
  91. stream.Close();
  92. return result;
  93. }
  94. string HttpPost(string url, IDictionary parameters) {
  95. byte[] data = Encoding.ASCII.GetBytes(JoinParameters(parameters));
  96. WebRequest req = WebRequest.Create(url);
  97. req.Method = "POST";
  98. req.ContentType = "application/x-www-form-urlencoded";
  99. req.ContentLength = data.Length;
  100. Stream reqStream = req.GetRequestStream();
  101. reqStream.Write(data, 0, data.Length);
  102. reqStream.Close();
  103. WebResponse res = req.GetResponse();
  104. Stream resStream = res.GetResponseStream();
  105. StreamReader reader = new StreamReader(resStream, Encoding.UTF8);
  106. string result = reader.ReadToEnd();
  107. reader.Close();
  108. resStream.Close();
  109. return result;
  110. }
  111. private Dictionary ParseResponse(string response) {
  112. Dictionary result = new Dictionary();
  113. foreach (string s in response.Split('&')) {
  114. int index = s.IndexOf('=');
  115. if (index == -1)
  116. result.Add(s, "");
  117. else
  118. result.Add(s.Substring(0, index), s.Substring(index + 1));
  119. }
  120. return result;
  121. }
  122. private string JoinParameters(IDictionary parameters) {
  123. StringBuilder result = new StringBuilder();
  124. bool first = true;
  125. foreach (var parameter in parameters) {
  126. if (first)
  127. first = false;
  128. else
  129. result.Append('&');
  130. result.Append(parameter.Key);
  131. result.Append('=');
  132. result.Append(parameter.Value);
  133. }
  134. return result.ToString();
  135. }
  136. private string GenerateSignature(string tokenSecret, string httpMethod, string url, SortedDictionary parameters) {
  137. string signatureBase = GenerateSignatureBase(httpMethod, url, parameters);
  138. HMACSHA1 hmacsha1 = new HMACSHA1();
  139. hmacsha1.Key = Encoding.ASCII.GetBytes(UrlEncode(ConsumerSecret) + '&' + UrlEncode(tokenSecret));
  140. byte[] data = System.Text.Encoding.ASCII.GetBytes(signatureBase);
  141. byte[] hash = hmacsha1.ComputeHash(data);
  142. return Convert.ToBase64String(hash);
  143. }
  144. private string GenerateSignatureBase(string httpMethod, string url, SortedDictionary parameters) {
  145. StringBuilder result = new StringBuilder();
  146. result.Append(httpMethod);
  147. result.Append('&');
  148. result.Append(UrlEncode(url));
  149. result.Append('&');
  150. result.Append(UrlEncode(JoinParameters(parameters)));
  151. return result.ToString();
  152. }
  153. private SortedDictionary GenerateParameters(string token) {
  154. SortedDictionary result = new SortedDictionary();
  155. result.Add("oauth_consumer_key", ConsumerKey);
  156. result.Add("oauth_signature_method", "HMAC-SHA1");
  157. result.Add("oauth_timestamp", GenerateTimestamp());
  158. result.Add("oauth_nonce", GenerateNonce());
  159. result.Add("oauth_version", "1.0");
  160. if (!string.IsNullOrEmpty(token))
  161. result.Add("oauth_token", token);
  162. return result;
  163. }
  164. public string UrlEncode(string value) {
  165. string unreserved = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_.~";
  166. StringBuilder result = new StringBuilder();
  167. byte[] data = Encoding.UTF8.GetBytes(value);
  168. foreach (byte b in data) {
  169. if (b < 0x80 && unreserved.IndexOf((char)b) != -1)
  170. result.Append((char)b);
  171. else
  172. result.Append('%' + String.Format("{0:X2}", (int)b));
  173. }
  174. return result.ToString();
  175. }
  176. private string GenerateNonce() {
  177. string letters = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  178. StringBuilder result = new StringBuilder(8);
  179. for (int i = 0; i < 8; ++i)
  180. result.Append(letters[random.Next(letters.Length)]);
  181. return result.ToString();
  182. }
  183. private string GenerateTimestamp() {
  184. TimeSpan ts = DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0);
  185. return Convert.ToInt64(ts.TotalSeconds).ToString();
  186. }
  187. }
  188. class TweetEntity
  189. {
  190. public string UserName = "";
  191. public string Tweet = "";
  192. public string PostedTime = "";
  193. public override string ToString()
  194. {
  195. return UserName + "\r\n" + Tweet + "\r\n" + PostedTime + "\r\n";
  196. }
  197. }
  198. class Program
  199. {
  200. const string CONSUMER_KEY = "hogehogehogehogehoge";
  201. const string CONSUMER_SECRET = "fugafugafugafugafugafugafugafuga";
  202. static readonly string LOG_FILE = AppDomain.CurrentDomain.BaseDirectory.Substring(0,
  203. AppDomain.CurrentDomain.BaseDirectory.LastIndexOf("bin")) + "bin\\ConsoleTwitter.log";
  204. static void Main(string[] args)
  205. {
  206. int timespan = 60000;
  207. List prevTweetList = new List();
  208. List currTweetList = new List();
  209. for (; ; )
  210. {
  211. try
  212. {
  213. currTweetList = GetTweet();
  214. DisplayTweet(prevTweetList, currTweetList);
  215. prevTweetList = currTweetList;
  216. }
  217. catch (Exception ex)
  218. {
  219. Console.WriteLine(ex.Message);
  220. Console.WriteLine(ex.StackTrace);
  221. }
  222. System.Threading.Thread.Sleep(timespan);
  223. }
  224. }
  225. private static List GetTweet()
  226. {
  227. List tweetList = new List();
  228. Auth auth = GetAuth();
  229. // タイムラインから50件取得してみる
  230. Dictionary parameters = new Dictionary();
  231. parameters.Add("count", "50");
  232. string pagedata = auth.Get("http://twitter.com/statuses/home_timeline.xml", parameters);
  233. //Debug(pagedata);
  234. string tag = "status";
  235. XmlDocument xdoc = new XmlDocument();
  236. xdoc.LoadXml(pagedata);
  237. XmlElement root = xdoc.DocumentElement;
  238. XmlNodeList mylist = root.GetElementsByTagName(tag);
  239. for (int i = 0; i < mylist.Count; i++)
  240. {
  241. TweetEntity tweet = new TweetEntity();
  242. tweet.UserName = mylist[i].ChildNodes[11].ChildNodes[1].InnerText; // User name
  243. tweet.Tweet = mylist[i].ChildNodes[2].InnerText; // Tweet
  244. tweet.PostedTime = mylist[i].ChildNodes[0].InnerText; // posted time
  245. tweetList.Add(tweet);
  246. }
  247. return tweetList;
  248. }
  249. private static void DisplayTweet(List prevTweetList, List currTweetList)
  250. {
  251. StreamWriter wr = new StreamWriter(LOG_FILE, true, Encoding.GetEncoding("Shift_JIS"));
  252. foreach (TweetEntity currTweet in currTweetList)
  253. {
  254. bool isNew = true;
  255. foreach (TweetEntity prevTweet in prevTweetList)
  256. {
  257. if (prevTweet.ToString() == currTweet.ToString())
  258. {
  259. isNew = false;
  260. break;
  261. }
  262. }
  263. if (isNew)
  264. {
  265. Console.WriteLine(currTweet.ToString());
  266. wr.WriteLine(currTweet.ToString());
  267. }
  268. }
  269. wr.Close();
  270. }
  271. private static Auth GetAuth()
  272. {
  273. Auth auth;
  274. var settings = ConsoleTwitter.Properties.Settings.Default;
  275. if (string.IsNullOrEmpty((string)settings["AccessToken"]))
  276. {
  277. auth = new Auth(CONSUMER_KEY, CONSUMER_SECRET);
  278. // リクエストトークンを取得する
  279. auth.GetRequestToken();
  280. // ユーザーにRequestTokenを認証してもらう
  281. Console.WriteLine("次のURLにアクセスして暗証番号を取得してください:");
  282. Console.WriteLine(auth.GetAuthorizeUrl());
  283. Console.Write("暗証番号:");
  284. string pin = Console.ReadLine().Trim();
  285. // アクセストークンを取得する
  286. auth.GetAccessToken(pin);
  287. // 結果を表示する
  288. Console.WriteLine("AccessToken: " + auth.AccessToken);
  289. Console.WriteLine("AccessTokenSecret: " + auth.AccessTokenSecret);
  290. Console.WriteLine("UserId: " + auth.UserId);
  291. Console.WriteLine("ScreenName: " + auth.ScreenName);
  292. // アクセストークンを設定ファイルに保存する
  293. settings["AccessToken"] = auth.AccessToken;
  294. settings["AccessTokenSecret"] = auth.AccessTokenSecret;
  295. settings["UserId"] = auth.UserId;
  296. settings["ScreenName"] = auth.ScreenName;
  297. settings.Save();
  298. }
  299. else
  300. {
  301. // 設定ファイルから読み込む
  302. auth = new Auth(CONSUMER_KEY, CONSUMER_SECRET,
  303. (string)settings["AccessToken"], (string)settings["AccessTokenSecret"],
  304. (string)settings["UserId"], (string)settings["ScreenName"]);
  305. }
  306. return auth;
  307. }
  308. private static void Debug(string s)
  309. {
  310. StreamWriter wr = new StreamWriter(LOG_FILE + ".DEBUG.txt", true, Encoding.GetEncoding("Shift_JIS"));
  311. wr.WriteLine(s);
  312. wr.Close();
  313. }
  314. }
  315. }

2012年5月30日水曜日

Windows Live Writerを使ってみる

ブログを書くのに神的によいそうなので、使ってみる。
すべてのプログラム-Windows Live-Windows Writerを選択する。
使用許諾はOKする。
Windows Live Writerの構成では「次へ」を選択する。
20120530 Windows Live Writer 1
20120530 Windows Live Writer 1 posted by (C)てる

ブログサービスの選択では「他のサービス」を選択する。
20120530 Windows Live Writer 2
20120530 Windows Live Writer 2 posted by (C)てる

ブログアカウントの追加で、ブログ情報を入力する。
20120530 Windows Live Writer 3
20120530 Windows Live Writer 3 posted by (C)てる

ここがすごいところなのだが、上記で「次へ」とした後に、ブログに接続して、情報を取得してきてくれる。ブログの選択はそのまま「次へ」。
20120530 Windows Live Writer 4
20120530 Windows Live Writer 4 posted by (C)てる

ここも「次へ」でよい。
20120530 Windows Live Writer 5
20120530 Windows Live Writer 5 posted by (C)てる

以下がブログ作成画面で、タイトルと本文を少し入力したところ。
20120530 Windows Live Writer 6
20120530 Windows Live Writer 6 posted by (C)てる

機能的には、

  • 複数のブログに対応している
  • エディタ上での編集、プレビュー(テンプレートも含んでのプレビュー)、HTMLの直接編集が可能
  • エディタ上での編集は、フォントスタイル、段落、リンク等の挿入、表の挿入などができる
  • 新規の投稿、投稿されたものの修正、削除などができる
など、なかなかよさそうである。

2012年4月3日火曜日

RLO対策

以下のサイトに従って、「secpol.msc」を実行して、ソフトウェア制限のポリシーを設定する。
ファイル名は「左から右に読む」とは限らない?! - @IT
なお、設定後は、「gpupdate.exe」を実行して更新を反映後に、再ログインをする必要がある。

2012年3月22日木曜日

Excelのクリップボードがおかしくなる


Firefoxアドオンの「Skype Click to Call 5.10.0.9560」を有効にしていると、
Excelでコピー時にテキスト情報のみしかコピーされないようになる。
(書式情報がコピーされなくなる)

「Skype Click to Call 5.10.0.9560」を無効にすることで、この現象が解決できる。

組み合わせは Windows 7 + Office 2010 + FireFox 11 で発生した。

※Internet Explorer 、 Google Chrome 用の「Skype Click to Call」アドオンでも同様の現象が発生した。