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