Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/StateOfJS/state-of-js-graphql-results-api
State of JS GraphQL API
https://github.com/StateOfJS/state-of-js-graphql-results-api
graphql javascript survey typescript
Last synced: about 22 hours ago
JSON representation
State of JS GraphQL API
- Host: GitHub
- URL: https://github.com/StateOfJS/state-of-js-graphql-results-api
- Owner: Devographics
- Archived: true
- Created: 2019-12-02T10:46:51.000Z (about 5 years ago)
- Default Branch: master
- Last Pushed: 2022-06-11T08:05:03.000Z (over 2 years ago)
- Last Synced: 2025-01-08T04:02:17.759Z (7 days ago)
- Topics: graphql, javascript, survey, typescript
- Language: TypeScript
- Homepage:
- Size: 4.17 MB
- Stars: 64
- Watchers: 7
- Forks: 43
- Open Issues: 31
-
Metadata Files:
- Readme: README.md
- Codeowners: .github/CODEOWNERS
Awesome Lists containing this project
- awesome-list - state-of-js-graphql-results-api
README
# StateOfJS API
- [data format](#data-format)
- [type](#type-definition)
- [example](#example-data)## Translations
You can use the [GraphQL API](http://localhost:4001/) to get more info about a specific translation locale. Here is a sample query:
```graphql
query GetLocaleData {
locale(localeId: "ru-RU") {
completion
totalCount
translatedCount
translators
untranslatedKeys
}
}
```## Data Format
### Type definition
The response are normalized using the following structure:
```typescript
// Might be extended depending on the survey we add.
type SurveyType =
| 'state_of_js'
| 'state_of_css'// Depends on the survey type, for example `css_frameworks`.
type SectionId = string// Depends on the survey type, for example `react`.
type ToolId = stringtype ToolExperience =
| 'would_use'
| 'would_not_use'
| 'interested'
| 'not_interested'
| 'never_heard'// Other tools mentioned for a given section,
// which have been normalized, for example,
// other CSS frameworks, it comes from the freeform
// field available for each major section of the survey:
// "Other options you use for the current section not mentioned in the list above".
type SectionOtherTools = {
others: NormalizedValueMultiple
}// Other tools not specific to the survey type,
// for example browsers, IDE... which can contain
// both predefined choices and an optional freeform answer
// which get normalized.
type OtherTools = {
choices: string[]
others?: NormalizedValueMultiple
}// Depends on the survey type, for example `flexbox`.
type FeatureId = stringtype FeatureExperience =
| 'never_heard'
| 'heard'
| 'used'type ResourceType =
| 'blogs_news_magazines'
| 'podcasts'
| 'sites_courses'// Depends on the survey type, for example `css_easy_to_learn`.
type OpinionId = string// Date string in ISO format (ISO 8601).
type DateString = string// Generic rating.
type Rating = 0 | 1 | 2 | 3 | 4// Used to rate opinions, for example:
// "CSS is a programming language".
// Goes from strong disagreement to strong agreement.
type AgreementRating = Rating// Used to rate happiness, for example:
// "How happy are you with the general state of CSS?".
// Goes from very unhappy to very happy.
type HappinessRating = Rating// Used to represent the normalization of free form
// fields, when a single normalized result is expected,
// used for example for `user_info.source`.
type NormalizedValueUnique = {
// Raw input
raw: string
// String representation of the regexp which led
// to the normalization.
pattern: string
// Normalized value which matched `pattern`.
normalized: string
}type NormalizedValueMultiple = {
// Raw input
raw: string
// String representations of the regexps which led
// to the normalizations.
patterns: string[]
// Normalized values which matched `pattern`,
// match `patterns` order, meaning `normalized[1]`
// was generated using `patterns[1]`.
normalized: string[]
}type WorkExperienceRange =
| 'range_less_than_1'
| 'range_1_2'
| 'range_2_5'
| 'range_5_10'
| 'range_10_20'
| 'range_more_than_20'type CompanySizeRange =
| 'range_1'
| 'range_1_5'
| 'range_5_10'
| 'range_10_20'
| 'range_20_50'
| 'range_50_100'
| 'range_100_1000'
| 'range_more_than_1000'type SalaryRange =
| 'range_work_for_free'
| 'range_0_10'
| 'range_10_30'
| 'range_30_50'
| 'range_50_100'
| 'range_100_200'
| 'range_more_than_200'type Gender =
| 'male'
| 'female'
| 'non_binary'
| 'prefer_not_to_say'// Might change depending on the surveys we make.
type JobTitle =
| 'cto'
| 'front_end_developer'
| 'full_stack_developer'
| 'back_end_developer'
| 'web_developer'
| 'web_designer'
| 'ui_designer'
| 'ux_designer'type RaceEthnicityId =
| 'white_european'
| 'hispanic_latin'
| 'east_asian'
| 'middle_eastern'
| 'black_african'
| 'multiracial'
| 'biracial'
| 'native_american_islander_australian'// Proficiency rating for a specific domain,
// from completely inexperienced to expert.
type Proficiency = Ratingtype EnvironmentId = string
interface NormalizedResponse {
// Auto generated ID.
_id: number
// Reference `users` collection,
// only available for survey made in the Vulcan survey system.
userId?: number
// Reference non-normalized `responses` collection,
// only available for survey made in the Vulcan survey system.
responseId?: number
survey: SurveyType
// Year the response belongs too
year: number
// Concatenation of `survey` + `year`
surveySlug: string
// Creation and update dates, can be used to compute
// time spent filling out the survey.
createdAt: DateString
updatedAt: DateString
// Date when the response was normalized.
generatedAt: DateString
user_info: {
os: string
device: string
browser: string
version: string
browser_type: string
user_agent: string
platform: string
// How participants found the survey,
// it's a freeform field which is normalized.
source: NormalizedValueUnique
referrer: string
job_title: JobTitle
years_of_experience: WorkExperienceRange
company_size: CompanySizeRange
yearly_salary: SalaryRange
gender: Gender
skin_tone: number
country_name: string
country_alpha3: string
backend_proficiency: Proficiency
javascript_proficiency: Proficiency
css_proficiency: Proficiency
race_ethnicity: {
choices: [RaceEthnicityId]
}
// Used to identify users, especially useful when
// we want to cross responses for surveys filled out
// on a different platform (typically TypeForm),
// where we don't have a corresponding entry
// in the `users` collection.
hash: string
}
tools: Record
tools_others: Record<
SectionId | string,
SectionOtherTools | OtherTools
>
features: Record
// Contains lists of features grouped by
// specific topics, for example, CSS units.
// Participants are asked to select the
// features they used from a predefined list
// of options.
features_others: Record
// Opinions on a specific topic, expressed as a rating.
opinions: Record
// Opinions on a specific topic, freeform field, for example:
// "What do you feel is currently missing from CSS?"
// value is normalized.
opinions_others: Record
// How happy participants are about certain topics,
// can be both tied to sections or generic.
happiness: Record
// Resources relevant to the survey type such as blogs, courses...
resources: Record
// Contain answers related to environments,
// for example, browsers, form factors...
// It has both choices and rating based answers.
environments: Record
}
```### Example data
Here's an example of a sample survey response for the 2020 State of CSS survey:
```js
{
"_id": "123foobar",
"surveySlug": "css2020",
"createdAt": "2020-10-13T21:32:25.571Z",
"updatedAt": "2020-10-13T21:50:08.693Z",
"year": 2020,
"completion": 89,
"userId": "456barbaz",
"responseId": "789foobaz",
"generatedAt": "2020-10-13T21:53:55.219Z",
"survey": "css",
"user_info": {
"device": "desktop",
"browser": "Chrome",
"version": "86.0.4240.75",
"os": "macOS",
"referrer": "",
"backend_proficiency": 3,
"company_size": "range_1",
"css_proficiency": 4,
"gender": "male",
"how_did_user_find_out_about_the_survey": "I made it!",
"javascript_proficiency": 4,
"job_title": "full_stack_developer",
"race_ethnicity": {
"choices": [
"white_european"
]
},
"yearly_salary": "range_10_30",
"years_of_experience": "range_10_20",
"country": "JP",
"email_hash": "63bea61eeda07586ced5fc60400672b2b61b06986445a1de665a827a5456d519",
"country_name": "Japan",
"country_alpha3": "JPN"
},
"features": {
"aspect_ratio": {
"experience": "never_heard"
},
"content_visibility": {
"experience": "never_heard"
},
"exclusions": {
"experience": "never_heard"
},
"flexbox": {
"experience": "used"
}
},
"features_others": {
"attributes": {
"choices": [
"presence",
"equality"
]
},
"combinators": {
"choices": [
"descendant",
"child",
"next_sibling"
]
}
},
"happiness": {
"pre_post_processors": 3,
"css_frameworks": 3,
"css_methodologies": 2,
"css_in_js": 2,
"state_of_css": 3,
"state_of_the_web": 3
},
"tools": {
"less": {
"experience": "not_interested"
},
"post_css": {
"experience": "would_use"
},
"sass": {
"experience": "would_use"
},
"stylus": {
"experience": "not_interested"
}
},
"tools_others": {
"css_frameworks": {
"others": {
"raw": "angular native, react native",
"normalized": [
"angular_native",
"reactnative"
],
"patterns": [
"/angular( |-)?native/i",
"/react( |-|_)?native/i"
]
}
},
"utilities": {
"choices": [
"stylelint",
"purge_css"
],
"others": {
"raw": "some other utility",
"normalized": [],
"patterns": []
}
},
"text_editors": {
"others": {
"raw": "angular native",
"normalized": [
"angular_native"
],
"patterns": [
"/angular( |-)?native/i"
]
}
}
},
"environments": {
"browsers": {
"choices": [
"chrome",
"safari",
"firefox",
"safari_ios"
]
},
"css_for_email": 1,
"css_for_print": 0,
"form_factors": {
"choices": [
"desktop",
"smartphone"
]
}
},
"resources": {
"blogs_news_magazines": {
"choices": [
"css_tricks",
"codrops",
"sidebar",
"heydesigner"
],
"others": {
"raw": "some other blog",
"normalized": [],
"patterns": []
}
},
"podcasts": {
"choices": [
"shop_talk_show",
"syntaxfm"
]
},
"sites_courses": {
"choices": [
"stack_overflow",
"mdn"
]
}
},
"opinions": {
"css_easy_to_learn": 1,
"css_evolving_slowly": 1,
"css_is_programming_language": 4,
"enjoy_writing_css": 2,
"selector_nesting_to_be_avoided": 3,
"utility_classes_to_be_avoided": 1
},
"opinions_other": {
"currently_missing_from_css": {
"others": {
"raw": "Container queries",
"normalized": [
"container_queries"
],
"patterns": [
"/container ?queries/i"
]
}
}
}
}
```