【Excel】横の表定義からINSERT文作成

横向けになっている表定義からINSERT文を作成します。
前提条件はコメントに記載の通り。
※DATE型に対するTO_DATEは手動。

Sub SqlInsertCreate()
'
' SqlInsertCreate Macro
'
' テーブル名、項目名、値からINSERT文を自動生成します。
'
' A1:テーブル名
' A2~C2:項目名
' A3~C3:値
' 上記状態の時にE2を選択し、マクロを実行する。

    '//=====項目用INSERT文作成===============================
    'セルをアクティブにする。
    ActiveCell.Offset(0, 0).Select

    'アクティブセル列数を取得する。
    Dim activCellNoCol As Long
    activCellNoCol = ActiveCell.Column

    'ループで利用する変数を宣言する。
    Dim i As Long
    Dim tmpStrCol As String

    'アクティブセル列数-2から列数が2になるまで繰り返す
    For i = activCellNoCol - 2 To 2 Step -1
        tmpStrCol = tmpStrCol & "RC[-" & i & "]&"",""&"
    Next i

    '項目用INSERT文を作成する。
    Dim tmpInsertCol As String
    tmpInsertCol = "=" & Left(tmpStrCol, Len(tmpStrCol) - 5) & "&"") VALUES ("""
    ActiveCell.Formula = tmpInsertCol



    '//=====値用INSERT文作成===============================
    'セルをアクティブにする。
    ActiveCell.Offset(1, 0).Select

    'アクティブセル列数を取得する。
    Dim activCellNoVal As Long
    activCellNoVal = ActiveCell.Column

    'ループで利用する変数を宣言する。
    Dim j As Long
    Dim tmpStrVal As String

    'アクティブセル列数-2から列数が2になるまで繰り返す
    For j = activCellNoVal - 2 To 2 Step -1
        tmpStrVal = tmpStrVal & "RC[-" & j & "]&""','""&"
    Next j

    '値用INSERT文を作成する。
    Dim tmpInsertVal As String
    tmpInsertVal = "=""'""&" & Left(tmpStrVal, Len(tmpStrVal) - 7) & "&""');"""
    ActiveCell.Formula = tmpInsertVal


    '//=====テーブル用INSERT文作成===============================
    'セルをアクティブにする。
    ActiveCell.Offset(-2, 0).Select

    'アクティブセル列数を取得する。
    Dim activCellNoTbl As Long
    activCellNoTbl = ActiveCell.Column

    'テーブル名の位置を求める。
    Dim tblNameNo As String
    tblNameNo = activCellNoTbl - 2

    Dim tmpInsertTbl As String
    tmpInsertTbl = "=""INSERT INTO ""&RC[-" & tblNameNo & "]&"" ("""
    ActiveCell.Formula = tmpInsertTbl


    'セルを初期値に戻す。
    ActiveCell.Offset(1, 0).Select
End Sub

【Excel】縦の表定義から横の表定義に変換

現場では表定義は縦形式が多いですが、
エビデンスや他作業時は横形式の方が良い場合もあるので、マクロで変換します。

Sub TblDefConversion()
'
' TblDefConversion Macro
'
' テーブル定義を縦から横に変換する。
    
    ' 変換後のエクセル用変数を宣言する。
    Dim newBook As String
    Dim NewWorkSheet As Worksheet

    ' 変換後のエクセルを追加する。
    Workbooks.Add
    newBook = ActiveWorkbook.Name

    ' 変換前のエクセル用変数を宣言する。
    Dim oldWorkBook As Workbook
    Set oldWorkBook = Workbooks("Book1.xlsx")
    Dim oldWorkSheet As Worksheet
    Dim tblNameRowPoint As Long   ' テーブル名が存在する行数
    Dim tblNameCellPoint As Long  ' テーブル名が存在する列数
    tblNameRowPoint = 4
    tblNameCellPoint = 21
    Dim colNameLogicalRowPoint As Long   ' 項目名(論理名)が存在する行数
    Dim colNameLogicalCellPoint As Long  ' 項目名(論理名)が存在する列数
    Dim colNamePhysicalRowPoint As Long  ' 項目名(物理名)が存在する行数
    Dim colNamePhysicalCellPoint As Long ' 項目名(物理名)が存在する列数
    Dim colTypeRowPoint As Long          ' 型が存在する行数
    Dim colTypeCellPoint As Long         ' 型が存在する列数
    Dim colSizeRowPoint As Long          ' サイズが存在する行数
    Dim colSizeCellPoint As Long         ' サイズが存在する列数
    colNameLogicalRowPoint = 9
    colNameLogicalCellPoint = 2
    colNamePhysicalRowPoint = colNameLogicalRowPoint
    colNamePhysicalCellPoint = 13
    colTypeRowPoint = colNameLogicalRowPoint
    colTypeCellPoint = 24
    colSizeRowPoint = colNameLogicalRowPoint
    colSizeCellPoint = 28

    ' 変換前のエクセルシート分繰り返す。
    For Each oldWorkSheet In oldWorkBook.Worksheets

        ' 変換後のエクセルシートとA1セルに変換前のエクセルシート名を設定する。
        ' エクセルでは関数に()をつけると戻り値を伴う。
        Workbooks(newBook).Activate
        Set NewWorkSheet = Worksheets.Add()
        MsgBox oldWorkSheet.Cells(tblNameRowPoint, tblNameCellPoint).Value
        NewWorkSheet.Name = oldWorkSheet.Cells(tblNameRowPoint, tblNameCellPoint).Value
        Cells(1, 1).Value = oldWorkSheet.Cells(tblNameRowPoint, tblNameCellPoint).Value
        
        ' 変換前エクセルの項目名の最初と最後を設定する。
        oldWorkBook.Activate
        oldWorkSheet.Activate
        Dim i As Long
        Dim MaxRow As Long
        MaxRow = oldWorkSheet.Cells(Rows.Count, colNameLogicalCellPoint).End(xlUp).Row
        
        ' 変換前エクセルの項目名の最初から最後までを繰り返す。
        For i = colNameLogicalRowPoint To MaxRow

            ' 項目名(論理名)、項目名(物理名)、型、サイズを取得する。
            Dim colNameLogical As String
            Dim colNamePhysical As String
            Dim colType As String
            Dim colSize As String
            colNameLogical = Cells(i, colNameLogicalCellPoint).Value
            colNamePhysical = Cells(i, colNamePhysicalCellPoint).Value
            colType = Cells(i, colTypeCellPoint).Value
            colSize = Cells(i, colSizeCellPoint).Value

            ' 変換後のエクセルに各々の値を取得する。
            Workbooks(newBook).Activate
            NewWorkSheet.Cells(2, i - colNameLogicalRowPoint + 1).Value = colNameLogical
            NewWorkSheet.Cells(3, i - colNamePhysicalRowPoint + 1).Value = colNamePhysical
            NewWorkSheet.Cells(4, i - colTypeRowPoint + 1).Value = colType
            NewWorkSheet.Cells(5, i - colSizeRowPoint + 1).Value = colSize

            ' 変換前のエクセルをアクティブにしておく。
            oldWorkBook.Activate

        Next i
    Next oldWorkSheet

End Sub

【Excel】印刷初期設定

納品用に横印刷、左上にファイル名、右上に日付、
真ん中下にページ数/総ページ数を設定します。
※縦印刷は未記載。

Sub PageSetting()
'
' PageSetting Macro
'
' シートの初期設定を行います。
'
    ' 印刷初期設定を行う。
    ' 印刷高速設定をfalseにして設定する。
    Application.PrintCommunication = False
    With ActiveSheet.PageSetup
        .CenterHorizontally = False
        .CenterVertically = False
        .Orientation = xlLandscape
        .Draft = False
        .PaperSize = xlPaperA4
        .FirstPageNumber = xlAutomatic
        .Order = xlDownThenOver
        .BlackAndWhite = False
        .Zoom = False
        .FitToPagesWide = 1
        .FitToPagesTall = False
        .PrintErrors = xlPrintErrorsDisplayed
        .OddAndEvenPagesHeaderFooter = False
        .DifferentFirstPageHeaderFooter = False
        .ScaleWithDocHeaderFooter = True
        .AlignMarginsHeaderFooter = True
    End With
    ' 印刷高速設定をtureに戻す。
    Application.PrintCommunication = True

    ' 印刷高速設定を設定せずに設定する。(印刷高速設定を設定すると設定不可)
    With ActiveSheet.PageSetup
        .LeftHeader = "&F"
        .CenterHeader = ""
        .RightHeader = "&D"
        .LeftFooter = ""
        .CenterFooter = "&P/&N"
        .RightFooter = ""
        .LeftMargin = Application.InchesToPoints(0.708661417322835)
        .RightMargin = Application.InchesToPoints(0.708661417322835)
        .TopMargin = Application.InchesToPoints(0.748031496062992)
        .BottomMargin = Application.InchesToPoints(0.748031496062992)
        .HeaderMargin = Application.InchesToPoints(0.31496062992126)
        .FooterMargin = Application.InchesToPoints(0.31496062992126)
        .PrintHeadings = False
        .PrintGridlines = False
        .PrintComments = xlPrintNoComments
        .PrintQuality = 600
        .EvenPage.LeftHeader.Text = ""
        .EvenPage.CenterHeader.Text = ""
        .EvenPage.RightHeader.Text = ""
        .EvenPage.LeftFooter.Text = ""
        .EvenPage.CenterFooter.Text = ""
        .EvenPage.RightFooter.Text = ""
        .FirstPage.LeftHeader.Text = ""
        .FirstPage.CenterHeader.Text = ""
        .FirstPage.RightHeader.Text = ""
        .FirstPage.LeftFooter.Text = ""
        .FirstPage.CenterFooter.Text = ""
        .FirstPage.RightFooter.Text = ""
    End With

    ' 文字初期設定を行う。
    Cells.Select
    With Selection.Font
        .Name = "MS Pゴシック"
        .Size = 9
        .Strikethrough = False
        .Superscript = False
        .Subscript = False
        .OutlineFont = False
        .Shadow = False
        .Underline = xlUnderlineStyleNone
        .ThemeColor = xlThemeColorLight1
        .TintAndShade = 0
        .ThemeFont = xlThemeFontNone
    End With
    Selection.NumberFormatLocal = "@"
    Range("A1").Select

End Sub

【Excel】赤い太枠自動作成

設計書の記述内容を強調したり、
エビデンスの確認箇所を強調したりする際に
赤い太枠をよく使うのでマクロで作成します。

Sub RedBox()
'
' RedBox Macro
'
' 赤い太枠四角線を作ります。

    ActiveSheet.Shapes.AddShape(msoShapeRectangle, Selection.Left, Selection.Top, 96.75, 96.75).Select

    Selection.ShapeRange.Line.Weight = 2.25
    Selection.ShapeRange.Line.Visible = msoTrue
    Selection.ShapeRange.Line.Style = msoLineSingle
    Selection.ShapeRange.Line.ForeColor.SchemeColor = 10
    Selection.ShapeRange.Line.Visible = msoTrue
    Selection.ShapeRange.Fill.Visible = msoFlase

End Sub

【Excel】A1セル選択

納品物はA1セルにカーソルが選択されていないといけないことが多いので、
マクロでA1を選択します。

Sub TopCursor()
'
' TopCursor Macro
'
' A1にカーソルを合わせます。
'
    ' 画面のチラつきの防止設定を行う。
    Application.ScreenUpdating = False

    ' 全シート分繰り返す。
    Dim Sht As Worksheet
    For Each Sht In Worksheets
        ' シートを選択する。
        Sht.Select

        ' スクロールバーを上にする。
        Dim i As Integer
        For i = 1 To Windows(1).Panes.Count
            Windows(1).Panes(i).ScrollColumn = 1
            Windows(1).Panes(i).scrollRow = 1
        Next i

        ' A1セルを選択する。
        ActiveSheet.Cells(1, 1).Select
    Next Sht

    ' 先頭シートを選択する。
    Worksheets(1).Activate

    ' 画面のチラつきの防止設定を元に戻す。
    Application.ScreenUpdating = True
End Sub

SJISとUTF-8を一緒にgrep

JavaファイルだとSJIS、propertiesだとSJISまたはUTF-8の
どちらでも定義されている”氏名”という単語を
ファイル名と行数を出力しながらgrepしたい。

検索してみると下記のような方法が多かったです。

grep -rn `echo "氏名" | nkf -s` *.java *.properties | nkf -w

上記方法だと、入力値の”氏名”をSJISに変換してからgrepしているため、
SJISで書かれているJavaファイルしかgrepできないんじゃ?と思い、
for文を利用してみることに。
(そもそも上記方法だとなぜかgrepすら出来ず。なぜだろう。。)

for X in `find ( -name "*.java" -or -name "*.properties" )`; do nkf -w $X | grep -rn "氏名" | while read line; do echo "$X:$line"; done done
./Sample.java:15: //氏名を変換
./test_utf8.properties:1:#氏名を定義
./test_sjis.properties:1:#氏名を定義

これでファイル名と行数とそれに対するgrep結果が得られました。
findでHITしたファイルを一時変数に格納し、「nkf -w」で全てUTF-8に変換してからgrepしてます。
次にgrep結果をwhileで繰り返しながら一時変数(ファイル)とgrep結果を出力。

もうシェル作って引数取ればいいじゃないと思いますが、1行でやりたかったんです。

【Excel】個人用マクロブック

エクセルでいつも使うマクロを個人用マクロブックに保存しました。
保存場所は下記の通り。

C:ユーザー(アカウント名)AppDateRoamingMicrosoftExcelXLSTARTPERSONAL.XLSB

マクロを記録する際に「マクロの保存先」で「個人用マクロブック」を選択すると保存可能。
保存すると新規エクセル立ち上げる度に「PERSONAL.XLSB」が表示されていて邪魔なので、
エクセルの「表示」-「ウィンドウ」-「表示しない」を選択し、非表示にします。

【Eclipse】リモートデバッグ

ローカル環境でWebアプリケーションサーバを入れ、
その上でデバッグで動かしながら開発することはよくありますが、
他環境で動作しているWebアプリケーションに対してデバッグする方法。

前提条件として、デバッグ可能な状態でWebアプリケーションを起動していること。
Raspberry PiではTomcatが起動しているので、Tomcatを例にします。

  • デバッグモードでTomcat起動
    「$TOMCAT_HOME/bin/catalina.sh」を確認すると下記ようになっています。

    if [ "$1" = "jpda" ] ; then
      if [ -z "$JPDA_TRANSPORT" ]; then
        JPDA_TRANSPORT="dt_socket"
      fi
      if [ -z "$JPDA_ADDRESS" ]; then
        JPDA_ADDRESS="8000"
      fi
      if [ -z "$JPDA_SUSPEND" ]; then
        JPDA_SUSPEND="n"
      fi
      if [ -z "$JPDA_OPTS" ]; then
        JPDA_OPTS="-agentlib:jdwp=transport=$JPDA_TRANSPORT,address=$JPDA_ADDRESS, server=y,suspend=$JPDA_SUSPEND"
      fi
      CATALINA_OPTS="$CATALINA_OPTS $JPDA_OPTS"
      shift
    fi
    

    引数に「jpda」を与えるとデバッグ可能な状態で起動できます。
    自作シェルではそんな引数を受け取れるよう作成していないので、
    「catalina.sh」を直接呼び出します。

    /etc/init.d/tomcat.sh stop
    /tomcat/apache-tomcat-7.0.55/bin/catalina.sh jpda start
    
  • Eclipseの設定

    Eclipseから「実行」-「デバッグの構成」を選択。
    リモートJavaアプリケーションで右クリックして新規を選択。
    接続プロパティーのホスト:z-area.net または Raspberry PiのIP
    接続プロパティーのホスト:8000
    他はデフォルトのまま「適用」ボタン押下。
    「デバッグ」ボタン押下。
    

    ローカルのソースにデバッグポイントを置いて
    Webアプリケーションに接続し、デバッグポイントを通るような操作を行うと
    他環境のWebアプリケーションに対してデバッグできます。

    Eclipseから「デバッグ」ボタン押下してタイムアウトになる場合、
    サーバ側のFWでポート8000を許可する必要があります。

【Raspbian】ShellShock対策

bashに脆弱性が発見されたらしく、Rapspberry Piも対策してみました。

x='() { :;}; echo vulnerable' bash -c 'echo this is a test'

上記コマンド実行した際に「vulnerable」がコンソールに出ると要対策。

vulnerable
this is a test

というわけで最新版にアップデートします。

apt-get update
apt-get --only-upgrade install bash

再度コマンド実行して対策できたか確認します。

x='() { :;}; echo vulnerable' bash -c 'echo this is a test'
this is a test

「vulnerable」が表示されていなければ問題なし。
アップデートしただけで対策完了。

【サクラエディタ】正規表現

正規表現でメモ。

  • 1つ目の/までを検索

    ^[^/]*/
    

    先頭から/を含まない0文字以上の文字列を持ち、/で終わる文字列を検索。

  • /を含まない1行を検索

    ^[^/]*$
    

    先頭から/を含まない0文字以上の文字列を持つ文末までの1行を検索。

  • HTMLタグのtdタグのみを検索

    <td[^>]*>|<[^>]td>
    

    <tdで始まる>を含まない0文字以上の文字列を持ち、>で終わる文字列を検索。
    または
    <で始まる>を含まない0文字以上の文字列を持ち、td>で終わる文字列を検索。