axyseo 2.1.13 → 2.1.20
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/cjs/const/analysis.js +40 -3
- package/build/cjs/const/analysis.js.map +1 -1
- package/build/cjs/index.js.map +1 -1
- package/build/cjs/languageProcessing/AbstractResearcher.js +3 -1
- package/build/cjs/languageProcessing/AbstractResearcher.js.map +1 -1
- package/build/cjs/languageProcessing/researches/charactersInUrl.js +26 -0
- package/build/cjs/languageProcessing/researches/charactersInUrl.js.map +1 -0
- package/build/cjs/languageProcessing/researches/checkRelatedKeywords.js +0 -1
- package/build/cjs/languageProcessing/researches/checkRelatedKeywords.js.map +1 -1
- package/build/cjs/languageProcessing/researches/getKeywordDensity.js.map +1 -1
- package/build/cjs/scoring/assessments/assessment.js +5 -6
- package/build/cjs/scoring/assessments/assessment.js.map +1 -1
- package/build/cjs/scoring/assessments/index.js +12 -4
- package/build/cjs/scoring/assessments/index.js.map +1 -1
- package/build/cjs/scoring/assessments/readability/ParagraphTooLongAssessment.js +18 -17
- package/build/cjs/scoring/assessments/readability/ParagraphTooLongAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/readability/RelatedKeywordsAssessment.js +11 -25
- package/build/cjs/scoring/assessments/readability/RelatedKeywordsAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/readability/SentenceBeginningsAssessment.js +1 -1
- package/build/cjs/scoring/assessments/readability/SentenceBeginningsAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/readability/SentenceLengthInTextAssessment.js +9 -10
- package/build/cjs/scoring/assessments/readability/SentenceLengthInTextAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/readability/TransitionWordsAssessment.js +1 -1
- package/build/cjs/scoring/assessments/readability/TransitionWordsAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/FAQsAssessment.js +7 -52
- package/build/cjs/scoring/assessments/seo/FAQsAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/FAQsStructureDataAssessment.js +83 -0
- package/build/cjs/scoring/assessments/seo/FAQsStructureDataAssessment.js.map +1 -0
- package/build/cjs/scoring/assessments/seo/ImageCountAssessment.js +2 -2
- package/build/cjs/scoring/assessments/seo/ImageCountAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/InternalLinksAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/InternalLinksAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/IntroductionKeywordAssessment.js +2 -4
- package/build/cjs/scoring/assessments/seo/IntroductionKeywordAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/InvalidCharactersInUrl.js +93 -0
- package/build/cjs/scoring/assessments/seo/InvalidCharactersInUrl.js.map +1 -0
- package/build/cjs/scoring/assessments/seo/KeyphraseAssessment.js +4 -4
- package/build/cjs/scoring/assessments/seo/KeyphraseAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/KeyphraseLengthAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/KeyphraseLengthAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/KeywordDensityAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/KeywordDensityAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/KeywordFAQsAssessment.js +3 -3
- package/build/cjs/scoring/assessments/seo/KeywordFAQsAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/MetaDescriptionKeywordAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/MetaDescriptionKeywordAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/MetaDescriptionLengthAssessment.js +3 -6
- package/build/cjs/scoring/assessments/seo/MetaDescriptionLengthAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/MetaTitleKeywordAssessment.js +2 -2
- package/build/cjs/scoring/assessments/seo/MetaTitleKeywordAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/NumberInMetaTitleAssessment.js +2 -2
- package/build/cjs/scoring/assessments/seo/NumberInMetaTitleAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/OutboundLinksAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/OutboundLinksAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/PageTitleWidthAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/PageTitleWidthAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/RelatedKeywordsDensityAssessment.js +103 -0
- package/build/cjs/scoring/assessments/seo/RelatedKeywordsDensityAssessment.js.map +1 -0
- package/build/cjs/scoring/assessments/seo/SchemaAssessment.js +16 -2
- package/build/cjs/scoring/assessments/seo/SchemaAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/SingleH1Assessment.js +3 -4
- package/build/cjs/scoring/assessments/seo/SingleH1Assessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/SingleTitleAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/SingleTitleAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/SubHeadingsKeywordAssessment.js +3 -5
- package/build/cjs/scoring/assessments/seo/SubHeadingsKeywordAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/TextImagesAssessment.js +2 -3
- package/build/cjs/scoring/assessments/seo/TextImagesAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/TextLengthAssessment.js +22 -7
- package/build/cjs/scoring/assessments/seo/TextLengthAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/UrlKeywordAssessment.js +3 -4
- package/build/cjs/scoring/assessments/seo/UrlKeywordAssessment.js.map +1 -1
- package/build/cjs/scoring/assessments/seo/UrlLengthAssessment.js +2 -4
- package/build/cjs/scoring/assessments/seo/UrlLengthAssessment.js.map +1 -1
- package/build/cjs/scoring/assessors/assessor.js +0 -1
- package/build/cjs/scoring/assessors/assessor.js.map +1 -1
- package/build/cjs/scoring/assessors/avadaAssessor.js +1 -2
- package/build/cjs/scoring/assessors/avadaAssessor.js.map +1 -1
- package/build/cjs/scoring/assessors/contentAssessor.js +1 -1
- package/build/cjs/scoring/assessors/contentAssessor.js.map +1 -1
- package/build/cjs/scoring/assessors/index.js +0 -14
- package/build/cjs/scoring/assessors/index.js.map +1 -1
- package/build/cjs/scoring/assessors/seoAssessor.js +5 -2
- package/build/cjs/scoring/assessors/seoAssessor.js.map +1 -1
- package/build/cjs/values/Paper.js +11 -2
- package/build/cjs/values/Paper.js.map +1 -1
- package/build/esm/const/analysis.js +39 -2
- package/build/esm/const/analysis.js.map +1 -1
- package/build/esm/index.js.map +1 -1
- package/build/esm/languageProcessing/AbstractResearcher.js +3 -1
- package/build/esm/languageProcessing/AbstractResearcher.js.map +1 -1
- package/build/esm/languageProcessing/researches/charactersInUrl.js +21 -0
- package/build/esm/languageProcessing/researches/charactersInUrl.js.map +1 -0
- package/build/esm/languageProcessing/researches/checkRelatedKeywords.js +0 -1
- package/build/esm/languageProcessing/researches/checkRelatedKeywords.js.map +1 -1
- package/build/esm/languageProcessing/researches/getKeywordDensity.js.map +1 -1
- package/build/esm/scoring/assessments/assessment.js +6 -7
- package/build/esm/scoring/assessments/assessment.js.map +1 -1
- package/build/esm/scoring/assessments/index.js +12 -4
- package/build/esm/scoring/assessments/index.js.map +1 -1
- package/build/esm/scoring/assessments/readability/ParagraphTooLongAssessment.js +19 -18
- package/build/esm/scoring/assessments/readability/ParagraphTooLongAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/readability/RelatedKeywordsAssessment.js +12 -26
- package/build/esm/scoring/assessments/readability/RelatedKeywordsAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/readability/SentenceBeginningsAssessment.js +2 -2
- package/build/esm/scoring/assessments/readability/SentenceBeginningsAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/readability/SentenceLengthInTextAssessment.js +10 -11
- package/build/esm/scoring/assessments/readability/SentenceLengthInTextAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/readability/TransitionWordsAssessment.js +2 -2
- package/build/esm/scoring/assessments/readability/TransitionWordsAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/FAQsAssessment.js +8 -53
- package/build/esm/scoring/assessments/seo/FAQsAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/FAQsStructureDataAssessment.js +76 -0
- package/build/esm/scoring/assessments/seo/FAQsStructureDataAssessment.js.map +1 -0
- package/build/esm/scoring/assessments/seo/ImageCountAssessment.js +3 -3
- package/build/esm/scoring/assessments/seo/ImageCountAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/InternalLinksAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/InternalLinksAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/IntroductionKeywordAssessment.js +3 -5
- package/build/esm/scoring/assessments/seo/IntroductionKeywordAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/InvalidCharactersInUrl.js +87 -0
- package/build/esm/scoring/assessments/seo/InvalidCharactersInUrl.js.map +1 -0
- package/build/esm/scoring/assessments/seo/KeyphraseAssessment.js +5 -5
- package/build/esm/scoring/assessments/seo/KeyphraseAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/KeyphraseLengthAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/KeyphraseLengthAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/KeywordDensityAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/KeywordDensityAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/KeywordFAQsAssessment.js +4 -4
- package/build/esm/scoring/assessments/seo/KeywordFAQsAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/MetaDescriptionKeywordAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/MetaDescriptionKeywordAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/MetaDescriptionLengthAssessment.js +4 -7
- package/build/esm/scoring/assessments/seo/MetaDescriptionLengthAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/MetaTitleKeywordAssessment.js +3 -3
- package/build/esm/scoring/assessments/seo/MetaTitleKeywordAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/NumberInMetaTitleAssessment.js +3 -3
- package/build/esm/scoring/assessments/seo/NumberInMetaTitleAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/OutboundLinksAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/OutboundLinksAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/PageTitleWidthAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/PageTitleWidthAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/RelatedKeywordsDensityAssessment.js +96 -0
- package/build/esm/scoring/assessments/seo/RelatedKeywordsDensityAssessment.js.map +1 -0
- package/build/esm/scoring/assessments/seo/SchemaAssessment.js +17 -3
- package/build/esm/scoring/assessments/seo/SchemaAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/SingleH1Assessment.js +4 -5
- package/build/esm/scoring/assessments/seo/SingleH1Assessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/SingleTitleAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/SingleTitleAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/SubHeadingsKeywordAssessment.js +4 -6
- package/build/esm/scoring/assessments/seo/SubHeadingsKeywordAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/TextImagesAssessment.js +3 -4
- package/build/esm/scoring/assessments/seo/TextImagesAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/TextLengthAssessment.js +24 -8
- package/build/esm/scoring/assessments/seo/TextLengthAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/UrlKeywordAssessment.js +4 -5
- package/build/esm/scoring/assessments/seo/UrlKeywordAssessment.js.map +1 -1
- package/build/esm/scoring/assessments/seo/UrlLengthAssessment.js +3 -5
- package/build/esm/scoring/assessments/seo/UrlLengthAssessment.js.map +1 -1
- package/build/esm/scoring/assessors/assessor.js +0 -1
- package/build/esm/scoring/assessors/assessor.js.map +1 -1
- package/build/esm/scoring/assessors/avadaAssessor.js +2 -3
- package/build/esm/scoring/assessors/avadaAssessor.js.map +1 -1
- package/build/esm/scoring/assessors/contentAssessor.js +1 -1
- package/build/esm/scoring/assessors/contentAssessor.js.map +1 -1
- package/build/esm/scoring/assessors/index.js +0 -2
- package/build/esm/scoring/assessors/index.js.map +1 -1
- package/build/esm/scoring/assessors/seoAssessor.js +5 -2
- package/build/esm/scoring/assessors/seoAssessor.js.map +1 -1
- package/build/esm/values/Paper.js +11 -2
- package/build/esm/values/Paper.js.map +1 -1
- package/package.json +3 -3
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { merge } from 'lodash';
|
|
2
2
|
import Assessment from "../assessment";
|
|
3
3
|
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
|
-
import { FAQs_ASSESSMENT_ID } from "../../../const/analysis";
|
|
4
|
+
import { FAQs_ASSESSMENT_ID, FAQS_POINTS } from "../../../const/analysis";
|
|
5
5
|
|
|
6
6
|
/**
|
|
7
7
|
* Represents an assessment that checks FAQs content and structure for SGE optimization.
|
|
@@ -22,14 +22,12 @@ export default class FAQsAssessment extends Assessment {
|
|
|
22
22
|
docUrl: 'https://blog.google/products/search/generative-ai-search/',
|
|
23
23
|
priority: 'high',
|
|
24
24
|
fixPosition: 'faqs',
|
|
25
|
-
factor: 2,
|
|
26
25
|
title: 'FAQs Optimization',
|
|
27
26
|
content: {
|
|
28
27
|
good: 'FAQs are well-structured and optimized for Google SGE.',
|
|
29
|
-
improve: '
|
|
28
|
+
improve: '',
|
|
30
29
|
bad: 'FAQs need significant improvement to meet Google SGE standards.',
|
|
31
30
|
details: {
|
|
32
|
-
hasSchema: 'Missing FAQ schema markup',
|
|
33
31
|
hasProperStructure: 'Empty questions or answers',
|
|
34
32
|
hasMinimumFAQs: 'Less than 3 FAQs'
|
|
35
33
|
}
|
|
@@ -67,7 +65,7 @@ export default class FAQsAssessment extends Assessment {
|
|
|
67
65
|
* @returns {{score: number, status: string, data: {hasSchema: ((function(): *)|boolean), hasProperStructure: boolean, hasMinimumFAQs: boolean}}|{score: number, status: string, data: {hasFAQs: boolean}}}
|
|
68
66
|
*/
|
|
69
67
|
calculateResult(paper) {
|
|
70
|
-
const faqs = paper.getFAQs();
|
|
68
|
+
const faqs = paper.getFAQs()?.questions || [];
|
|
71
69
|
let status = 'bad';
|
|
72
70
|
let score = 0;
|
|
73
71
|
if (!faqs || faqs.length === 0) {
|
|
@@ -79,58 +77,16 @@ export default class FAQsAssessment extends Assessment {
|
|
|
79
77
|
}
|
|
80
78
|
};
|
|
81
79
|
}
|
|
82
|
-
|
|
83
|
-
// Check FAQ structure and content
|
|
84
|
-
const hasSchema = paper.hasFAQsSchema();
|
|
85
|
-
const hasProperStructure = this.checkFAQStructure(faqs);
|
|
80
|
+
const isEnableFaqs = paper.getFAQs()?.status;
|
|
86
81
|
const hasMinimumFAQs = this.checkMinimumFAQs(faqs);
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
const criteriaMet = [hasSchema, hasProperStructure, hasMinimumFAQs].filter(Boolean).length;
|
|
90
|
-
if (criteriaMet === 3) {
|
|
91
|
-
status = 'good';
|
|
92
|
-
} else if (criteriaMet === 2) {
|
|
93
|
-
status = 'improve';
|
|
94
|
-
} else {
|
|
95
|
-
status = 'bad';
|
|
96
|
-
}
|
|
97
|
-
score = this.getScore(this._config.priority, status, this._config.factor);
|
|
82
|
+
status = hasMinimumFAQs && isEnableFaqs ? 'good' : 'bad';
|
|
83
|
+
score = this.getScore(FAQS_POINTS, status);
|
|
98
84
|
return {
|
|
99
85
|
score,
|
|
100
|
-
status
|
|
101
|
-
data: {
|
|
102
|
-
hasSchema,
|
|
103
|
-
hasProperStructure,
|
|
104
|
-
hasMinimumFAQs
|
|
105
|
-
}
|
|
86
|
+
status
|
|
106
87
|
};
|
|
107
88
|
}
|
|
108
89
|
|
|
109
|
-
/**
|
|
110
|
-
* Checks if FAQs have proper structure
|
|
111
|
-
* @param {Array} faqs
|
|
112
|
-
* @returns {boolean}
|
|
113
|
-
*/
|
|
114
|
-
checkFAQStructure(faqs) {
|
|
115
|
-
return faqs.every(faq => {
|
|
116
|
-
return faq.question && faq.answer && typeof faq.question === 'string' && typeof faq.answer === 'string' && faq.question.length > 0 && faq.answer.length > 0;
|
|
117
|
-
});
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
/**
|
|
121
|
-
* Checks if FAQs are relevant to the main content
|
|
122
|
-
* @param {Array} faqs
|
|
123
|
-
* @param {Array} keywords
|
|
124
|
-
* @returns {boolean}
|
|
125
|
-
*/
|
|
126
|
-
checkRelevance(faqs, keywords) {
|
|
127
|
-
if (!keywords || keywords.length === 0) return true;
|
|
128
|
-
return faqs.some(faq => {
|
|
129
|
-
const question = faq.question.toLowerCase();
|
|
130
|
-
return keywords.some(keyword => question.includes(keyword.toLowerCase()));
|
|
131
|
-
});
|
|
132
|
-
}
|
|
133
|
-
|
|
134
90
|
/**
|
|
135
91
|
* Checks if there are at least 3 FAQs
|
|
136
92
|
* @param {Array} faqs
|
|
@@ -148,8 +104,7 @@ export default class FAQsAssessment extends Assessment {
|
|
|
148
104
|
* @returns {boolean} True when there is text.
|
|
149
105
|
*/
|
|
150
106
|
isApplicable(paper) {
|
|
151
|
-
|
|
152
|
-
return 'faqs' in data;
|
|
107
|
+
return true;
|
|
153
108
|
}
|
|
154
109
|
}
|
|
155
110
|
//# sourceMappingURL=FAQsAssessment.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"FAQsAssessment.js","names":["merge","Assessment","AssessmentResult","FAQs_ASSESSMENT_ID","FAQsAssessment","constructor","config","defaultConfig","id","ctaType","docUrl","priority","fixPosition","
|
|
1
|
+
{"version":3,"file":"FAQsAssessment.js","names":["merge","Assessment","AssessmentResult","FAQs_ASSESSMENT_ID","FAQS_POINTS","FAQsAssessment","constructor","config","defaultConfig","id","ctaType","docUrl","priority","fixPosition","title","content","good","improve","bad","details","hasProperStructure","hasMinimumFAQs","identifier","_config","getResult","paper","researcher","calculatedResult","calculateResult","assessmentResult","setScore","score","setStatus","status","setData","data","faqs","getFAQs","questions","length","hasFAQs","isEnableFaqs","checkMinimumFAQs","getScore","isApplicable"],"sources":["../../../../../src/scoring/assessments/seo/FAQsAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {FAQs_ASSESSMENT_ID, FAQS_POINTS} from '@axyseo/const/analysis';\n\n/**\n * Represents an assessment that checks FAQs content and structure for SGE optimization.\n */\nexport default class FAQsAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} [config] The configuration to use.\n *\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: FAQs_ASSESSMENT_ID,\n ctaType: 'fix',\n docUrl: 'https://blog.google/products/search/generative-ai-search/',\n priority: 'high',\n fixPosition: 'faqs',\n title: 'FAQs Optimization',\n content: {\n good: 'FAQs are well-structured and optimized for Google SGE.',\n improve: '',\n bad: 'FAQs need significant improvement to meet Google SGE standards.',\n details: {\n hasProperStructure: 'Empty questions or answers',\n hasMinimumFAQs: 'Less than 3 FAQs'\n }\n }\n };\n\n this.identifier = FAQs_ASSESSMENT_ID;\n this._config = merge(defaultConfig, config);\n }\n\n /**\n * Executes the Assessment and returns a result.\n *\n * @param {Paper} paper The Paper object to assess.\n * @param {Researcher} researcher The Researcher object containing all available researches.\n *\n * @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.\n */\n getResult({paper, researcher}) {\n const calculatedResult = this.calculateResult(paper);\n\n const assessmentResult = new AssessmentResult({config: this._config});\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n assessmentResult.setData(calculatedResult.data);\n\n return assessmentResult;\n }\n\n /**\n *\n * @param paper\n * @returns {{score: number, status: string, data: {hasSchema: ((function(): *)|boolean), hasProperStructure: boolean, hasMinimumFAQs: boolean}}|{score: number, status: string, data: {hasFAQs: boolean}}}\n */\n calculateResult(paper) {\n const faqs = paper.getFAQs()?.questions || [];\n let status = 'bad';\n let score = 0;\n\n if (!faqs || faqs.length === 0) {\n return {score: 0, status: 'bad', data: {hasFAQs: false}};\n }\n const isEnableFaqs = paper.getFAQs()?.status;\n const hasMinimumFAQs = this.checkMinimumFAQs(faqs);\n status = hasMinimumFAQs && isEnableFaqs ? 'good' : 'bad';\n score = this.getScore(FAQS_POINTS, status);\n\n return {\n score,\n status\n };\n }\n\n\n\n /**\n * Checks if there are at least 3 FAQs\n * @param {Array} faqs\n * @returns {boolean}\n */\n checkMinimumFAQs(faqs) {\n return faqs.length >= 3;\n }\n\n /**\n * Checks whether the paper has text.\n *\n * @param {Paper} paper The paper to use for the assessment.\n *\n * @returns {boolean} True when there is text.\n */\n isApplicable(paper) {\n return true;\n }\n}\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAC5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,kBAAkB,EAAEC,WAAW;;AAEvC;AACA;AACA;AACA,eAAe,MAAMC,cAAc,SAASJ,UAAU,CAAC;EACrD;AACF;AACA;AACA;AACA;AACA;AACA;EACEK,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAE;IACvB,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEN,kBAAkB;MACtBO,OAAO,EAAE,KAAK;MACdC,MAAM,EAAE,2DAA2D;MACnEC,QAAQ,EAAE,MAAM;MAChBC,WAAW,EAAE,MAAM;MACnBC,KAAK,EAAE,mBAAmB;MAC1BC,OAAO,EAAE;QACPC,IAAI,EAAE,wDAAwD;QAC9DC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE,iEAAiE;QACtEC,OAAO,EAAE;UACPC,kBAAkB,EAAE,4BAA4B;UAChDC,cAAc,EAAE;QAClB;MACF;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGnB,kBAAkB;IACpC,IAAI,CAACoB,OAAO,GAAGvB,KAAK,CAACQ,aAAa,EAAED,MAAM,CAAC;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEiB,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B,MAAMC,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAACH,KAAK,CAAC;IAEpD,MAAMI,gBAAgB,GAAG,IAAI3B,gBAAgB,CAAC;MAACK,MAAM,EAAE,IAAI,CAACgB;IAAO,CAAC,CAAC;IACrEM,gBAAgB,CAACC,QAAQ,CAACH,gBAAgB,CAACI,KAAK,CAAC;IACjDF,gBAAgB,CAACG,SAAS,CAACL,gBAAgB,CAACM,MAAM,CAAC;IACnDJ,gBAAgB,CAACK,OAAO,CAACP,gBAAgB,CAACQ,IAAI,CAAC;IAE/C,OAAON,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;EACED,eAAeA,CAACH,KAAK,EAAE;IACrB,MAAMW,IAAI,GAAGX,KAAK,CAACY,OAAO,CAAC,CAAC,EAAEC,SAAS,IAAI,EAAE;IAC7C,IAAIL,MAAM,GAAG,KAAK;IAClB,IAAIF,KAAK,GAAG,CAAC;IAEb,IAAI,CAACK,IAAI,IAAIA,IAAI,CAACG,MAAM,KAAK,CAAC,EAAE;MAC9B,OAAO;QAACR,KAAK,EAAE,CAAC;QAAEE,MAAM,EAAE,KAAK;QAAEE,IAAI,EAAE;UAACK,OAAO,EAAE;QAAK;MAAC,CAAC;IAC1D;IACA,MAAMC,YAAY,GAAGhB,KAAK,CAACY,OAAO,CAAC,CAAC,EAAEJ,MAAM;IAC5C,MAAMZ,cAAc,GAAG,IAAI,CAACqB,gBAAgB,CAACN,IAAI,CAAC;IAClDH,MAAM,GAAGZ,cAAc,IAAIoB,YAAY,GAAG,MAAM,GAAG,KAAK;IACxDV,KAAK,GAAG,IAAI,CAACY,QAAQ,CAACvC,WAAW,EAAE6B,MAAM,CAAC;IAE1C,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;;EAIA;AACF;AACA;AACA;AACA;EACES,gBAAgBA,CAACN,IAAI,EAAE;IACrB,OAAOA,IAAI,CAACG,MAAM,IAAI,CAAC;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEK,YAAYA,CAACnB,KAAK,EAAE;IAClB,OAAO,IAAI;EACb;AACF","ignoreList":[]}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { merge } from 'lodash';
|
|
2
|
+
import Assessment from "../assessment";
|
|
3
|
+
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
|
+
import { FAQS_POINTS, FAQs_STRUCTURE_DATA_ID } from "../../../const/analysis";
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Represents an assessment that checks FAQs content and structure for SGE optimization.
|
|
8
|
+
*/
|
|
9
|
+
export default class FAQsStructureDataAssessment extends Assessment {
|
|
10
|
+
/**
|
|
11
|
+
* Sets the identifier and the config.
|
|
12
|
+
*
|
|
13
|
+
* @param {Object} [config] The configuration to use.
|
|
14
|
+
*
|
|
15
|
+
* @returns {void}
|
|
16
|
+
*/
|
|
17
|
+
constructor(config = {}) {
|
|
18
|
+
super();
|
|
19
|
+
const defaultConfig = {
|
|
20
|
+
id: FAQs_STRUCTURE_DATA_ID,
|
|
21
|
+
ctaType: 'fix',
|
|
22
|
+
docUrl: 'https://blog.google/products/search/generative-ai-search/',
|
|
23
|
+
priority: 'high',
|
|
24
|
+
fixPosition: 'faqs',
|
|
25
|
+
title: 'FAQs structured data',
|
|
26
|
+
content: {
|
|
27
|
+
good: 'FAQ structured data added',
|
|
28
|
+
improve: '',
|
|
29
|
+
bad: 'Add FAQs structured data for Google rich results eligibility.'
|
|
30
|
+
}
|
|
31
|
+
};
|
|
32
|
+
this.identifier = FAQs_STRUCTURE_DATA_ID;
|
|
33
|
+
this._config = merge(defaultConfig, config);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Executes the Assessment and returns a result.
|
|
38
|
+
*
|
|
39
|
+
* @param {Paper} paper The Paper object to assess.
|
|
40
|
+
* @param {Researcher} researcher The Researcher object containing all available researches.
|
|
41
|
+
*
|
|
42
|
+
* @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.
|
|
43
|
+
*/
|
|
44
|
+
getResult({
|
|
45
|
+
paper,
|
|
46
|
+
researcher
|
|
47
|
+
}) {
|
|
48
|
+
const calculatedResult = this.calculateResult(paper);
|
|
49
|
+
const assessmentResult = new AssessmentResult({
|
|
50
|
+
config: this._config
|
|
51
|
+
});
|
|
52
|
+
assessmentResult.setScore(calculatedResult.score);
|
|
53
|
+
assessmentResult.setStatus(calculatedResult.status);
|
|
54
|
+
return assessmentResult;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
*
|
|
59
|
+
* @param paper
|
|
60
|
+
* @returns {{score: number, status: string, data: {hasSchema: ((function(): *)|boolean), hasProperStructure: boolean, hasMinimumFAQs: boolean}}|{score: number, status: string, data: {hasFAQs: boolean}}}
|
|
61
|
+
*/
|
|
62
|
+
calculateResult(paper) {
|
|
63
|
+
// Check FAQ structure and content
|
|
64
|
+
const hasSchema = paper.hasFAQsSchema();
|
|
65
|
+
const status = hasSchema ? 'good' : 'bad';
|
|
66
|
+
const score = this.getScore(FAQS_POINTS, status);
|
|
67
|
+
return {
|
|
68
|
+
score,
|
|
69
|
+
status
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
isApplicable(paper) {
|
|
73
|
+
return true;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=FAQsStructureDataAssessment.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"FAQsStructureDataAssessment.js","names":["merge","Assessment","AssessmentResult","FAQS_POINTS","FAQs_STRUCTURE_DATA_ID","FAQsStructureDataAssessment","constructor","config","defaultConfig","id","ctaType","docUrl","priority","fixPosition","title","content","good","improve","bad","identifier","_config","getResult","paper","researcher","calculatedResult","calculateResult","assessmentResult","setScore","score","setStatus","status","hasSchema","hasFAQsSchema","getScore","isApplicable"],"sources":["../../../../../src/scoring/assessments/seo/FAQsStructureDataAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {FAQS_POINTS, FAQs_STRUCTURE_DATA_ID} from '@axyseo/const/analysis';\n\n/**\n * Represents an assessment that checks FAQs content and structure for SGE optimization.\n */\nexport default class FAQsStructureDataAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} [config] The configuration to use.\n *\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: FAQs_STRUCTURE_DATA_ID,\n ctaType: 'fix',\n docUrl: 'https://blog.google/products/search/generative-ai-search/',\n priority: 'high',\n fixPosition: 'faqs',\n title: 'FAQs structured data',\n content: {\n good: 'FAQ structured data added',\n improve: '',\n bad: 'Add FAQs structured data for Google rich results eligibility.',\n }\n };\n\n this.identifier = FAQs_STRUCTURE_DATA_ID;\n this._config = merge(defaultConfig, config);\n }\n\n /**\n * Executes the Assessment and returns a result.\n *\n * @param {Paper} paper The Paper object to assess.\n * @param {Researcher} researcher The Researcher object containing all available researches.\n *\n * @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.\n */\n getResult({paper, researcher}) {\n const calculatedResult = this.calculateResult(paper);\n const assessmentResult = new AssessmentResult({config: this._config});\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n return assessmentResult;\n }\n\n /**\n *\n * @param paper\n * @returns {{score: number, status: string, data: {hasSchema: ((function(): *)|boolean), hasProperStructure: boolean, hasMinimumFAQs: boolean}}|{score: number, status: string, data: {hasFAQs: boolean}}}\n */\n calculateResult(paper) {\n // Check FAQ structure and content\n const hasSchema = paper.hasFAQsSchema();\n const status = hasSchema ? 'good' : 'bad';\n const score = this.getScore(FAQS_POINTS, status);\n return {\n score,\n status\n };\n }\n\n\n\n\n isApplicable(paper) {\n return true;\n }\n}\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAC5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,WAAW,EAAEC,sBAAsB;;AAE3C;AACA;AACA;AACA,eAAe,MAAMC,2BAA2B,SAASJ,UAAU,CAAC;EAClE;AACF;AACA;AACA;AACA;AACA;AACA;EACEK,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAE;IACvB,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEL,sBAAsB;MAC1BM,OAAO,EAAE,KAAK;MACdC,MAAM,EAAE,2DAA2D;MACnEC,QAAQ,EAAE,MAAM;MAChBC,WAAW,EAAE,MAAM;MACnBC,KAAK,EAAE,sBAAsB;MAC7BC,OAAO,EAAE;QACPC,IAAI,EAAE,2BAA2B;QACjCC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE;MACP;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGf,sBAAsB;IACxC,IAAI,CAACgB,OAAO,GAAGpB,KAAK,CAACQ,aAAa,EAAED,MAAM,CAAC;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEc,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B,MAAMC,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAACH,KAAK,CAAC;IACpD,MAAMI,gBAAgB,GAAG,IAAIxB,gBAAgB,CAAC;MAACK,MAAM,EAAE,IAAI,CAACa;IAAO,CAAC,CAAC;IACrEM,gBAAgB,CAACC,QAAQ,CAACH,gBAAgB,CAACI,KAAK,CAAC;IACjDF,gBAAgB,CAACG,SAAS,CAACL,gBAAgB,CAACM,MAAM,CAAC;IACnD,OAAOJ,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;EACED,eAAeA,CAACH,KAAK,EAAE;IACrB;IACA,MAAMS,SAAS,GAAGT,KAAK,CAACU,aAAa,CAAC,CAAC;IACvC,MAAMF,MAAM,GAAGC,SAAS,GAAG,MAAM,GAAG,KAAK;IAC1C,MAAMH,KAAK,GAAG,IAAI,CAACK,QAAQ,CAAC9B,WAAW,EAAE2B,MAAM,CAAC;IAC/C,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;EAKAI,YAAYA,CAACZ,KAAK,EAAE;IAClB,OAAO,IAAI;EACb;AACF","ignoreList":[]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { merge } from 'lodash';
|
|
2
2
|
import Assessment from "../assessment";
|
|
3
3
|
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
|
-
import { MEDIA_ID } from "../../../const/analysis";
|
|
4
|
+
import { FAQS_POINTS, MEDIA_ID } from "../../../const/analysis";
|
|
5
5
|
|
|
6
6
|
/**
|
|
7
7
|
* Represents the assessment that checks if the text has any images present, including videos in product pages.
|
|
@@ -64,7 +64,7 @@ export default class TextImagesAssessment extends Assessment {
|
|
|
64
64
|
* @returns {boolean} True when there is text.
|
|
65
65
|
*/
|
|
66
66
|
isApplicable(paper) {
|
|
67
|
-
return
|
|
67
|
+
return true;
|
|
68
68
|
}
|
|
69
69
|
|
|
70
70
|
/**
|
|
@@ -81,7 +81,7 @@ export default class TextImagesAssessment extends Assessment {
|
|
|
81
81
|
} else {
|
|
82
82
|
status = 'good';
|
|
83
83
|
}
|
|
84
|
-
const score = this.getScore(
|
|
84
|
+
const score = this.getScore(FAQS_POINTS, status);
|
|
85
85
|
return {
|
|
86
86
|
score,
|
|
87
87
|
status
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ImageCountAssessment.js","names":["merge","Assessment","AssessmentResult","MEDIA_ID","TextImagesAssessment","constructor","config","countVideos","defaultConfig","id","docUrl","priority","fixPosition","title","content","good","improve","bad","identifier","_config","_countVideos","getResult","paper","researcher","imageCount","getResearch","videoCount","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","
|
|
1
|
+
{"version":3,"file":"ImageCountAssessment.js","names":["merge","Assessment","AssessmentResult","FAQS_POINTS","MEDIA_ID","TextImagesAssessment","constructor","config","countVideos","defaultConfig","id","docUrl","priority","fixPosition","title","content","good","improve","bad","identifier","_config","_countVideos","getResult","paper","researcher","imageCount","getResearch","videoCount","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","mediaCount","getScore"],"sources":["../../../../../src/scoring/assessments/seo/ImageCountAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {FAQS_POINTS, MEDIA_ID} from '@axyseo/const/analysis';\n\n/**\n * Represents the assessment that checks if the text has any images present, including videos in product pages.\n */\nexport default class TextImagesAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {object} config The configuration to use.\n * @param {boolean} countVideos Whether videos are also included in the assessment or not.\n *\n * @returns {void}\n */\n constructor(config = {}, countVideos = true) {\n super();\n\n const defaultConfig = {\n id: MEDIA_ID,\n docUrl:\n 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#images-and-videos',\n priority: 'high',\n fixPosition: 'description',\n title: 'Media',\n content: {\n good: 'Media are used effectively.',\n improve: '',\n bad: 'No media found. Add visual content to enhance engagement.'\n }\n };\n\n this.identifier = MEDIA_ID;\n this._config = merge(defaultConfig, config);\n this._countVideos = countVideos;\n }\n\n /**\n * Execute the Assessment and return a result.\n *\n * @param {Paper} paper The Paper object to assess.\n * @param {Researcher} researcher The Researcher object containing all available researches.\n * @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.\n */\n getResult({paper, researcher}) {\n this.imageCount = researcher.getResearch('imageCount');\n this.videoCount = researcher.getResearch('videoCount');\n\n const assessmentResult = new AssessmentResult({config: this._config});\n\n const calculatedResult = this.calculateResult();\n\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n\n return assessmentResult;\n }\n\n /**\n * Checks whether the paper has text.\n *\n * @param {Paper} paper The paper to use for the assessment.\n *\n * @returns {boolean} True when there is text.\n */\n isApplicable(paper) {\n return true;\n }\n\n /**\n *\n * @returns {{score: number, status: string}}\n */\n calculateResult() {\n // If \"countVideos\" is on, we include videos in the assessment\n const mediaCount = this._countVideos ? this.imageCount + this.videoCount : this.imageCount;\n\n let status = '';\n // No images.\n if (mediaCount === 0) {\n status = 'bad';\n } else {\n status = 'good';\n }\n\n const score = this.getScore(FAQS_POINTS, status);\n\n return {\n score,\n status\n };\n }\n}\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAC5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,WAAW,EAAEC,QAAQ;;AAE7B;AACA;AACA;AACA,eAAe,MAAMC,oBAAoB,SAASJ,UAAU,CAAC;EAC3D;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEK,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAEC,WAAW,GAAG,IAAI,EAAE;IAC3C,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEN,QAAQ;MACZO,MAAM,EACJ,+FAA+F;MACjGC,QAAQ,EAAE,MAAM;MAChBC,WAAW,EAAE,aAAa;MAC1BC,KAAK,EAAE,OAAO;MACdC,OAAO,EAAE;QACPC,IAAI,EAAE,6BAA6B;QACnCC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE;MACP;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGf,QAAQ;IAC1B,IAAI,CAACgB,OAAO,GAAGpB,KAAK,CAACS,aAAa,EAAEF,MAAM,CAAC;IAC3C,IAAI,CAACc,YAAY,GAAGb,WAAW;EACjC;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEc,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B,IAAI,CAACC,UAAU,GAAGD,UAAU,CAACE,WAAW,CAAC,YAAY,CAAC;IACtD,IAAI,CAACC,UAAU,GAAGH,UAAU,CAACE,WAAW,CAAC,YAAY,CAAC;IAEtD,MAAME,gBAAgB,GAAG,IAAI1B,gBAAgB,CAAC;MAACK,MAAM,EAAE,IAAI,CAACa;IAAO,CAAC,CAAC;IAErE,MAAMS,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAAC,CAAC;IAE/CF,gBAAgB,CAACG,QAAQ,CAACF,gBAAgB,CAACG,KAAK,CAAC;IACjDJ,gBAAgB,CAACK,SAAS,CAACJ,gBAAgB,CAACK,MAAM,CAAC;IAEnD,OAAON,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEO,YAAYA,CAACZ,KAAK,EAAE;IAClB,OAAO,IAAI;EACb;;EAEA;AACF;AACA;AACA;EACEO,eAAeA,CAAA,EAAG;IAChB;IACA,MAAMM,UAAU,GAAG,IAAI,CAACf,YAAY,GAAG,IAAI,CAACI,UAAU,GAAG,IAAI,CAACE,UAAU,GAAG,IAAI,CAACF,UAAU;IAE1F,IAAIS,MAAM,GAAG,EAAE;IACf;IACA,IAAIE,UAAU,KAAK,CAAC,EAAE;MACpBF,MAAM,GAAG,KAAK;IAChB,CAAC,MAAM;MACLA,MAAM,GAAG,MAAM;IACjB;IAEA,MAAMF,KAAK,GAAG,IAAI,CAACK,QAAQ,CAAClC,WAAW,EAAE+B,MAAM,CAAC;IAEhD,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;AACF","ignoreList":[]}
|
|
@@ -2,7 +2,7 @@ import { merge } from 'lodash';
|
|
|
2
2
|
import Assessment from "../assessment";
|
|
3
3
|
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
4
|
import { LinkIcon } from '@shopify/polaris-icons';
|
|
5
|
-
import { INTERNAL_LINKS_ID } from "../../../const/analysis";
|
|
5
|
+
import { INTERNAL_LINKS_ID, LINK_RECOMMEND_POINTS } from "../../../const/analysis";
|
|
6
6
|
|
|
7
7
|
/**
|
|
8
8
|
* Assessment to check whether the text has internal links and whether they are followed or no-followed.
|
|
@@ -69,7 +69,7 @@ class InternalLinksAssessment extends Assessment {
|
|
|
69
69
|
* @returns {boolean} Whether the paper has text.
|
|
70
70
|
*/
|
|
71
71
|
isApplicable(paper) {
|
|
72
|
-
return
|
|
72
|
+
return true;
|
|
73
73
|
}
|
|
74
74
|
|
|
75
75
|
/**
|
|
@@ -77,13 +77,11 @@ class InternalLinksAssessment extends Assessment {
|
|
|
77
77
|
* @returns {{score: number, status: string}}
|
|
78
78
|
*/
|
|
79
79
|
calculateResult() {
|
|
80
|
-
let status = '';
|
|
80
|
+
let status = 'good';
|
|
81
81
|
if (this.linkStatistics.internalTotal === 0) {
|
|
82
82
|
status = 'bad';
|
|
83
|
-
} else {
|
|
84
|
-
status = 'good';
|
|
85
83
|
}
|
|
86
|
-
const score = this.getScore(
|
|
84
|
+
const score = this.getScore(LINK_RECOMMEND_POINTS, status);
|
|
87
85
|
return {
|
|
88
86
|
score,
|
|
89
87
|
status
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"InternalLinksAssessment.js","names":["merge","Assessment","AssessmentResult","LinkIcon","INTERNAL_LINKS_ID","InternalLinksAssessment","constructor","config","defaultConfig","id","fixPosition","docUrl","icon","priority","title","content","good","improve","bad","identifier","_config","getResult","paper","researcher","linkStatistics","getResearch","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","
|
|
1
|
+
{"version":3,"file":"InternalLinksAssessment.js","names":["merge","Assessment","AssessmentResult","LinkIcon","INTERNAL_LINKS_ID","LINK_RECOMMEND_POINTS","InternalLinksAssessment","constructor","config","defaultConfig","id","fixPosition","docUrl","icon","priority","title","content","good","improve","bad","identifier","_config","getResult","paper","researcher","linkStatistics","getResearch","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","internalTotal","getScore"],"sources":["../../../../../src/scoring/assessments/seo/InternalLinksAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {LinkIcon} from '@shopify/polaris-icons';\nimport {INTERNAL_LINKS_ID, LINK_RECOMMEND_POINTS} from '@axyseo/const/analysis';\n\n/**\n * Assessment to check whether the text has internal links and whether they are followed or no-followed.\n */\nclass InternalLinksAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} [config] The configuration to use.\n * @param {number} [config.parameters.recommendedMinimum] The recommended minimum number of internal links in the text.\n * @param {number} [config.scores.allInternalFollow] The score to return if all internal links are do-follow.\n * @param {number} [config.scores.someInternalFollow] The score to return if some but not all internal links are do-follow.\n * @param {number} [config.scores.noneInternalFollow] The score to return if all internal links are no-follow.\n * @param {number} [config.scores.noInternal] The score to return if there are no internal links.\n * @param {string} [config.url] The URL to the relevant KB article.\n *\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: INTERNAL_LINKS_ID,\n fixPosition: '',\n docUrl:\n 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#internal-links-outbound-links',\n icon: LinkIcon,\n priority: 'high',\n title: 'Internal links',\n content: {\n good: 'Internal links are used effectively.',\n improve: '',\n bad: 'No internal links found. Add links to other relevant pages on your website.'\n }\n };\n\n this.identifier = INTERNAL_LINKS_ID;\n this._config = merge(defaultConfig, config);\n }\n\n /**\n * Runs the getLinkStatistics module, based on this returns an assessment result with score.\n *\n * @param {Paper} paper The paper to use for the assessment.\n * @param {Researcher} researcher The researcher used for calling research.\n * @returns {AssessmentResult} The result of the assessment.\n */\n getResult({paper, researcher}) {\n this.linkStatistics = researcher.getResearch('getLinkStatistics');\n const assessmentResult = new AssessmentResult({config: this._config});\n\n const calculatedResult = this.calculateResult();\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n\n return assessmentResult;\n }\n\n /**\n * Checks if assessment is applicable to the paper.\n *\n * @param {Paper} paper The paper to be analyzed.\n *\n * @returns {boolean} Whether the paper has text.\n */\n isApplicable(paper) {\n return true;\n }\n\n /**\n *\n * @returns {{score: number, status: string}}\n */\n calculateResult() {\n let status = 'good';\n if (this.linkStatistics.internalTotal === 0) {\n status = 'bad';\n }\n\n const score = this.getScore(LINK_RECOMMEND_POINTS, status);\n\n return {\n score,\n status\n };\n }\n}\n\nexport default InternalLinksAssessment;\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAC5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,QAAQ,QAAO,wBAAwB;AAC/C,SAAQC,iBAAiB,EAAEC,qBAAqB;;AAEhD;AACA;AACA;AACA,MAAMC,uBAAuB,SAASL,UAAU,CAAC;EAC/C;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACEM,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAE;IACvB,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEN,iBAAiB;MACrBO,WAAW,EAAE,EAAE;MACfC,MAAM,EACJ,2GAA2G;MAC7GC,IAAI,EAAEV,QAAQ;MACdW,QAAQ,EAAE,MAAM;MAChBC,KAAK,EAAE,gBAAgB;MACvBC,OAAO,EAAE;QACPC,IAAI,EAAE,sCAAsC;QAC5CC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE;MACP;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGhB,iBAAiB;IACnC,IAAI,CAACiB,OAAO,GAAGrB,KAAK,CAACS,aAAa,EAAED,MAAM,CAAC;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEc,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B,IAAI,CAACC,cAAc,GAAGD,UAAU,CAACE,WAAW,CAAC,mBAAmB,CAAC;IACjE,MAAMC,gBAAgB,GAAG,IAAIzB,gBAAgB,CAAC;MAACM,MAAM,EAAE,IAAI,CAACa;IAAO,CAAC,CAAC;IAErE,MAAMO,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAAC,CAAC;IAC/CF,gBAAgB,CAACG,QAAQ,CAACF,gBAAgB,CAACG,KAAK,CAAC;IACjDJ,gBAAgB,CAACK,SAAS,CAACJ,gBAAgB,CAACK,MAAM,CAAC;IAEnD,OAAON,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEO,YAAYA,CAACX,KAAK,EAAE;IAClB,OAAO,IAAI;EACb;;EAEA;AACF;AACA;AACA;EACEM,eAAeA,CAAA,EAAG;IAChB,IAAII,MAAM,GAAG,MAAM;IACnB,IAAI,IAAI,CAACR,cAAc,CAACU,aAAa,KAAK,CAAC,EAAE;MAC3CF,MAAM,GAAG,KAAK;IAChB;IAEA,MAAMF,KAAK,GAAG,IAAI,CAACK,QAAQ,CAAC/B,qBAAqB,EAAE4B,MAAM,CAAC;IAE1D,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;AACF;AAEA,eAAe3B,uBAAuB","ignoreList":[]}
|
|
@@ -2,7 +2,7 @@ import { merge } from 'lodash';
|
|
|
2
2
|
import Assessment from "../assessment";
|
|
3
3
|
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
4
|
import { KeyIcon } from '@shopify/polaris-icons';
|
|
5
|
-
import { INTRODUCTION_KEYWORD_ID } from "../../../const/analysis";
|
|
5
|
+
import { INTRODUCTION_KEYWORD_ID, MAIN_CONTENT_POINTS } from "../../../const/analysis";
|
|
6
6
|
import { enhanceResearcherForChinese } from "../../../languageProcessing/helpers/language/chineseHelperFactory.js";
|
|
7
7
|
|
|
8
8
|
/**
|
|
@@ -70,7 +70,7 @@ class IntroductionKeywordAssessment extends Assessment {
|
|
|
70
70
|
* @returns {boolean} Whether the assessment is applicable or not.
|
|
71
71
|
*/
|
|
72
72
|
isApplicable(paper) {
|
|
73
|
-
return
|
|
73
|
+
return true;
|
|
74
74
|
}
|
|
75
75
|
|
|
76
76
|
/**
|
|
@@ -81,10 +81,8 @@ class IntroductionKeywordAssessment extends Assessment {
|
|
|
81
81
|
let status = 'bad';
|
|
82
82
|
if (this._firstParagraphMatches.foundInOneSentence || this._firstParagraphMatches.foundInParagraph) {
|
|
83
83
|
status = 'good';
|
|
84
|
-
} else {
|
|
85
|
-
status = 'bad';
|
|
86
84
|
}
|
|
87
|
-
const score = this.getScore(
|
|
85
|
+
const score = this.getScore(MAIN_CONTENT_POINTS, status);
|
|
88
86
|
return {
|
|
89
87
|
score,
|
|
90
88
|
status
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"IntroductionKeywordAssessment.js","names":["merge","Assessment","AssessmentResult","KeyIcon","INTRODUCTION_KEYWORD_ID","enhanceResearcherForChinese","IntroductionKeywordAssessment","constructor","config","defaultConfig","id","ctaType","priority","docUrl","fixPosition","icon","title","content","improve","bad","good","identifier","_config","getResult","paper","researcher","assessmentResult","enhancedResearcher","getText","getKeyword","_firstParagraphMatches","getResearch","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","
|
|
1
|
+
{"version":3,"file":"IntroductionKeywordAssessment.js","names":["merge","Assessment","AssessmentResult","KeyIcon","INTRODUCTION_KEYWORD_ID","MAIN_CONTENT_POINTS","enhanceResearcherForChinese","IntroductionKeywordAssessment","constructor","config","defaultConfig","id","ctaType","priority","docUrl","fixPosition","icon","title","content","improve","bad","good","identifier","_config","getResult","paper","researcher","assessmentResult","enhancedResearcher","getText","getKeyword","_firstParagraphMatches","getResearch","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","foundInOneSentence","foundInParagraph","getScore"],"sources":["../../../../../src/scoring/assessments/seo/IntroductionKeywordAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {KeyIcon} from '@shopify/polaris-icons';\nimport {INTRODUCTION_KEYWORD_ID, MAIN_CONTENT_POINTS} from '@axyseo/const/analysis';\nimport {enhanceResearcherForChinese} from '../../../languageProcessing/helpers/language/chineseHelperFactory.js';\n\n/**\n * Assessment to check whether the keyphrase or synonyms are encountered in the first paragraph of the article.\n */\nclass IntroductionKeywordAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} [config] The configuration to use.\n * @param {string} [config.url] The URL to the relevant article on Yoast.com.\n *\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: INTRODUCTION_KEYWORD_ID,\n ctaType: 'fix',\n priority: 'low',\n docUrl:\n 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#keyword-in-introduction',\n fixPosition: 'highlightFirstParagraph',\n icon: KeyIcon,\n title: 'Introduction Keyword',\n content: {\n improve: '',\n bad: 'No keywords found in the first sentence. Add one to improve keyword rank.',\n good: 'Keyword is included in the first sentence.'\n }\n };\n\n this.identifier = INTRODUCTION_KEYWORD_ID;\n this._config = merge(defaultConfig, config);\n }\n\n /**\n * Assesses the presence of keyphrase or synonyms in the first paragraph.\n *\n * @param {Paper} paper The paper to use for the assessment.\n * @param {Researcher} researcher The researcher used for calling research.\n *\n * @returns {AssessmentResult} The result of this assessment.\n */\n getResult({paper, researcher}) {\n const assessmentResult = new AssessmentResult({config: this._config});\n\n // Enhance researcher for Chinese if needed\n const enhancedResearcher = enhanceResearcherForChinese(\n researcher,\n paper.getText(),\n paper.getKeyword()\n );\n\n this._firstParagraphMatches = enhancedResearcher.getResearch('findKeywordInFirstParagraph');\n const calculatedResult = this.calculateResult();\n\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n\n return assessmentResult;\n }\n\n /**\n * Checks if the paper has both keyword and text.\n *\n * @param {Paper} paper The paper to be analyzed.\n *\n * @returns {boolean} Whether the assessment is applicable or not.\n */\n isApplicable(paper) {\n return true;\n }\n\n /**\n *\n * @returns {{score: number, status: string}}\n */\n calculateResult() {\n let status = 'bad';\n\n if (\n this._firstParagraphMatches.foundInOneSentence ||\n this._firstParagraphMatches.foundInParagraph\n ) {\n status = 'good';\n }\n\n const score = this.getScore(MAIN_CONTENT_POINTS, status);\n\n return {\n score,\n status\n };\n }\n}\n\nexport default IntroductionKeywordAssessment;\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAC5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,OAAO,QAAO,wBAAwB;AAC9C,SAAQC,uBAAuB,EAAEC,mBAAmB;AACpD,SAAQC,2BAA2B;;AAEnC;AACA;AACA;AACA,MAAMC,6BAA6B,SAASN,UAAU,CAAC;EACrD;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEO,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAE;IACvB,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEP,uBAAuB;MAC3BQ,OAAO,EAAE,KAAK;MACdC,QAAQ,EAAE,KAAK;MACfC,MAAM,EACJ,qGAAqG;MACvGC,WAAW,EAAE,yBAAyB;MACtCC,IAAI,EAAEb,OAAO;MACbc,KAAK,EAAE,sBAAsB;MAC7BC,OAAO,EAAE;QACPC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE,2EAA2E;QAChFC,IAAI,EAAE;MACR;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGlB,uBAAuB;IACzC,IAAI,CAACmB,OAAO,GAAGvB,KAAK,CAACU,aAAa,EAAED,MAAM,CAAC;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEe,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B,MAAMC,gBAAgB,GAAG,IAAIzB,gBAAgB,CAAC;MAACO,MAAM,EAAE,IAAI,CAACc;IAAO,CAAC,CAAC;;IAErE;IACA,MAAMK,kBAAkB,GAAGtB,2BAA2B,CACpDoB,UAAU,EACVD,KAAK,CAACI,OAAO,CAAC,CAAC,EACfJ,KAAK,CAACK,UAAU,CAAC,CACnB,CAAC;IAED,IAAI,CAACC,sBAAsB,GAAGH,kBAAkB,CAACI,WAAW,CAAC,6BAA6B,CAAC;IAC3F,MAAMC,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAAC,CAAC;IAE/CP,gBAAgB,CAACQ,QAAQ,CAACF,gBAAgB,CAACG,KAAK,CAAC;IACjDT,gBAAgB,CAACU,SAAS,CAACJ,gBAAgB,CAACK,MAAM,CAAC;IAEnD,OAAOX,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEY,YAAYA,CAACd,KAAK,EAAE;IAClB,OAAO,IAAI;EACb;;EAEA;AACF;AACA;AACA;EACES,eAAeA,CAAA,EAAG;IAChB,IAAII,MAAM,GAAG,KAAK;IAElB,IACE,IAAI,CAACP,sBAAsB,CAACS,kBAAkB,IAC9C,IAAI,CAACT,sBAAsB,CAACU,gBAAgB,EAC5C;MACAH,MAAM,GAAG,MAAM;IACjB;IAEA,MAAMF,KAAK,GAAG,IAAI,CAACM,QAAQ,CAACrC,mBAAmB,EAAEiC,MAAM,CAAC;IAExD,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;AACF;AAEA,eAAe/B,6BAA6B","ignoreList":[]}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { merge } from 'lodash';
|
|
2
|
+
import Assessment from "../assessment";
|
|
3
|
+
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
|
+
import { CHARACTERS_URL, URLS_POINTS } from "../../../const/analysis";
|
|
5
|
+
import { enhanceResearcherForChinese } from "../../../languageProcessing/helpers/language/chineseHelperFactory";
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Represents the Slug keyword assessment. This assessment checks if the keyword is present in the slug.
|
|
9
|
+
*/
|
|
10
|
+
class InvalidCharactersInUrl extends Assessment {
|
|
11
|
+
/**
|
|
12
|
+
* Sets the identifier and the config.
|
|
13
|
+
*
|
|
14
|
+
* @param {Object} config The configuration to use.
|
|
15
|
+
* @returns {void}
|
|
16
|
+
*/
|
|
17
|
+
constructor(config = {}) {
|
|
18
|
+
super();
|
|
19
|
+
const defaultConfig = {
|
|
20
|
+
id: CHARACTERS_URL,
|
|
21
|
+
docUrl: 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#keyword-in-url',
|
|
22
|
+
fixPosition: 'url handle',
|
|
23
|
+
ctaType: 'fix',
|
|
24
|
+
priority: 'medium',
|
|
25
|
+
title: 'Invalid characters in URL',
|
|
26
|
+
content: {
|
|
27
|
+
good: 'URL characters valid',
|
|
28
|
+
improve: '',
|
|
29
|
+
bad: 'Use only lowercase letters, numbers, and hyphens.'
|
|
30
|
+
}
|
|
31
|
+
};
|
|
32
|
+
this.identifier = CHARACTERS_URL;
|
|
33
|
+
this._config = merge(defaultConfig, config);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Executes the Assessment and returns a result.
|
|
38
|
+
*
|
|
39
|
+
* @param {Paper} paper The Paper object to assess.
|
|
40
|
+
* @param {Researcher} researcher The Researcher object containing all available researches.
|
|
41
|
+
*
|
|
42
|
+
* @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.
|
|
43
|
+
*/
|
|
44
|
+
getResult({
|
|
45
|
+
paper,
|
|
46
|
+
researcher
|
|
47
|
+
}) {
|
|
48
|
+
// Enhance researcher for Chinese if needed - specifically for URL context
|
|
49
|
+
const enhancedResearcher = enhanceResearcherForChinese(researcher, paper.getSlug(), paper.getKeyword(), true // isUrlContext = true for enhanced Chinese URL matching
|
|
50
|
+
);
|
|
51
|
+
this._charactersInUrl = enhancedResearcher.getResearch('charactersInUrl');
|
|
52
|
+
const assessmentResult = new AssessmentResult({
|
|
53
|
+
config: this._config
|
|
54
|
+
});
|
|
55
|
+
const calculatedResult = this.calculateResult();
|
|
56
|
+
assessmentResult.setScore(calculatedResult.score);
|
|
57
|
+
assessmentResult.setStatus(calculatedResult.status);
|
|
58
|
+
return assessmentResult;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Checks whether the paper has a keyword and a slug.
|
|
63
|
+
*
|
|
64
|
+
* @param {Paper} paper The paper to use for the assessment.
|
|
65
|
+
* @param {Researcher} researcher The researcher object.
|
|
66
|
+
*
|
|
67
|
+
* @returns {boolean} True if the paper contains a keyword and a slug, and if the keywordCountInSlug research is available on the researcher.
|
|
68
|
+
*/
|
|
69
|
+
isApplicable(paper, researcher) {
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
*
|
|
75
|
+
* @returns {{score: number, status: (string)}}
|
|
76
|
+
*/
|
|
77
|
+
calculateResult() {
|
|
78
|
+
const status = this._charactersInUrl.isInvalidCharacters ? 'bad' : 'good';
|
|
79
|
+
const score = this.getScore(URLS_POINTS, status);
|
|
80
|
+
return {
|
|
81
|
+
score,
|
|
82
|
+
status
|
|
83
|
+
};
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
export default InvalidCharactersInUrl;
|
|
87
|
+
//# sourceMappingURL=InvalidCharactersInUrl.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"InvalidCharactersInUrl.js","names":["merge","Assessment","AssessmentResult","CHARACTERS_URL","URLS_POINTS","enhanceResearcherForChinese","InvalidCharactersInUrl","constructor","config","defaultConfig","id","docUrl","fixPosition","ctaType","priority","title","content","good","improve","bad","identifier","_config","getResult","paper","researcher","enhancedResearcher","getSlug","getKeyword","_charactersInUrl","getResearch","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","isInvalidCharacters","getScore"],"sources":["../../../../../src/scoring/assessments/seo/InvalidCharactersInUrl.js"],"sourcesContent":["import {merge} from 'lodash';\n\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {CHARACTERS_URL, URLS_POINTS} from '@axyseo/const/analysis';\nimport {enhanceResearcherForChinese} from '@axyseo/languageProcessing/helpers/language/chineseHelperFactory';\n\n/**\n * Represents the Slug keyword assessment. This assessment checks if the keyword is present in the slug.\n */\nclass InvalidCharactersInUrl extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} config The configuration to use.\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: CHARACTERS_URL,\n docUrl:\n 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#keyword-in-url',\n fixPosition: 'url handle',\n ctaType: 'fix',\n priority: 'medium',\n title: 'Invalid characters in URL',\n content: {\n good: 'URL characters valid',\n improve: '',\n bad: 'Use only lowercase letters, numbers, and hyphens.'\n }\n };\n\n this.identifier = CHARACTERS_URL;\n this._config = merge(defaultConfig, config);\n }\n\n /**\n * Executes the Assessment and returns a result.\n *\n * @param {Paper} paper The Paper object to assess.\n * @param {Researcher} researcher The Researcher object containing all available researches.\n *\n * @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.\n */\n getResult({paper, researcher}) {\n // Enhance researcher for Chinese if needed - specifically for URL context\n const enhancedResearcher = enhanceResearcherForChinese(\n researcher,\n paper.getSlug(),\n paper.getKeyword(),\n true // isUrlContext = true for enhanced Chinese URL matching\n );\n\n this._charactersInUrl = enhancedResearcher.getResearch('charactersInUrl');\n const assessmentResult = new AssessmentResult({config: this._config});\n\n const calculatedResult = this.calculateResult();\n\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n return assessmentResult;\n }\n\n /**\n * Checks whether the paper has a keyword and a slug.\n *\n * @param {Paper} paper The paper to use for the assessment.\n * @param {Researcher} researcher The researcher object.\n *\n * @returns {boolean} True if the paper contains a keyword and a slug, and if the keywordCountInSlug research is available on the researcher.\n */\n isApplicable(paper, researcher) {\n return true;\n }\n\n /**\n *\n * @returns {{score: number, status: (string)}}\n */\n calculateResult() {\n const status = this._charactersInUrl.isInvalidCharacters ? 'bad' : 'good'\n const score = this.getScore(URLS_POINTS , status);\n\n return {\n score,\n status\n };\n }\n}\n\nexport default InvalidCharactersInUrl;\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAE5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,cAAc,EAAEC,WAAW;AACnC,SAAQC,2BAA2B;;AAEnC;AACA;AACA;AACA,MAAMC,sBAAsB,SAASL,UAAU,CAAC;EAC9C;AACF;AACA;AACA;AACA;AACA;EACEM,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAE;IACvB,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEP,cAAc;MAClBQ,MAAM,EACJ,4FAA4F;MAC9FC,WAAW,EAAE,YAAY;MACzBC,OAAO,EAAE,KAAK;MACdC,QAAQ,EAAE,QAAQ;MAClBC,KAAK,EAAE,2BAA2B;MAClCC,OAAO,EAAE;QACPC,IAAI,EAAE,sBAAsB;QAC5BC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE;MACP;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGjB,cAAc;IAChC,IAAI,CAACkB,OAAO,GAAGrB,KAAK,CAACS,aAAa,EAAED,MAAM,CAAC;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEc,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B;IACA,MAAMC,kBAAkB,GAAGpB,2BAA2B,CACpDmB,UAAU,EACVD,KAAK,CAACG,OAAO,CAAC,CAAC,EACfH,KAAK,CAACI,UAAU,CAAC,CAAC,EAClB,IAAI,CAAC;IACP,CAAC;IAED,IAAI,CAACC,gBAAgB,GAAGH,kBAAkB,CAACI,WAAW,CAAC,iBAAiB,CAAC;IACzE,MAAMC,gBAAgB,GAAG,IAAI5B,gBAAgB,CAAC;MAACM,MAAM,EAAE,IAAI,CAACa;IAAO,CAAC,CAAC;IAErE,MAAMU,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAAC,CAAC;IAE/CF,gBAAgB,CAACG,QAAQ,CAACF,gBAAgB,CAACG,KAAK,CAAC;IACjDJ,gBAAgB,CAACK,SAAS,CAACJ,gBAAgB,CAACK,MAAM,CAAC;IACnD,OAAON,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEO,YAAYA,CAACd,KAAK,EAAEC,UAAU,EAAE;IAC9B,OAAO,IAAI;EACb;;EAEA;AACF;AACA;AACA;EACEQ,eAAeA,CAAA,EAAG;IAChB,MAAMI,MAAM,GAAG,IAAI,CAACR,gBAAgB,CAACU,mBAAmB,GAAG,KAAK,GAAG,MAAM;IACzE,MAAMJ,KAAK,GAAG,IAAI,CAACK,QAAQ,CAACnC,WAAW,EAAGgC,MAAM,CAAC;IAEjD,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;AACF;AAEA,eAAe9B,sBAAsB","ignoreList":[]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { merge } from 'lodash';
|
|
2
2
|
import Assessment from "../assessment";
|
|
3
3
|
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
|
-
import { KEYPHRASE_ID } from "../../../const/analysis";
|
|
4
|
+
import { KEYPHRASE_ID, KEYWORD_POINTS } from "../../../const/analysis";
|
|
5
5
|
|
|
6
6
|
/**
|
|
7
7
|
* Represents the Slug keyword assessment. This assessment checks if the keyword is present in the slug.
|
|
@@ -21,11 +21,11 @@ class KeyphraseAssessment extends Assessment {
|
|
|
21
21
|
ctaType: 'fix',
|
|
22
22
|
fixPosition: 'keyword search',
|
|
23
23
|
priority: 'high',
|
|
24
|
-
title: '
|
|
24
|
+
title: 'Missing main keyword',
|
|
25
25
|
content: {
|
|
26
26
|
improve: '',
|
|
27
|
-
bad: '
|
|
28
|
-
good: 'Main keyword
|
|
27
|
+
bad: 'Main keyword field is empty. Add a 2-5 word primary keyword.',
|
|
28
|
+
good: 'Main keyword set'
|
|
29
29
|
}
|
|
30
30
|
};
|
|
31
31
|
this.identifier = KEYPHRASE_ID;
|
|
@@ -75,7 +75,7 @@ class KeyphraseAssessment extends Assessment {
|
|
|
75
75
|
const hasKeyphrase = paper.hasKeyword();
|
|
76
76
|
return hasKeyphrase ? 'good' : 'bad';
|
|
77
77
|
})();
|
|
78
|
-
const score = this.getScore(
|
|
78
|
+
const score = this.getScore(KEYWORD_POINTS, status);
|
|
79
79
|
return {
|
|
80
80
|
score,
|
|
81
81
|
status
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"KeyphraseAssessment.js","names":["merge","Assessment","AssessmentResult","KEYPHRASE_ID","KeyphraseAssessment","constructor","config","defaultConfig","id","docUrl","ctaType","fixPosition","priority","title","content","improve","bad","good","identifier","_config","getResult","paper","researcher","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","hasKeyphrase","hasKeyword","getScore"],"sources":["../../../../../src/scoring/assessments/seo/KeyphraseAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\n\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {KEYPHRASE_ID} from '@axyseo/const/analysis';\n\n/**\n * Represents the Slug keyword assessment. This assessment checks if the keyword is present in the slug.\n */\nclass KeyphraseAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} config The configuration to use.\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: KEYPHRASE_ID,\n docUrl:\n 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#use-of-main-keyword',\n ctaType: 'fix',\n fixPosition: 'keyword search',\n priority: 'high',\n title: '
|
|
1
|
+
{"version":3,"file":"KeyphraseAssessment.js","names":["merge","Assessment","AssessmentResult","KEYPHRASE_ID","KEYWORD_POINTS","KeyphraseAssessment","constructor","config","defaultConfig","id","docUrl","ctaType","fixPosition","priority","title","content","improve","bad","good","identifier","_config","getResult","paper","researcher","assessmentResult","calculatedResult","calculateResult","setScore","score","setStatus","status","isApplicable","hasKeyphrase","hasKeyword","getScore"],"sources":["../../../../../src/scoring/assessments/seo/KeyphraseAssessment.js"],"sourcesContent":["import {merge} from 'lodash';\n\nimport Assessment from '../assessment';\nimport AssessmentResult from '../../../values/AssessmentResult';\nimport {KEYPHRASE_ID, KEYWORD_POINTS} from '@axyseo/const/analysis';\n\n/**\n * Represents the Slug keyword assessment. This assessment checks if the keyword is present in the slug.\n */\nclass KeyphraseAssessment extends Assessment {\n /**\n * Sets the identifier and the config.\n *\n * @param {Object} config The configuration to use.\n * @returns {void}\n */\n constructor(config = {}) {\n super();\n\n const defaultConfig = {\n id: KEYPHRASE_ID,\n docUrl:\n 'https://docs.avada.io/seo-suite-help-center/seo-audit/on-page-seo/checklist#use-of-main-keyword',\n ctaType: 'fix',\n fixPosition: 'keyword search',\n priority: 'high',\n title: 'Missing main keyword',\n content: {\n improve: '',\n bad: 'Main keyword field is empty. Add a 2-5 word primary keyword.',\n good: 'Main keyword set'\n }\n };\n\n this.identifier = KEYPHRASE_ID;\n this._config = merge(defaultConfig, config);\n }\n\n /**\n * Executes the Assessment and returns a result.\n *\n * @param {Paper} paper The Paper object to assess.\n * @param {Researcher} researcher The Researcher object containing all available researches.\n *\n * @returns {AssessmentResult} The result of the assessment, containing both a score and a descriptive text.\n */\n getResult({paper, researcher}) {\n const assessmentResult = new AssessmentResult({config: this._config});\n\n const calculatedResult = this.calculateResult(paper);\n\n assessmentResult.setScore(calculatedResult.score);\n assessmentResult.setStatus(calculatedResult.status);\n return assessmentResult;\n }\n\n /**\n * Checks whether the paper has a keyword and a slug.\n *\n * @param {Paper} paper The paper to use for the assessment.\n * @param {Researcher} researcher The researcher object.\n *\n * @returns {boolean} True if the paper contains a keyword and a slug, and if the keywordCountInSlug research is available on the researcher.\n */\n isApplicable(paper, researcher) {\n return true;\n }\n\n /**\n *\n * @param paper\n * @returns {{score: number, status: (string)}}\n */\n calculateResult(paper) {\n const status = (() => {\n const hasKeyphrase = paper.hasKeyword();\n\n return hasKeyphrase ? 'good' : 'bad';\n })();\n\n const score = this.getScore(KEYWORD_POINTS, status);\n\n return {\n score,\n status\n };\n }\n}\n\nexport default KeyphraseAssessment;\n"],"mappings":"AAAA,SAAQA,KAAK,QAAO,QAAQ;AAE5B,OAAOC,UAAU;AACjB,OAAOC,gBAAgB;AACvB,SAAQC,YAAY,EAAEC,cAAc;;AAEpC;AACA;AACA;AACA,MAAMC,mBAAmB,SAASJ,UAAU,CAAC;EAC3C;AACF;AACA;AACA;AACA;AACA;EACEK,WAAWA,CAACC,MAAM,GAAG,CAAC,CAAC,EAAE;IACvB,KAAK,CAAC,CAAC;IAEP,MAAMC,aAAa,GAAG;MACpBC,EAAE,EAAEN,YAAY;MAChBO,MAAM,EACJ,iGAAiG;MACnGC,OAAO,EAAE,KAAK;MACdC,WAAW,EAAE,gBAAgB;MAC7BC,QAAQ,EAAE,MAAM;MAChBC,KAAK,EAAE,sBAAsB;MAC7BC,OAAO,EAAE;QACPC,OAAO,EAAE,EAAE;QACXC,GAAG,EAAE,8DAA8D;QACnEC,IAAI,EAAE;MACR;IACF,CAAC;IAED,IAAI,CAACC,UAAU,GAAGhB,YAAY;IAC9B,IAAI,CAACiB,OAAO,GAAGpB,KAAK,CAACQ,aAAa,EAAED,MAAM,CAAC;EAC7C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEc,SAASA,CAAC;IAACC,KAAK;IAAEC;EAAU,CAAC,EAAE;IAC7B,MAAMC,gBAAgB,GAAG,IAAItB,gBAAgB,CAAC;MAACK,MAAM,EAAE,IAAI,CAACa;IAAO,CAAC,CAAC;IAErE,MAAMK,gBAAgB,GAAG,IAAI,CAACC,eAAe,CAACJ,KAAK,CAAC;IAEpDE,gBAAgB,CAACG,QAAQ,CAACF,gBAAgB,CAACG,KAAK,CAAC;IACjDJ,gBAAgB,CAACK,SAAS,CAACJ,gBAAgB,CAACK,MAAM,CAAC;IACnD,OAAON,gBAAgB;EACzB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACEO,YAAYA,CAACT,KAAK,EAAEC,UAAU,EAAE;IAC9B,OAAO,IAAI;EACb;;EAEA;AACF;AACA;AACA;AACA;EACEG,eAAeA,CAACJ,KAAK,EAAE;IACrB,MAAMQ,MAAM,GAAG,CAAC,MAAM;MACpB,MAAME,YAAY,GAAGV,KAAK,CAACW,UAAU,CAAC,CAAC;MAEvC,OAAOD,YAAY,GAAG,MAAM,GAAG,KAAK;IACtC,CAAC,EAAE,CAAC;IAEJ,MAAMJ,KAAK,GAAG,IAAI,CAACM,QAAQ,CAAC9B,cAAc,EAAE0B,MAAM,CAAC;IAEnD,OAAO;MACLF,KAAK;MACLE;IACF,CAAC;EACH;AACF;AAEA,eAAezB,mBAAmB","ignoreList":[]}
|
|
@@ -3,7 +3,7 @@ import Assessment from "../assessment";
|
|
|
3
3
|
import AssessmentResult from "../../../values/AssessmentResult";
|
|
4
4
|
import { KeyIcon } from '@shopify/polaris-icons';
|
|
5
5
|
import { countWords } from "../../../helpers";
|
|
6
|
-
import { KEYPHRASE_LENGTH_ID } from "../../../const/analysis";
|
|
6
|
+
import { KEYPHRASE_LENGTH_ID, MAIN_CONTENT_POINTS } from "../../../const/analysis";
|
|
7
7
|
|
|
8
8
|
/**
|
|
9
9
|
* Assessment to check whether the keyphrase has a good length.
|
|
@@ -64,13 +64,11 @@ class KeyphraseLengthAssessment extends Assessment {
|
|
|
64
64
|
*/
|
|
65
65
|
calculateResult(paper) {
|
|
66
66
|
const keywordLength = countWords(paper.getKeyword());
|
|
67
|
-
let status = '';
|
|
67
|
+
let status = 'bad';
|
|
68
68
|
if (keywordLength <= 4) {
|
|
69
69
|
status = 'good';
|
|
70
|
-
} else {
|
|
71
|
-
status = 'bad';
|
|
72
70
|
}
|
|
73
|
-
const score = this.getScore(
|
|
71
|
+
const score = this.getScore(MAIN_CONTENT_POINTS, status);
|
|
74
72
|
return {
|
|
75
73
|
score,
|
|
76
74
|
status
|
|
@@ -85,7 +83,7 @@ class KeyphraseLengthAssessment extends Assessment {
|
|
|
85
83
|
* @returns {boolean} True when there is text.
|
|
86
84
|
*/
|
|
87
85
|
isApplicable(paper) {
|
|
88
|
-
return
|
|
86
|
+
return true;
|
|
89
87
|
}
|
|
90
88
|
}
|
|
91
89
|
export default KeyphraseLengthAssessment;
|