[.net core 3.0] Excelファイル書き出し

Microsoft.Office.Interop.Excelはcoreに対応していないようなので、有名なEPPlusをNuGetでゲット。一個前のエントリにあるソースコードを流用して、MySqlのデータをExcelファイルに書き出す。例外処理してません。

using System;
using System.IO;
using System.Data;
using MySql.Data.MySqlClient;
using OfficeOpenXml;

namespace CliTool {
    class Program {
        static void Main(string[] args) {
            CreateExcel(GetTrySailMember());
            /* end */
            Console.WriteLine("Press Any key to continue...");
            Console.ReadKey(true);
        }

        private static void CreateExcel(DataTable table) {
            using var saveExcel = new ExcelPackage(new FileInfo("trysail.xlsx"));
            var sheet = saveExcel.Workbook.Worksheets.Add("trysail");
            // header
            var row = 1;
            var col = 1;
            foreach (var column in table.Columns) {
                sheet.Cells[row, col].Value = column.ToString();
                col++;
            }
            // data
            foreach (var records in table.Select()) {
                ++row;
                for(var i = 0; i < records.ItemArray.Length; i++) {
                    sheet.Cells[row, (i + 1)].Value = records[i].ToString() ?? string.Empty;
                }
            }
            saveExcel.Save();
        }

        private static DataTable GetTrySailMember() {
            var cString = "Host=localhost;Database=benchmark;CharSet=utf8mb4;User Id=root;Password=";
            var query = "select * from member";
            var datatable = new DataTable();
            using var con = new MySqlConnection(cString);
            using var adapter = new MySqlDataAdapter {
                SelectCommand = new MySqlCommand(query, con)
            };
            adapter.Fill(datatable);
            return datatable;
        }
    }
}

DataRowはいつものfor文で回す方法に変更した。

[.net core 3.0] MySqlへのアクセス (生SQL)

このWordPressブログなんだけど、フロント側だけ.net coreで実装したい。なので、MySqlにアクセスしてデータ取ってくるところまでを書く。EFではなく生SQL使いたいのでそんな人が入れば。

NugetでMySql.Data 8.0.18を追加。Oracleさん公式なので安心。ほんとかよ。

ex) https://dev.mysql.com/doc/dev/connector-net/6.10/html/N_MySql_Data_MySqlClient.htm

こんなテーブルを全件取得。

MariaDB [benchmark]> select * from member;
+-----------+-------------+--------------------+------------+---------------------+---------------------+
| member_id | member_name | email              | birthday   | created_at          | updated_at          |
+-----------+-------------+--------------------+------------+---------------------+---------------------+
|         1 | 雨宮天      | hoge@dobusarai.net | 1993-08-28 | 2019-09-30 15:16:52 | 2019-09-30 15:19:39 |
|         2 | 麻倉もも    | fuga@dobusarai.net | 1994-06-25 | 2019-09-30 15:16:52 | 2019-09-30 15:16:52 |
|         3 | 夏川椎菜    | piyo@dobusarai.net | 1996-07-18 | 2019-09-30 15:16:52 | 2019-09-30 15:16:52 |
+-----------+-------------+--------------------+------------+---------------------+---------------------+
3 rows in set (0.36 sec)
using System;
using System.Data;
using MySql.Data.MySqlClient;

namespace CliTool {
    class Program {
        static void Main(string[] args) {
            var cString     = "Host=localhost;Database=benchmark;CharSet=utf8mb4;User Id=root;Password=";
            var query       = "select * from member";
            var datatable   = new DataTable();
            using(var con = new MySqlConnection(cString)) {
                using var adapter = new MySqlDataAdapter {
                    SelectCommand = new MySqlCommand(query, con)
                };
                adapter.Fill(datatable);
            }
            foreach(var column in datatable.Columns) {
                Console.Write(column.ToString() + " ");
            }
            Console.WriteLine(string.Empty);
            foreach(var row in datatable.Select()) {
                Console.WriteLine(string.Format("{0} {1} {2} {3} {4} {5}", row[0], row[1], row[2], row[3], row[4], row[5]));
            }

            /* end */
            Console.WriteLine("Press Any key to continue...");
            Console.ReadKey(true);
        }
    }
}
member_id member_name email birthday created_at updated_at
1 雨宮天 hoge@dobusarai.net 1993/08/28 0:00:00 2019/09/30 15:16:52 2019/09/30 15:19:39
2 麻倉もも fuga@dobusarai.net 1994/06/25 0:00:00 2019/09/30 15:16:52 2019/09/30 15:16:52
3 夏川椎菜 piyo@dobusarai.net 1996/07/18 0:00:00 2019/09/30 15:16:52 2019/09/30 15:16:52
Press Any key to continue...

PHPに慣れちゃったせいでDataTable使いづらく感じるが、sqlに頼らずにクエリ使えるので慣れれば便利なはず。というかサンプル探しててもDataRowをforeachで回さずにindex指定してるのが多くてなんでや?とデバッグしつつ色々触ってみたのだが、enumerableじゃないんだね。

[.net core 3.0] Linux用のCliツール作成(単一ファイル)

新しいプロジェクトの作成

コンソールアプリ(.NET Core) -> プロジェクト名(ここでは適当にTestTool)

code

using System;

namespace TestTool {
    class Program {
        static void Main(string[] args) {
            Console.WriteLine("Hello World!");
            foreach (var arg in args) {
                Console.WriteLine(arg);
            }
            /* end */
            Console.WriteLine("Press Any key to continue...");
            Console.ReadKey(true);
        }
    }
}

発行のプロファイルを作る

ビルド -> TestToolの発行 -> 発行先はフォルダ -> プロファイルの作成

配置モードを自己完結, ターゲットランタイムをlinux-x64に変更して保存。

単一ファイルに変更するためにprofileファイルを編集。ソリューションエクスプローラからProperties>PublishProfiles>FolderProfile.pubxmlを選択して開く。

<?xml version="1.0" encoding="utf-8"?>
<!--
https://go.microsoft.com/fwlink/?LinkID=208121. 
-->
<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
  <PropertyGroup>
    <PublishProtocol>FileSystem</PublishProtocol>
    <Configuration>Release</Configuration>
    <Platform>Any CPU</Platform>
    <TargetFramework>netcoreapp3.0</TargetFramework>
    <PublishDir>bin\publish\</PublishDir>
    <RuntimeIdentifier>linux-x64</RuntimeIdentifier>
    <SelfContained>true</SelfContained>
    <PublishSingleFile>true</PublishSingleFile> <!--単一ファイル出力指定-->
  </PropertyGroup>
</Project>

PublishSingleFileを追加して保存。

発行

ビルド -> TestToolの発行 -> 発行

実行

単一ファイルと言いつつpdbファイル(Releaseビルドなのに)も出力される。作成されたlinuxバイナリ(TestTool)(76MB)をこのサーバにアップロードして実行。

[dobu@133-130-123-5 ~]$ chmod +x TestTool
[dobu@133-130-123-5 ~]$ ./TestTool hoge fuga piyo
Hello World!
hoge
fuga
piyo
Press Any key to continue...
[dobu@133-130-123-5 ~]$

明るくないsh書くよりこっちのほうが楽だなー。個人では複雑怪奇なことをlinuxでやることはないのだけど、debuggerが強力なので。単一ファイルにすることで各バージョン気にしなくてすむのが最高。ファイルサイズの肥大化は……このご時世ある程度までなら増えても別にいいだろって。

ex) https://docs.microsoft.com/ja-jp/dotnet/core/whats-new/dotnet-core-3-0#single-file-executables

[.net framework] windows.formsの2重起動禁止

表題の件だけど、こんなサンプルコードが多く見られる。

上記は動作はするけどダメな例だ。Mutexはdisposableなのだし、Local変数にするとGCが葬ってしまう可能性がある。こう書くの。

ネットに転がってるサンプルを鵜呑みにして実装しちゃだめだなーって。誰も見ないんだけど自戒の念を込めて。

[.net framework] chromium web browser

久々にWindows.Formsです。本当は.net coreで作りたかったのだけど、WebBrowserコントロールがIE7なのな!さすがにEdgeだとは思っていないがIE11かと……。ということで埋め込みchromiumコントロールを使うことにした。これが.net core3.0ではどうにもこうにもエラーになるので、今回は.net framework 4.7.2で。

https://gist.github.com/dobusarai2016/9e3d179d8f5ac7fa46468b468d6e8616

マニフェスト追加して高DPI対応とかにもしてるけど、基本はこれだけ。ボタンとかはさすがに面倒なのでフォームデザイナでペタペタ。CefSharpはデザイナで貼るとバージョンが古いだのビルド時にようわからんエラー吐いたりだの鬱陶しいので直書き。

Application -> Chromiumは基本的にJavascriptを走らせることでDOM操作とか何でも出来るみたいだが、Chromium -> Application、例えばDOMの取得などはレンダリング時にしかできないっぽい。まー目的は片方向で良いので調べるのはやめた。生スレッド使うのはダセえとかは言うな。

※20191017 生スレッドの部分を修正

※20191017 LoadError(httpdが存在しないページで404を返すのではなく、サーバ自体に接続できない、存在しない場合は真っ白で終わり)時に、エラーhtmlを読み込んで描画するように変更

※20191017 DOM要素へのアクセスも試してみた。が、idやインデックスを指定しての取得は出来るが、複数の要素(getElementsByClassNameでまとめて取得)はdebuggerで見てもnullになってしまって取れないみたい。

        private async void ToolStripButton2_Click(object sender, EventArgs e)
        {
            if (!chromium.IsLoading)
            {
                //var js = "document.getElementsByClassName('chromium-test');"; // NG
                //var js = "document.getElementsByClassName('chromium-test')[0];"; // NG
                var js = "document.getElementsByClassName('chromium-test')[0].innerText;"; // OK
                //var js = "document.getElementById('user_id').value;"; // OK
                var test = "";
                await chromium.GetBrowser().MainFrame.EvaluateScriptAsync(js).ContinueWith(
                    x => {
                        var response = x.Result;
                        if(response.Success && response.Result != null)
                        {
                            test = response.Result.ToString();
                        }
                    }
                );
                StatusMessageLabel.Text = test;
            }
        }

https://www.dobusarai.net/env/