Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(csharp/ExcelAddIn): First version of the Excel AddIn #6004

Merged
merged 2 commits into from
Aug 30, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
67 changes: 67 additions & 0 deletions csharp/ExcelAddIn/DeephavenExcelFunctions.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using Deephaven.ExcelAddIn.ExcelDna;
using Deephaven.ExcelAddIn.Factories;
using Deephaven.ExcelAddIn.Models;
using Deephaven.ExcelAddIn.Operations;
using Deephaven.ExcelAddIn.Providers;
using Deephaven.ExcelAddIn.Viewmodels;
using Deephaven.ExcelAddIn.Views;
using ExcelDna.Integration;

namespace Deephaven.ExcelAddIn;

public static class DeephavenExcelFunctions {
private static readonly StateManager StateManager = new();

[ExcelCommand(MenuName = "Deephaven", MenuText = "Connections")]
public static void ShowConnectionsDialog() {
ConnectionManagerDialogFactory.CreateAndShow(StateManager);
}

[ExcelFunction(Description = "Snapshots a table", IsThreadSafe = true)]
public static object DEEPHAVEN_SNAPSHOT(string tableDescriptor, object filter, object wantHeaders) {
if (!TryInterpretCommonArgs(tableDescriptor, filter, wantHeaders, out var td, out var filt, out var wh, out var errorText)) {
return errorText;
}

// These two are used by ExcelDNA to share results for identical invocations. The functionName is arbitary but unique.
const string functionName = "Deephaven.ExcelAddIn.DeephavenExcelFunctions.DEEPHAVEN_SNAPSHOT";
var parms = new[] { tableDescriptor, filter, wantHeaders };
ExcelObservableSource eos = () => new SnapshotOperation(td!, filt, wh, StateManager);
return ExcelAsyncUtil.Observe(functionName, parms, eos);
}

[ExcelFunction(Description = "Subscribes to a table", IsThreadSafe = true)]
public static object DEEPHAVEN_SUBSCRIBE(string tableDescriptor, object filter, object wantHeaders) {
if (!TryInterpretCommonArgs(tableDescriptor, filter, wantHeaders, out var td, out var filt, out var wh, out string errorText)) {
return errorText;
}
// These two are used by ExcelDNA to share results for identical invocations. The functionName is arbitary but unique.
const string functionName = "Deephaven.ExcelAddIn.DeephavenExcelFunctions.DEEPHAVEN_SUBSCRIBE";
var parms = new[] { tableDescriptor, filter, wantHeaders };
ExcelObservableSource eos = () => new SubscribeOperation(td, filt, wh, StateManager);
return ExcelAsyncUtil.Observe(functionName, parms, eos);
}

private static bool TryInterpretCommonArgs(string tableDescriptor, object filter, object wantHeaders,
[NotNullWhen(true)]out TableTriple? tableDescriptorResult, out string filterResult, out bool wantHeadersResult, out string errorText) {
filterResult = "";
wantHeadersResult = false;
if (!TableTriple.TryParse(tableDescriptor, out tableDescriptorResult, out errorText)) {
return false;
}

if (!ExcelDnaHelpers.TryInterpretAs(filter, "", out filterResult)) {
errorText = "Can't interpret FILTER argument";
return false;
}


if (!ExcelDnaHelpers.TryInterpretAs(wantHeaders, false, out wantHeadersResult)) {
errorText = "Can't interpret WANT_HEADERS argument";
return false;
}
return true;
}
}
24 changes: 24 additions & 0 deletions csharp/ExcelAddIn/ExcelAddIn.csproj
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<TargetFramework>net8.0-windows</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<UseWindowsForms>true</UseWindowsForms>
<AllowUnsafeBlocks>True</AllowUnsafeBlocks>
</PropertyGroup>

<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|AnyCPU'">
<TreatWarningsAsErrors>True</TreatWarningsAsErrors>
</PropertyGroup>

<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'">
<TreatWarningsAsErrors>True</TreatWarningsAsErrors>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="ExcelDna.Addin" Version="*-*" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\client\DeephavenClient\DeephavenClient.csproj" />
</ItemGroup>
</Project>
12 changes: 12 additions & 0 deletions csharp/ExcelAddIn/ExcelAddIn.csproj.user
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="Current" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup />
<ItemGroup>
<Compile Update="views\ConnectionManagerDialog.cs">
<SubType>Form</SubType>
</Compile>
<Compile Update="views\CredentialsDialog.cs">
<SubType>Form</SubType>
</Compile>
</ItemGroup>
</Project>
31 changes: 31 additions & 0 deletions csharp/ExcelAddIn/ExcelAddIn.sln
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.7.34221.43
MinimumVisualStudioVersion = 10.0.40219.1
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "ExcelAddIn", "ExcelAddIn.csproj", "{08852A0D-DB96-404E-B3CE-BF30F2AD3F74}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "DeephavenClient", "..\client\DeephavenClient\DeephavenClient.csproj", "{6848407D-1CEF-4433-92F4-6047AE3D2C52}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{08852A0D-DB96-404E-B3CE-BF30F2AD3F74}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{08852A0D-DB96-404E-B3CE-BF30F2AD3F74}.Debug|Any CPU.Build.0 = Debug|Any CPU
{08852A0D-DB96-404E-B3CE-BF30F2AD3F74}.Release|Any CPU.ActiveCfg = Release|Any CPU
{08852A0D-DB96-404E-B3CE-BF30F2AD3F74}.Release|Any CPU.Build.0 = Release|Any CPU
{6848407D-1CEF-4433-92F4-6047AE3D2C52}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{6848407D-1CEF-4433-92F4-6047AE3D2C52}.Debug|Any CPU.Build.0 = Debug|Any CPU
{6848407D-1CEF-4433-92F4-6047AE3D2C52}.Release|Any CPU.ActiveCfg = Release|Any CPU
{6848407D-1CEF-4433-92F4-6047AE3D2C52}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {A22A4DB3-DD84-46EB-96A6-7935E9E59356}
EndGlobalSection
EndGlobal
9 changes: 9 additions & 0 deletions csharp/ExcelAddIn/Properties/launchSettings.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
{
"profiles": {
"Excel": {
"commandName": "Executable",
"executablePath": "C:\\Program Files\\Microsoft Office\\root\\Office16\\EXCEL.EXE",
"commandLineArgs": "/x \"ExcelAddIn-AddIn64.xll\""
}
}
}
200 changes: 200 additions & 0 deletions csharp/ExcelAddIn/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
# Building the Excel Add-In on Windows 10 / 11.

These instructions show how to install and run the Deephaven Excel Add-In
on Windows. These instructions also happen to build the Deephaven C# Client as a
side-effect. However if your goal is to build the Deephaven C# Client,
please see [repository root]/csharp/client/README.md (does not exist yet).

We have tested these instructions on Windows 10 and 11 with Visual Studio
Community Edition.

# Before using the Excel Add-In

To actually use the Deephaven Excel Add-In, you will eventually need to have
at least one Community Core or Enterprise Core+ server running. You don't need
the server yet, and you can successfully follow these build instructions
without a server. However, you will eventually need a server when you want to
run it.

If you don't have a Deephaven Community Core server installation,
you can use these instructions to build one.
https://deephaven.io/core/docs/how-to-guides/launch-build/

Note that it is only possible to build a server on Linux. Building a server
on Windows is not currently supported.

For Deephaven Enterprise Core+, contact your IT administrator.

# Building the Excel Add-In on Windows 10 / Windows 11

## Prerequisites

## Build machine specifications

In our experience following this instructions on a fresh Windows 11 VM
required a total of 125G of disk space to install and build everything.
We recommend a machine with at least 200G of free disk space in order to
leave a comfortable margin.

Also, building the dependencies with vcpkg is very resource-intensive.
A machine that has more cores will be able to finish faster.
We recommend at least 16 cores for the build process. Note that *running*
the Excel Add-In does not require that many cores.

## Tooling

### Excel

You will need a recent version of Excel installed. We recommend Office 21
or Office 365. Note that the Add-In only works with locally-installed
versions of Excel (i.e. software installed on your computer). It does not
work with the browser-based web version of Excel.

### .NET

Install the .NET Core SDK, version 8.0.
Look for the "Windows | x64" link
[here](https://dotnet.microsoft.com/en-us/download/dotnet/8.0)

### Visual Studio

Install Visual Studio 2022 Community Edition (or Professional, or Enterprise)
from [here](https://visualstudio.microsoft.com/downloads/)

When the installer runs, select both workloads
"Desktop development with C++" and ".NET desktop development".

If Visual Studio is already installed, use Tools -> Get Tools and Features
to add those workloads if necessary.

### git

Use your preferred version of git, or install Git from
[here](https://git-scm.com/download/win)

## C++ client

The Deephaven Excel Add-In relies on the Deephaven C# Client, which in turn
requires the Deephaven C++ Client (Community Core version). Furthermore, if
you want to use Enterprise Core+ features, you also need the Deephaven C++
Client for Enterprise Core+.

The instructions below describe how to build these libraries.

### Build the Deephaven C++ Client (Community Core version)

Follow the instructions at [repository root]/cpp-client/README.md under the
section, under "Building the C++ client on Windows 10 / Windows 11".

When that process is done, you will have C++ client binaries in the
directory you specified in your DHINSTALL environment variable.

### (Optional) build the Deephaven C++ Client (Enterprise Core+ version)

To access Enterprise features, build the Enterprise Core+ version as well.
It will also store its binaries in the same DHINSTALL directory.

(instructions TODO)

## Build the Excel Add-In and C# Add-In

You can build the Add-In from inside Visual Studio or from the Visual Studio
Command Prompt.

### From within Visual Studio

1. Open the Visual Studio solution file
[repository root]\csharp\ExcelAddIn\ExcelAddIn.sln

2. Click on BUILD -> Build solution

### From the Visual Studio Command Prompt

```
cd [repository root]\csharp\ExcelAddIn
devenv ExcelAddIn.sln /build Release
```

## Run the Add-In

### From within Visual Studio

1. In order to actually function, the Add-In requires the C++ Client binaries
built in the above steps. The easiest thing to do is simply copy all the
binaries into your Visual Studio build directory:

Assuming a Debug build:

copy /Y %DHINSTALL%\bin [repository root]\csharp\ExcelAddIn\bin\Debug\net8.0-windows

If you are doing a Release build, change "Debug" to "Release" in the above path.

2. Inside Visual Studio Select Debug -> Start Debugging

Visual Studio will launch Excel automatically. Excel will launch with a
Security Notice because the add-in is not signed. Click "Enable this add-in
for this session only."

### From standalone Excel

To install the add-in into Excel, we need put the relevant files into a
directory and then point to that directory. For simplicity, we will use
the already-established %DHINSTALL%\bin directory, which already has all the
relevant files except for the add-in's XLL file.

```
copy [repository root]\csharp\ExcelAddIn\bin\Debug\net8.0-windows\publish\ExcelAddIn-Addin64-packed.xll %DHINSTALL%\bin
```

Note the above file comes from the "publish" subdirectory.

Then, run Excel and follow the following steps.

1. Click on "Options" at the lower left of the window.
2. Click on "Add-ins" on the left, second from the bottom.
3. At the bottom of the screen click, near "Manage", select "Excel Add-ins"
from the pulldown, and then click "Go..."
4. In the next screen click "Browse..."
5. Navigate to your %DHINSTALL%\bin directory and click on the ExcelAddIn-Addin64-packed.xll file that you recently copied there
6. Click OK
7. Click OK


## Test the add-in

### Without connecting to a Deephaven server

1. In Excel, click on Add-ins -> Deephaven -> Connections. This should bring
up a Connections form. If so, the C# code is functioning correctly.

2. In the following steps we deliberately use nonsense connection settings
in order to quickly trigger an error. Even thought he connection settings
are nonsense, getting an error quickly confirms that the functionality
of the C++ code is working.

3. Inside Connections, click the "New Connection" button. A "Credentials
Editor" window will pop up. Inside this window, enter "con1" for the
Connection ID, select the "Community Core" button, and enter a nonsense
endpoint address like "abc.def"

4. Press Test Credentials. You should immediately see an error like
"Can't get configuration constants. Error 14: DNS resolution failed for
abc.def"

5. Enterprise users can do a similar test by selecting the Enterprise Core+
button. Putting a nonsense protocol like abc://def in the JSON URl field
will quickly lead to an error.

### By connecting to a Deephaven server

If the above tests pass, the add-in is probably installed correctly.

To test the add-in with a Deephaven server, you will need the following
information.

1. For Community Core, you will need a Connection string in the form of
address:port. For example, 10.0.1.50:10000

2. For Enterprise Core+, you will need a JSON URL that references your
Core+ installation. For example
https://10.0.1.50:8123/iris/connection.json
Loading