mirror of https://github.com/Squidex/squidex.git
Browse Source
# Conflicts: # backend/src/Squidex.Domain.Apps.Entities/Apps/DefaultAppLogStore.cs # backend/src/Squidex.Domain.Apps.Entities/Apps/IAppLogStore.cs # backend/tests/Squidex.Domain.Apps.Entities.Tests/Apps/DefaultAppLogStoreTests.cs # backend/tests/Squidex.Web.Tests/Pipeline/UsageMiddlewareTests.cspull/568/head
31 changed files with 838 additions and 116 deletions
@ -0,0 +1,47 @@ |
|||
// ==========================================================================
|
|||
// Squidex Headless CMS
|
|||
// ==========================================================================
|
|||
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|||
// All rights reserved. Licensed under the MIT license.
|
|||
// ==========================================================================
|
|||
|
|||
using System; |
|||
using Jint; |
|||
using Jint.Native; |
|||
|
|||
namespace Squidex.Domain.Apps.Core.Scripting.Extensions |
|||
{ |
|||
public sealed class StringWordsJintExtension : IJintExtension |
|||
{ |
|||
private readonly Func<string, JsValue> wordCount = text => |
|||
{ |
|||
try |
|||
{ |
|||
return TextHelpers.WordCount(text); |
|||
} |
|||
catch |
|||
{ |
|||
return JsValue.Undefined; |
|||
} |
|||
}; |
|||
|
|||
private readonly Func<string, JsValue> characterCount = text => |
|||
{ |
|||
try |
|||
{ |
|||
return TextHelpers.CharacterCount(text); |
|||
} |
|||
catch |
|||
{ |
|||
return JsValue.Undefined; |
|||
} |
|||
}; |
|||
|
|||
public void Extend(Engine engine) |
|||
{ |
|||
engine.SetValue("wordCount", wordCount); |
|||
|
|||
engine.SetValue("characterCount", characterCount); |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,31 @@ |
|||
// ==========================================================================
|
|||
// Squidex Headless CMS
|
|||
// ==========================================================================
|
|||
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|||
// All rights reserved. Licensed under the MIT license.
|
|||
// ==========================================================================
|
|||
|
|||
using Fluid; |
|||
using Fluid.Values; |
|||
|
|||
namespace Squidex.Domain.Apps.Core.Templates.Extensions |
|||
{ |
|||
public class StringWordsFluidExtension : IFluidExtension |
|||
{ |
|||
private static readonly FilterDelegate WordCount = (input, arguments, context) => |
|||
{ |
|||
return FluidValue.Create(TextHelpers.WordCount(input.ToStringValue())); |
|||
}; |
|||
|
|||
private static readonly FilterDelegate CharacterCount = (input, arguments, context) => |
|||
{ |
|||
return FluidValue.Create(TextHelpers.CharacterCount(input.ToStringValue())); |
|||
}; |
|||
|
|||
public void RegisterGlobalTypes(IMemberAccessStrategy memberAccessStrategy) |
|||
{ |
|||
TemplateContext.GlobalFilters.AddFilter("word_count", WordCount); |
|||
TemplateContext.GlobalFilters.AddFilter("character_count", CharacterCount); |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,137 @@ |
|||
// ==========================================================================
|
|||
// Squidex Headless CMS
|
|||
// ==========================================================================
|
|||
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|||
// All rights reserved. Licensed under the MIT license.
|
|||
// ==========================================================================
|
|||
|
|||
using System.Text; |
|||
using HtmlAgilityPack; |
|||
using Markdig; |
|||
|
|||
namespace Squidex.Domain.Apps.Core |
|||
{ |
|||
public static class TextHelpers |
|||
{ |
|||
public static string Markdown2Text(string markdown) |
|||
{ |
|||
return Markdown.ToPlainText(markdown).Trim(' ', '\n', '\r'); |
|||
} |
|||
|
|||
public static string Html2Text(string html) |
|||
{ |
|||
var document = LoadHtml(html); |
|||
|
|||
var sb = new StringBuilder(); |
|||
|
|||
WriteTextTo(document.DocumentNode, sb); |
|||
|
|||
return sb.ToString().Trim(' ', '\n', '\r'); |
|||
} |
|||
|
|||
private static HtmlDocument LoadHtml(string text) |
|||
{ |
|||
var document = new HtmlDocument(); |
|||
|
|||
document.LoadHtml(text); |
|||
|
|||
return document; |
|||
} |
|||
|
|||
private static void WriteTextTo(HtmlNode node, StringBuilder sb) |
|||
{ |
|||
switch (node.NodeType) |
|||
{ |
|||
case HtmlNodeType.Comment: |
|||
break; |
|||
case HtmlNodeType.Document: |
|||
WriteChildrenTextTo(node, sb); |
|||
break; |
|||
case HtmlNodeType.Text: |
|||
var html = ((HtmlTextNode)node).Text; |
|||
|
|||
if (HtmlNode.IsOverlappedClosingElement(html)) |
|||
{ |
|||
break; |
|||
} |
|||
|
|||
if (!string.IsNullOrWhiteSpace(html)) |
|||
{ |
|||
sb.Append(HtmlEntity.DeEntitize(html)); |
|||
} |
|||
|
|||
break; |
|||
|
|||
case HtmlNodeType.Element: |
|||
switch (node.Name) |
|||
{ |
|||
case "p": |
|||
sb.AppendLine(); |
|||
break; |
|||
case "br": |
|||
sb.AppendLine(); |
|||
break; |
|||
case "style": |
|||
return; |
|||
case "script": |
|||
return; |
|||
} |
|||
|
|||
if (node.HasChildNodes) |
|||
{ |
|||
WriteChildrenTextTo(node, sb); |
|||
} |
|||
|
|||
break; |
|||
} |
|||
} |
|||
|
|||
private static void WriteChildrenTextTo(HtmlNode node, StringBuilder sb) |
|||
{ |
|||
foreach (var child in node.ChildNodes) |
|||
{ |
|||
WriteTextTo(child, sb); |
|||
} |
|||
} |
|||
|
|||
public static int CharacterCount(string text) |
|||
{ |
|||
var characterCount = 0; |
|||
|
|||
for (int i = 0; i < text.Length; i++) |
|||
{ |
|||
if (char.IsLetter(text[i])) |
|||
{ |
|||
characterCount++; |
|||
} |
|||
} |
|||
|
|||
return characterCount; |
|||
} |
|||
|
|||
public static int WordCount(string text) |
|||
{ |
|||
var numWords = 0; |
|||
|
|||
for (int i = 1; i < text.Length; i++) |
|||
{ |
|||
if (char.IsWhiteSpace(text[i - 1])) |
|||
{ |
|||
var character = text[i]; |
|||
|
|||
if (char.IsLetterOrDigit(character) || char.IsPunctuation(character)) |
|||
{ |
|||
numWords++; |
|||
} |
|||
} |
|||
} |
|||
|
|||
if (text.Length > 2) |
|||
{ |
|||
numWords++; |
|||
} |
|||
|
|||
return numWords; |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,30 @@ |
|||
// ==========================================================================
|
|||
// Squidex Headless CMS
|
|||
// ==========================================================================
|
|||
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|||
// All rights reserved. Licensed under the MIT license.
|
|||
// ==========================================================================
|
|||
|
|||
using NodaTime; |
|||
|
|||
namespace Squidex.Domain.Apps.Entities.Apps |
|||
{ |
|||
public struct RequestLog |
|||
{ |
|||
public Instant Timestamp; |
|||
|
|||
public string? RequestMethod; |
|||
|
|||
public string? RequestPath; |
|||
|
|||
public string? UserId; |
|||
|
|||
public string? UserClientId; |
|||
|
|||
public long ElapsedMs; |
|||
|
|||
public long Bytes; |
|||
|
|||
public double Costs; |
|||
} |
|||
} |
|||
@ -0,0 +1,25 @@ |
|||
version: '3.4' |
|||
|
|||
networks: |
|||
grafana: |
|||
|
|||
services: |
|||
influxdb: |
|||
image: influxdb:latest |
|||
networks: |
|||
- grafana |
|||
ports: |
|||
- "8086:8086" |
|||
environment: |
|||
- INFLUXDB_DB=k6 |
|||
|
|||
grafana: |
|||
image: grafana/grafana:latest |
|||
networks: |
|||
- grafana |
|||
ports: |
|||
- "4000:3000" |
|||
environment: |
|||
- GF_AUTH_ANONYMOUS_ORG_ROLE=Admin |
|||
- GF_AUTH_ANONYMOUS_ENABLED=true |
|||
- GF_AUTH_BASIC_ENABLED=false |
|||
@ -0,0 +1,35 @@ |
|||
import { check } from 'k6'; |
|||
import http from 'k6/http'; |
|||
import { variables, getBearerToken } from './shared.js'; |
|||
|
|||
export let options = { |
|||
stages: [ |
|||
{ duration: "2m", target: 200 }, |
|||
{ duration: "2m", target: 200 }, |
|||
{ duration: "2m", target: 0 }, |
|||
], |
|||
thresholds: { |
|||
'http_req_duration': ['p(99)<1500'], // 99% of requests must complete below 1.5s
|
|||
} |
|||
}; |
|||
|
|||
|
|||
export function setup() { |
|||
const token = getBearerToken(); |
|||
|
|||
return { token }; |
|||
} |
|||
|
|||
export default function (data) { |
|||
const url = `${variables.serverUrl}/api/apps/${variables.appName}/clients`; |
|||
|
|||
const response = http.get(url, { |
|||
headers: { |
|||
Authorization: `Bearer ${data.token}` |
|||
} |
|||
}); |
|||
|
|||
check(response, { |
|||
'is status 200': (r) => r.status === 200, |
|||
}); |
|||
} |
|||
@ -0,0 +1,35 @@ |
|||
import http from 'k6/http'; |
|||
|
|||
export const variables = { |
|||
appName: getValue('APP__NAME', 'integration-tests'), |
|||
clientId: getValue('CLIENT__ID', 'root'), |
|||
clientSecret: getValue('CLIENT__SECRET', 'xeLd6jFxqbXJrfmNLlO2j1apagGGGSyZJhFnIuHp4I0='), |
|||
serverUrl: getValue('SERVER__URL', 'https://localhost:5001') |
|||
}; |
|||
|
|||
let bearerToken = null; |
|||
|
|||
export function getBearerToken() { |
|||
if (!bearerToken) { |
|||
const url = `${variables.serverUrl}/identity-server/connect/token`; |
|||
|
|||
const response = http.post(url, { |
|||
grant_type: 'client_credentials', |
|||
client_id: variables.clientId, |
|||
client_secret: variables.clientSecret, |
|||
scope: 'squidex-api' |
|||
}); |
|||
|
|||
const json = JSON.parse(response.body); |
|||
|
|||
bearerToken = json.access_token; |
|||
} |
|||
|
|||
return bearerToken; |
|||
} |
|||
|
|||
function getValue(key, fallback) { |
|||
const result = __ENV[key] || fallback; |
|||
|
|||
return result; |
|||
} |
|||
Loading…
Reference in new issue