Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tyler-technologies-oss/tyler-socrata-dotnet-client


https://github.com/tyler-technologies-oss/tyler-socrata-dotnet-client

Last synced: 24 days ago
JSON representation

Awesome Lists containing this project

README

        

# tyler-socrata-net-sdk
[![build](https://github.com/tyler-technologies-oss/tyler-socrata-dotnet-client/actions/workflows/dotnet-core.yml/badge.svg)](https://github.com/tyler-technologies-oss/tyler-socrata-dotnet-client/actions/workflows/dotnet-core.yml)

.NET Socrata SDK for reading and writing to Socrata APIs

SocrataClient
-----
```c#
// Create the client
ISocrataClient client = new SocrataClient(new Uri("https://{domain}.data.socrata.com"), "apikey", "apisecret");

// Optionally validate the client connection
bool IsValidConnection = client.ValidateConnection();

// Listing all schedules on a domain
List allSchedules = client.GetSchedules();

// Listing all Gateway agents on a domain
List allAgents = client.GetAgents();
```

Socrata Resources
-----
```c#
// Get a resource by its ID*
Resource dataset = client.GetResource("abcd-1234");

// Get a resource by its Resource ID Alias*
Resource dataset = client.GetResourceByAlias("transaction_amounts");

// Get all resources on a domain
List resources = socrataClient.GetResources();
resources.ForEach((resource) => { System.Console.WriteLine(resource.Id) });

// Get Latest Activity Log on the Domain
List activities = socrataClient.GetLatestActivityLog();
activities.ForEach((activity) => {
Console.WriteLine(activity.ActivityType);
});
```
***Note***
[Dataset IDs](https://support.socrata.com/hc/en-us/articles/202950258-What-is-a-Dataset-UID-or-a-Dataset-4x4-)

Creating a Resource
-----
```c#
// Through DSMAPI
DsmapiResourceBuilder builder = socrataClient.CreateDsmapiResourceBuilder("ToDelete");
Revision initialRevision = builder
.SetDescription($"{System.DateTime.Now} TEST")
.Build();
Source source = initialRevision.CreateUploadSource("test-csv.csv");
string filepath = @"Incidents.csv";
string csv = System.IO.File.ReadAllText(filepath);
source.AddBytesToSource(csv);
source.AwaitCompletion(status => Console.WriteLine(status));
OutputSchema os = source.GetLatestOutputSchema();
os.ChangeColumnName("agency", "agency2")
.ChangeColumnDescription("area_district", "TEST")
.ChangeTransform("area_beat", Transforms.NUMBER("area_beat"))
.ChangeTransform("area_station", Transforms.CUSTOM("upper('TEST')"))
.ChangeTransform("area_quadrant", Transforms.CUSTOM("replace(" + Transforms.TEXT("incident_id").Value + ", '0', '1')"))
.Submit();
source.AwaitCompletion(status => Console.WriteLine(status));
bool valid = os.ValidateRowId("incident_id");
if(!valid)
{
throw Exception("Invalid Row Id");
}
os.SetRowId("incident_id").Submit();
source.AwaitCompletion(status => Console.WriteLine(status));
initialRevision.Apply();
initialRevision.AwaitCompletion(status => Console.WriteLine(status));
Resource newResource = new Resource(initialRevision.Metadata.FourFour, socrataClient.httpClient);
// Optionally set the Resource Alias - this allows you to refer to the dataset
// later by this alias without knowing the dataset ID
// Aliases MUST be lower case, alphanumeric + `-` and `_` , unique to that domain
newResource.SetResourceIdAlias("to_delete")

// Optionally set the permission level of the resource - by default they are created as PRIVATE assets
newResource.SetAudience(AudienceLevel.INTERNAL);

// Through SODA
Column id = new Column("id", SocrataDataType.TEXT);
SODASchema schema = new SchemaBuilder()
.AddColumn(id)
.AddColumn(new Column("text", SocrataDataType.TEXT))
.AddColumn(new Column("number", SocrataDataType.NUMBER))
.AddColumn(new Column("point", SocrataDataType.POINT))
.AddColumn(new Column("date", SocrataDataType.DATETIME, "Data Column Description"))
.AddColumn(new Column("bool", SocrataDataType.BOOLEAN))
.Build();
Resource newDataset = socrataClient.CreateSodaResourceBuilder("My New Dataset")
.SetSchema(schema)
.SetRowIdentifier(id)
.Build();
// Optionally set the Resource Alias - this allows you to refer to the dataset
// later by this alias without knowing the dataset ID
// Aliases MUST be lower case, alphanumeric + `-` and `_` , unique to that domain
newDataset.SetResourceIdAlias("my_new_dataset")
// Optionally set the permission level of the resource - by default they are created as PRIVATE assets
newDataset.SetAudience(AudienceLevel.INTERNAL);

```

Creating a View
-----
*NOTE* When creating a SoQL View, some known query patterns are currently not supported. In particular:
- `SELECT *` is not permitted, columns must be explicitly referenced

```c#
//// BASIC QUERIES
// Get a resource (by ID or ALIAS)
Resource dataset = client.GetResource("abcd-1234");

string query = @"
SELECT transaction_type, sum(amount)
";

View view = cases.CreateViewFromSoQL(query);

//// Advanced Queries
// Get a resource (by ID or ALIAS)
Resource cases = client.GetResourceByAlias("cases");
Resource caseoffenses = client.GetResourceByAlias("caseoffenses");
Resource addresses = client.GetResourceByAlias("addresses");

string query = @"
SELECT casenumber, @cs.case_offense_type, @ad.centroid
LEFT OUTER JOIN @caseoffenses as cs ON @cs.case_id = case_id
LEFT OUTER JOIN @addresses as ad ON @ad.address_id = address_id
";

CollocationJob collocate = cases.CollocateToResources(List{
caseoffenses,
addresses
});
// Run the collocation (might take a while if datasets are large)
collocate.Run(status => Console.WriteLine(status));

// Create your view
View view = cases.CreateViewFromSoQL(query);
```

Working with Resources
----
### Permissions
Permissions are managed at the view or resource level.
```c#
// Audience Levels:
// AudienceLevel.PRIVATE = private asset, only available to roled domain managers, dataset owner, and administrators
// AudienceLevel.INTERNAL = internally private to the site, only available to roled users on the domain
// AudienceLevel.PUBLIC = externally public, *NOTE* on most sites the asset will need to be APPROVED in order to be made public

// Private assets can have private, internal, and public views.

// Get the Resource
Resource resource = socrataClient.GetResournce("abcd-1234");
// Change its Audience Level
resource.SetAudience(AudienceLevel.PRIVATE);

// Create an internal view
View view = resource.CreateViewFromSoQL("view", "SELECT id", AudienceLevel.INTERNAL);
// Optionally, change the view after the fact
view.SetAudience(AudienceLevel.PRIVATE);

```

### Scheduling A Resource
```c#
Resource resource = socrataClient.GetResournce("abcd-1234");
// Get Existing Schedule
Schedule schedule = resource.GetSchedule()
// ==> Results in an HTTP Error if no schedule exists

// New Schedules
ScheduleResource newSchedule = new ScheduleResource
{
Paused = false,
Cadence = new ScheduleCadence {
Cron = "0 9 * * *",
Timezone = "America/New_York"
},
Action = new ScheduleAction {
Type = "connection_agent",
Parameters = new ScheduleParameters {
AgentUid = "1234-abcd-lmno-9876",
Parameters = new Dictionary {},
Path = new List {},
Namespace = new ScheduleAgentNamespace {
Type = "csv",
Name = "CSV Plugin"
}
}
}
};
resource.CreateSchedule(newSchedule);
// Updating Existing Schedules
newSchedule.Paused = true; // This will pause the schedule
resource.UpdateSchedule(newSchedule);
// Deleting a schedule
resource.DeleteSchedule();
```
### Data Manipulation
#### Socrata Data Types and You
| Socrata Data Type | Rough SQL Mapping | Example |
| ----------------- | ----------------- | ------- |
| Text | varchar, text | "Text" |
| Number | int, float, double | 1, 3.14 |
| Date | convert(date, 426) | "2020-01-01" |
| Timestamp | convert(datetime, 426) | "2020-01-01T13:42:10" |
| Boolean | boolean, bit | 1, true |
| Null | null | null |
***Notes:***
- The escape character for text values is `\`
- Dates must be passed in as string
- Datetimes will not be converted to specific timezones, so be sure all the dates are in the correct/same timezone when they are exported
- Decimals are demarcated with a `.`
- All of these will be converted to JSON when sent to Socrata, so when in doubt, consult the

#### Geospatial Data

Any geospatial data will need to converted into [Well-Known Text (WKT)](https://en.wikipedia.org/wiki/Well-known_text_representation_of_geometry) with WGS84 (ESPG 4326) Latitude and Longitude values or geocoded via a DSMAPI transform.

In general, a decimal precision of 5 is sufficient.

| Socrata Data Type | Example |
| ----------------- | ------- |
| Point | "POINT (37.192 -168.223)" |
| MultiPoint | "MULTIPOINT ((37.192 -168.223),(42.1110 -172.9339))" |
| LineString | "LINESTRING (30 10, 10 30, 40 40)" |
| MultiLineString | "MULTILINESTRING ((10 10, 20 20, 10 40),(40 40, 30 30, 40 20, 30 10))" |
| Polygon | "POLYGON ((35 10, 45 45, 15 40, 10 20, 35 10),(20 30, 35 35, 30 20, 20 30))" |
| MultiPolygon | "MULTIPOLYGON (((30 20, 45 40, 10 40, 30 20)),((15 5, 40 10, 10 20, 5 10, 15 5)))" |

Direct Row Manipulation
---

#### Insert
```c#
// Insert Functionality
List> newRecords = new List>
{
new Dictionary{{"api_field", "nextvalue"},{"other_field", "test"}}
};
Result result = dataset.Rows().Insert(newRecords);
```
##### Update
```c#
// Update Functionality
List> recordsToUpdate = new List>
{
new Dictionary{{"row_identifier", 1},{"column_that_changed", "newvalue"}}
};
Result result = dataset.Rows().Update(recordsToUpdate);
```
##### Delete
```c#
// Delete Functionality
List> recordsToDelete = new List>
{
new Dictionary{{"row_identifier", "ID-1"}}
};
Result result = dataset.Rows().Delete(recordsToDelete);
```
##### Check Status
```c#
// Check if it was successful
Assert.IsFalse(Result.IsError);
// Check the results
Assert.AreEqual(newRecords.Length, result.Inserted);
```

Dataset Management API
---
[Full Documentation](https://socratapublishing.docs.apiary.io/#introduction/examples)

### File Types
| file extension | content type |
| --- | --- |
| CSV | text/csv |
| TSV | text/tsv |
| KML | google-kml |
| ZIP (shapefile) | application/octet-stream |
| BLOB | application/octet-stream |

### Create a Revision
```c#
Resource resource = socrataClient.GetResource("tzmz-8bnb");
Revision revision = resource.OpenRevision(RevisionType.REPLACE); // UPDATE/DELETE
Source source = revision.CreateUploadSource("test-csv.csv");
```

### Small (in memory) file uploads
```c#
string filepath = @"C:\Path\To\MyFile.csv";
string csv = System.IO.File.ReadAllText(filepath);
source.AddBytesToSource(csv);
```

### Large (Streaming) file uploads
```c#
ByteSink sink = source.StreamingSource(ContentType.CSV);
//Open the stream and read the file.
sink.FileSink(@"C:\Path\To\MyFile.csv");
```

### Handling error rows
```c#
if(source.HasErrorRows())
{
string errorFile = "C:\Path\To\ErrorRows.csv";
long numberOfErrors = source.NumberOfErrors();
source.ExportErrorRows(errorFile);
throw new Exception($"{numberOfErrors} Error(s) were found in the upload");
}
```

### Dataset Metadata Only Revisions
```c#
Resource resource = socrataClient.GetResource("tzmz-8bnb");
Revision revision = resource.OpenRevision(RevisionType.REPLACE);
revision.CreateViewSource();
revision.SetDescription("New description for the resource");
revision.RenameResource("New Name");
```

### Apply the revision
```c#
source.AwaitCompletion(status => Console.WriteLine(status));
revision.Apply();
revision.AwaitCompletion(status => Console.WriteLine(status));
```

Reading Data
---
You can also use the Rows() to access records in the dataset
```c#
/* Example custom class
[DataContract]
class MyResource
{
[DataMember(Name = "column_field")]
public string ColumnField { get; internal set; }
}
*/
Resource resource = socrataClient.GetResource("tzmz-8bnb");
Rows rows = resource.Rows();
List allRecords = rows.FetchAll();
long limit = 1000;
long offset = 0;
List someRecords = rows.Fetch(limit, offset);
// Conversely, you can just use a Map
List> allRecords = rows.FetchAll>();
```

### Pagination
You might have a really big dataset, which will require paginating over the result set.

```c#
Resource resource = socrataClient.GetResource("tzmz-8bnb");
Rows rows = resource.Rows();
List> allRecords = new List>()
// Get the total number of records
long total = rows.Count();
long limit = 1000;
long offset = 0;
while(offset < total)
{
allRecordsrows.Add(rows.Fetch>(limit, offset));
offset += limit;
}
```

Metadata
---

### Accessing Resource Metadata
```c#
Resource dataset = client.GetResource("abcd-1234");
string resourceName = dataset.metadata.Name;
```

Running Tests
----

```bash
dotnet test
```

Or a specific test:
```bash
dotnet test --filter {testname}
```